Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

USB/IP PROJECT를 이용해 C#으로 USB Keyboard 가상 장치 만들기

지난 글에서,

(코드로 가상 USB 장치를 만들 수 있는) USB/IP PROJECT 소개
; https://www.sysnet.pe.kr/2/0/12213

USB/IP PROJECT에 버그가 있어 가상 장치를 detach 시 BSOD가 발생한다고 했는데요, 현재 해당 프로젝트는 종료되었으므로 더 이상 업데이트를 기대할 수 없지만 그 바통을 이어받은 프로젝트가 다행히 아직 살아 있습니다. ^^

cezanne / usbip-win
; https://github.com/cezanne/usbip-win

USB/IP for Windows
; https://github.com/cezanne/usbip-win/blob/master/README.md

(게다가 저 repo의 주인장인 cezanne라는 분은 한국인입니다. ^^)

따라서, 위의 repo에 있는 device driver를 빌드한 usbip_vhci.sys를 윈도우 10에 설치하면 BSOD 없는 안정적인 "USB/IP VHCI" 장치를 사용할 수 있습니다.




자, 그럼 usbip_vhci.sys에서 나열해 줄 가상 장치 역할을 할 클라이언트 코드가 필요한데 이것도 이미 전의 글에서 소개한,

lcgamboa / USBIP-Virtual-USB-Device
; https://github.com/lcgamboa/USBIP-Virtual-USB-Device

repo에 C 언어와 파이썬으로 구현한 코드를 함께 제공하고 있으니, C#에서는 그저 해당 기능들을 그대로 포팅만 하면 됩니다. 아래의 프로젝트는 그래서 만들어진 것이고,

USBIP-Virtual-USB-Device/dotnet/UsbipDevice/
; https://github.com/stjeong/USBIP-Virtual-USB-Device/tree/master/dotnet/UsbipDevice

위의 라이브러리를 바탕으로 예전에 Raspberry pi zero로 구현해 두었던 키보드 제어 방식을,

stjeong / rasp_vusb
; https://github.com/stjeong/rasp_vusb

동일하게 구현한 예제 프로젝트를 함께 실어 두었습니다.

USBIP-Virtual-USB-Device/dotnet/cs-hid-keyboard/
; https://github.com/stjeong/USBIP-Virtual-USB-Device/tree/master/dotnet/cs-hid-keyboard

사용 방법은 아래의 코드를 보면 대충 눈치채실 수 있을 것입니다. ^^

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Threading;
using UsbipDevice;

namespace cs_hid_keyboard
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("cs-hid-keyboard");
            bool waitLocalHost = true;

            if (args.Length >= 1)
            {
                if (args[0] == "-w")
                {
                    waitLocalHost = false;
                }
            }

            byte[] reportBuffer = KeyboardDescriptors.Report;
            using (Usbip device = new Usbip(UsbDescriptors.Device, KeyboardDescriptors.Hid, reportBuffer))
            {
                KeyboardDevice keyboard = new KeyboardDevice(device, reportBuffer);

                device.Run();

                if (waitLocalHost == true)
                {
                    Thread usbipServer = new Thread(usbipServer_Run);
                    usbipServer.IsBackground = true;
                    usbipServer.Start();
                }

                KeyboardTest(keyboard);
            }
        }

        private static void usbipServer_Run(object obj)
        {
            foreach (Process process in Process.GetProcessesByName("usbip"))
            {
                process.Kill();
            }

            // usbip attach -r 127.0.0.1 -b 1-1
            Process.Start("usbip", "attach -r 127.0.0.1 -b 1-1");
        }

        private static void KeyboardTest(KeyboardDevice keyboard)
        {
            //{
            //    string txt = "abc+*()xptmxm";
            //    _usbController.SendText(txt + Environment.NewLine);
            //}

            //{
            //    string txt = "abc";
            //    _usbController.SendText(txt + Environment.NewLine);
            //}

            //{
            //    string txt = "";
            //    _usbController.SendText(txt);
            //}

            //{
            //    string txt = "test is good";
            //    _usbController.SendText(txt);
            //}

            //{
            //    string txt = "";
            //    _usbController.SendText(txt);
            //}

            Console.WriteLine("Wait for usbip...");
            while (true)
            {
                Console.Write(".");

                if (keyboard.Connected == true)
                {
                    break;
                }

                Thread.Sleep(1000);
            }

            while (true)
            {
                Console.Write("Keyboard> ");
                string text = Console.ReadLine();

                Thread.Sleep(2000);

                if (text == "quit")
                {
                    keyboard.Dispose();
                    break;
                }

                keyboard.SendText(text);
            }
        }
    }
}

빌드하고 실행하면, "장치 관리자"에 새로운 "HID Keyboard Device" 장치가 생기고 "keyboard.SendText(text);"에 넣는 글자에 따라 키 입력이 그대로 되는 것을 확인할 수 있습니다. (이제 더 이상 "Raspberry PI Zero"가 없어도 소프트웨어적으로 완벽하게 제어할 수 있는 USB 가상 키보드를 얻게 되었습니다. ^^)




참고로, "cezanne / usbip-win" 소스 코드를 빌드하면 당연히 테스트 인증서로 서명되므로 윈도우 10의 testsigning 옵션을 켜야 합니다. 그런 후 "usbip.exe"로 다음과 같이 실행해 주면,

c:\temp> usbip install

설치가 됩니다. (물론, INF 파일을 이용해 장치 관리자에서 설치해도 됩니다.)

마지막으로, 제가 가지고 있던 책에서 관련 자료가 나오는데,

윈도우즈 드라이버 모델 : Writing Windows WDM Device Drivers
; http://www.yes24.com/Product/goods/53240347

Test 6
     Kbd report 1 0  0 0 0 0 0 0 Ctrl
     Kbd report 5 0  0 0 0 0 0 0 Ctrl+Alt
     Kbd report 5 0 63 0 0 0 0 0 Ctrl+Alt+Del
     Kbd report 4 0  0 0 0 0 0 0 Alt
     Kbd report 0 0  4 0 0 0 0 0 A
     Kbd report 0 0  5 0 0 0 0 0 B
     Kbd report 0 0  6 0 0 0 0 0 C
     Kbd report 0 0 29 0 0 0 0 0 Esc

위에서 보면, 제어 키는 0번, 문자 키는 2번 바이트에 할당됩니다. 즉, keyboard.SendText는 문자에 맞는 값으로 저 바이트 배열을 보내기만 하면 되는 것입니다. (실제로는 Report id 값도 보내기 때문에 약간 다르긴 합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]


donaricano-btn



[최초 등록일: ]
[최종 수정일: 5/16/2020 ]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer@outlook.com

비밀번호

댓글 쓴 사람
 




1  2  [3]  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
12443정성태12/8/2020205.NET Framework: 977. C# PInvoke - C++의 매개변수에 대한 마샬링을 tlbexp.exe를 이용해 확인하는 방법
12442정성태12/4/2020237오류 유형: 691. Visual Studio - Build Events에 robocopy를 사용할때 "Invalid Parameter #1" 오류가 발행하는 경우
12441정성태12/4/2020199오류 유형: 690. robocopy - ERROR : No Destination Directory Specified.
12440정성태12/4/2020186오류 유형: 689. SignTool Error: Invalid option: /as
12439정성태12/4/2020251디버깅 기술: 176. windbg - 특정 Win32 API에서 BP가 안 걸리는 경우 (2)
12438정성태12/2/2020299오류 유형: 688. .Visual C++ - Error C2011 'sockaddr': 'struct' type redefinition
12437정성태12/1/2020283VS.NET IDE: 155. pfx의 암호 키 파일을 Visual Studio 없이 등록하는 방법
12436정성태12/1/2020365오류 유형: 687. .NET Core 2.2 빌드 - error MSB4018: The "RazorTagHelper" task failed unexpectedly.
12435정성태12/11/20201512Windows: 181. 윈도우 환경에서 클라이언트 소켓의 최대 접속 수 (4) - ReuseUnicastPort를 이용한 포트 고갈 문제 해결파일 다운로드1
12434정성태12/17/2020337Windows: 180. C# - dynamicport 값의 범위를 알아내는 방법
12433정성태12/1/2020361Windows: 179. 윈도우 환경에서 클라이언트 소켓의 최대 접속 수 (3) - SO_PORT_SCALABILITY파일 다운로드1
12432정성태12/14/2020493Windows: 178. 윈도우 환경에서 클라이언트 소켓의 최대 접속 수 (2) - SO_REUSEADDR [1]파일 다운로드1
12431정성태11/27/2020344.NET Framework: 976. UnmanagedCallersOnly + C# 9.0 함수 포인터 사용 시 x86 빌드에서 오동작하는 문제파일 다운로드1
12430정성태11/27/2020228오류 유형: 686. Ubuntu - E: The repository 'cdrom://...' does not have a Release file.
12429정성태12/2/2020267디버깅 기술: 175. windbg - 특정 Win32 API에서 BP가 안 걸리는 경우
12428정성태11/25/2020226VS.NET IDE: 154. Visual Studio - .NET Core App 실행 시 dotnet.exe 실행 화면만 나오는 문제
12427정성태11/25/2020365.NET Framework: 975. .NET Core를 직접 호스팅해 (runtimeconfig.json 없이) EXE만 배포해 실행파일 다운로드1
12426정성태11/24/2020229오류 유형: 685. WinDbg Preview - error InitTypeRead
12425정성태11/24/2020278VC++: 141. Visual C++ - "Treat Warnings As Errors" 옵션이 꺼져 있는데도 일부 경고가 에러 처리되는 경우
12424정성태11/24/2020326VC++: 140. C++의 연산자 동의어(operator synonyms), 대체 토큰
12423정성태11/22/2020317.NET Framework: 974. C# 9.0 - (16) 제약 조건이 없는 형식 매개변수 주석(Unconstrained type parameter annotations)파일 다운로드1
12422정성태11/21/2020275.NET Framework: 973. .NET 5, .NET Framework에서만 허용하는 UnmanagedCallersOnly 사용예파일 다운로드1
12421정성태11/23/2020274.NET Framework: 972. DNNE가 출력한 NE DLL을 직접 생성하는 방법파일 다운로드1
12420정성태11/19/2020222오류 유형: 684. Visual C++ - MSIL .netmodule or module compiled with /GL found; restarting link with /LTCG; add /LTCG to the link command line to improve linker performance
12419정성태11/23/2020326VC++: 139. Visual C++ - .NET Core의 nethost.lib와 정적 링크파일 다운로드1
12418정성태11/19/2020295오류 유형: 683. Visual C++ - error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MT_StaticRelease' doesn't match value 'MDd_DynamicDebug'파일 다운로드1
1  2  [3]  4  5  6  7  8  9  10  11  12  13  14  15  ...