Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

OpenTabletDriver를 (관리자 권한으로 실행하지 않고도) 관리자 권한의 프로그램에서 동작하게 만드는 방법

지난 글에서,

(WACOM도 지원하는) Tablet 공통 디바이스 드라이버 - OpenTabletDriver
; https://www.sysnet.pe.kr/2/0/12632

OpenTabletDriver 프로그램이 관리자 권한의 프로그램이 실행됐을 때는, 즉 UAC 권한 상승 창이 뜨는 프로그램에서는 태블릿 입력이 안 된다는 문제를 설명했고 그것의 원인이 UIPI(User Interface Privilege Isolation) 제약이라고 했습니다. 그런데, 사실 이것은 UAC 권한 상승 창이 뜨지 않는 프로그램에서도 발생할 수 있습니다. 왜냐하면, 마이크로소프트의 경우 자사의 인증서로 서명해 특별 관리를 하는 프로그램에 한해 "UAC 권한 상승 창"이 뜨지 않으면서도 관리자 권한을 갖는 프로그램들을 제공하기 때문입니다. 대표적으로 "작업 관리자(Task Manager)"를 비롯해 각종 MMC(Microsoft Management Console) 기반의 관리 프로그램들이 그것입니다.

따라서, OpenTabletDriver을 관리자 권한으로 실행하지 않으면 여간 불편한 것이 아닙니다.

그래도 그나마 희망이 있다면, 마이크로소프트가 일반 권한의 프로그램에서도 UIPI 제약을 벗어날 수 있는 방법을 제공하고 있으며 이에 대해서는 아래의 글에서 이미 자세하게 설명했습니다.

UAC - 관리자 권한 없이 UIPI 제약을 없애는 방법
; https://www.sysnet.pe.kr/2/0/12633

OpenTabletDriver에 저 방법을 적용하기 위한 가장 큰 장애물은 바로 app.manifest에 대한 처리입니다. 이 파일은 응용 프로그램 빌드 단계에서 제공되어야 하기 때문에 반드시 소스 코드를 빌드할 수 있는 사람만이 이것을 처리할 수 있습니다.

따라서, 원래는 OpenTabletDriver 프로그램을 배포하는 사람에게 저 작업을 부탁해야 하는데요, 다행히 OpenTabletDriver는 Open Source로 공개된 것이기에,

OpenTabletDriver/OpenTabletDriver
; https://github.com/OpenTabletDriver/OpenTabletDriver

우리도 저것을 git clone해서 직접 빌드하는 식으로 처리하면 됩니다.




자, 그럼 단계별로 한 번 해볼까요? ^^

당연히 git clone을 해 소스 코드를 다운로드하고,

c:\temp> git clone https://github.com/OpenTabletDriver/OpenTabletDriver.git

비주얼 스튜디오로 로드해 전체 빌드를 합니다. (경고 하나 없이 아주 깨끗하게 빌드가 됩니다. ^^)

그다음, .\OpenTabletDriver\Configurations 디렉터리를 .\OpenTabletDriver.Daemon\bin\Debug 하위에 모두 복사합니다. (또는, 자신에게 해당하는 태블릿 설정이 있는 파일과 그것의 부모 디렉터리 구조만 복사해도 됩니다.)

이후 OpenTabletDriver.Daemon.exe를 실행하면 다음과 같은 출력이 나옵니다.

[Detect:Info]   Searching for tablet 'Wacom CTL-460'
[Detect:Debug]  Using device 'CTL-460'.
[Detect:Debug]  Using report parser type 'OpenTabletDriver.Tablet.TabletReportParser'.
[Detect:Debug]  Device path: \\?\hid#vid_056a&pid_00d4&mi_00&col02#7&131af743&0&0001#{4d1e55b2-f16f-11cf-88cb-001111000030}
[Device:Debug]  Set tablet feature: 02-02
[Detect:Info]   Found tablet 'Wacom CTL-460'
[Detect:Warning]        Failed to find auxiliary device, express keys may be unavailable.
[Display:Debug] DPI Awareness enabled
[Settings:Info] Output mode: Absolute Mode
[Settings:Info] Display area: [5760x1080@<2880, 540>:0°],
[Settings:Info] Tablet area: [147.2x92@<73.6, 46>:180°],
[Settings:Info] Clipping: Enabled
[Settings:Info] Ignoring reports outside area: Disabled
[Settings:Info] Tip Binding: [Mouse Button Binding: Left]@1%
[Settings:Info] Eraser Binding: []@0%
[Settings:Info] Pen Bindings: [0, Mouse Button Binding: Left], [1, Mouse Button Binding: Right]
[Settings:Info] Express Key Bindings: [0, ], [1, ], [2, ], [3, ], [4, ], [5, ], [6, ], [7, ]
[Settings:Info] Driver is auto-enabled.

이렇게 Daemon을 실행해 둔 상태에서 ".\OpenTabletDriver.UX.Wpf\bin\Debug\OpenTabletDriver.UX.Wpf.exe" 프로그램을 실행하면 됩니다. 이후, 설정을 하고 플러그인도 설치를 한 후 설정값을 저장하고 종료합니다.




프로젝트를 빌드해 보고 알았지만, 사실상 사용자 모드 device driver를 설치하는 작업은 OpenTabletDriver.Daemon.exe 프로세스입니다. 따라서 그것만 실행해도 됩니다. OpenTabletDriver.UX.Wpf.exe는 단지 User Interface를 제공해 플러그인을 설치하거나 값을 설정할 수 있는 편의 환경만 제공하는데 의미가 있습니다.

이제 아래의 글에서 설명했던,

UAC - 관리자 권한 없이 UIPI 제약을 없애는 방법
; https://www.sysnet.pe.kr/2/0/12633

  1. "Application Manifest File" 파일 추가 후, uiAccess를 true로 설정
  2. 인증서 생성(하고 로컬에 등록) 및 EXE 파일 서명
  3. C:\Program Files\OpenTabletDriver 폴더를 생성하고 그 하위에 .\OpenTabletDriver.Daemon\bin\Debug 빌드 결과물 모두 복사

작업들을 수행하면 됩니다. 그리고 그 작업은 OpenTabletDriver.Daemon 프로젝트에 대해서만 하면 되고, 모든 처리를 완료했으면 이제 OpenTabletDriver.Daemon.exe를 일반 권한으로 수행해도 작업 관리자 등의 프로그램에서 정상적으로 동작하는 것을 확인할 수 있습니다.




이렇게 일반 권한으로 수행해도 되기 때문에 해당 프로그램의 단축 아이콘을 생성해 "%APPDATA%\Microsoft\Windows\Start Menu\Programs\Startup" 폴더에 놓으면 됩니다. 그럼 윈도우가 부팅되어 로그인했을 때 자연스럽게 "Autorun"이 되기 때문에 별다른 수작업을 하지 않아도 태블릿 사용을 할 수 있게 됩니다.

자, 그럼 지난 글에서 이야기했던 불편한 점 3가지 중 2번과 3번이 해결되었습니다. 그럼, 기왕에 소스 코드가 있으니 (최대 절전 모드에서 복원했을 때 태블릿이 동작을 하지 않는) 1번도 해결해 볼까요? ^^

방법은 간단하게, 윈도우 운영체제가 Resume되었을 때 다시 태블릿을 연결하도록 하면 되는데, 대충 분석을 해보면 OpenTabletDriver.Daemon 프로젝트의 DriverDaemon.cs 파일에 이런 식으로 코드를 추가하면 됩니다.

// ...[생략]...
public class DriverDaemon : IDriverDaemon
{
    public DriverDaemon()
    {
        Log.Output += (sender, message) =>
        {
            LogMessages.Add(message);
            Console.WriteLine(Log.GetStringFormat(message));
            Message?.Invoke(sender, message);
        };
        Driver.TabletChanged += (sender, tablet) =>
        {
            TabletChanged?.Invoke(sender, tablet);
            if (debugging)
            {
                if (Driver.TabletReader != null)
                    Driver.TabletReader.Report += DebugReportHandler;
                if (Driver.AuxReader != null)
                    Driver.AuxReader.Report += DebugReportHandler;
            }
        };
        Driver.DevicesChanged += async (sender, args) =>
        {
            if (await GetTablet() == null && args.Additions.Count() > 0)
                await DetectTablets();
        };

        // Install-Package Microsoft.Win32.SystemEvents
        if (OperatingSystem.IsWindows() == true)
        {
#pragma warning disable CA1416 // Validate platform compatibility
            SystemEvents.PowerModeChanged += async (object sender, PowerModeChangedEventArgs e) =>
            {
                if (e.Mode == PowerModes.Resume)
                {
                    await DetectTablets();
                }
            };
#pragma warning restore CA1416 // Validate platform compatibility
        }

        LoadUserSettings();
    }

    // ...[생략]...
}

이렇게 변경하고 다시 서명, "%ProgramFiles%\OpenTabletDriver"에 복사하면 끝! ^^

이걸로 모든 단점은 해결되었지만, 그래도 유일하게 고칠 수 없는 딱 한 가지가 있습니다. 바로, 윈도우의 그림 암호(Picture Password) 화면에서는 태블릿 인식이 안 된다는 점입니다. 어쩔 수 없습니다, 이게 꼭 필요하다면 OpenTabletDriver를 포기하고 태블릿 제조사와 함께 제공되는 device driver를 그대로 사용해야 합니다.




참고로, 0.5.3.1 버전의 "OpenTabletDriver.win-x64.zip" 파일의 압축을 해제한 Configurations 폴더의 json 파일 스키마가 현재(2021-04-23) Opentablet 솔루션에서는 바뀌었으므로 주의해야 합니다. 만약 github 프로젝트로 지난 버전의 json 파일을 로드하면 다음과 같은 식의 오류가 발생합니다.

Newtonsoft.Json.JsonSerializationException
  HResult=0x80131500
  Message=Error converting value "AgI=" to type 'System.Collections.Generic.List`1[System.Byte[]]'. Path 'DigitizerIdentifiers[1].FeatureInitReport', line 43, position 33.
  Source=Newtonsoft.Json
  StackTrace:
   at Newtonsoft.Json.Serialization.JsonSerializerInternalReader.EnsureType(JsonReader reader, Object value, CultureInfo culture, JsonContract contract, Type targetType)

  This exception was originally thrown at this call stack:
    Newtonsoft.Json.Utilities.ConvertUtils.EnsureTypeAssignable(object, System.Type, System.Type)
    Newtonsoft.Json.Utilities.ConvertUtils.ConvertOrCast(object, System.Globalization.CultureInfo, System.Type)
    Newtonsoft.Json.Serialization.JsonSerializerInternalReader.EnsureType(Newtonsoft.Json.JsonReader, object, System.Globalization.CultureInfo, Newtonsoft.Json.Serialization.JsonContract, System.Type)

Inner Exception 1:
ArgumentException: Could not cast or convert from System.String to System.Collections.Generic.List`1[System.Byte[]].

오류가 발생한 json은,

"VendorID": 1386,
"ProductID": 212,
"InputReportLength": 9,
"OutputReportLength": null,
"ReportParser": "OpenTabletDriver.Tablet.TabletReportParser",
"FeatureInitReport": "AgI=",
"OutputInitReport": null,
"DeviceStrings": {},
"InitializationStrings": []"

다음과 같이 바뀌었고,

"VendorID": 1386,
"ProductID": 212,
"InputReportLength": 11,
"OutputReportLength": null,
"ReportParser": "OpenTabletDriver.Vendors.SkipByteTabletReportParser",
"FeatureInitReport": [
    "AgI="
],
"OutputInitReport": null,
"DeviceStrings": {},
"InitializationStrings": []

이외에도 많은 것이 변경되었으므로 전체적으로 OpenTabletDriver.Daemon 프로그램의 초기화가 진행되지 않습니다. 또한, 0.5.3.1 버전의 Plugin 들과 현재 github 버전의 소스 코드는 호환이 안 되어서 OpenTabletDriver.Daemon.exe 실행 시에 다음과 같은 (경고라고 볼 수 없는) 오류 로그가 나옵니다.

[Plugin:Debug]  Loading plugin 'HawkuFilters'
[Plugin:Warning]        Plugin 'HawkuFilters, Version=1.0.0.0' can't be loaded and is likely out of date.

메시지가 의미하는 바에 따라 HawkuFilters.dll을 (로드는 했으나) 타입 생성을 하지 못해 사용하지 않습니다. 정상적이라면 그냥 다음과 같이 로그가 나와야 합니다.

[Plugin:Debug]  Loading plugin 'HawkuFilters'
[Detect:Info]   Searching for tablet 'Wacom CTL-460'

따라서, 만약 v0.5.3.1의 Configurations 파일을 그대로 사용하고 싶다면 github의 소스 코드를 "v0.5.3.1"로 태그되어 있는 버전으로 돌려서 작업을 해야 합니다.

opentabletdriver_uipi_1.png




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 5/9/2021]

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

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...
NoWriterDateCnt.TitleFile(s)
12145정성태2/18/202010840.NET Framework: 892. eBEST C# XingAPI 래퍼 - Sqlite 지원 추가파일 다운로드1
12144정성태2/13/202010881.NET Framework: 891. 실행 시에 메서드 가로채기 - CLR Injection: Runtime Method Replacer 개선 - 두 번째 이야기파일 다운로드1
12143정성태2/13/20208907.NET Framework: 890. 상황별 GetFunctionPointer 반환값 정리 - x64파일 다운로드1
12142정성태2/12/202010753.NET Framework: 889. C# 코드로 접근하는 MethodDesc, MethodTable파일 다운로드1
12141정성태2/10/202010324.NET Framework: 888. C# - ASP.NET Core 웹 응용 프로그램의 출력 가로채기 [2]파일 다운로드1
12140정성태2/10/202010197.NET Framework: 887. C# - ASP.NET 웹 응용 프로그램의 출력 가로채기파일 다운로드1
12139정성태2/9/202011556.NET Framework: 886. C# - Console 응용 프로그램에서 UI 스레드 구현 방법
12138정성태2/9/202014324.NET Framework: 885. C# - 닷넷 응용 프로그램에서 SQLite 사용 [6]파일 다운로드1
12137정성태2/9/20209437오류 유형: 592. [AhnLab] 경고 - 디버거 실행을 탐지했습니다.
12136정성태2/6/20209873Windows: 168. Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
12135정성태2/6/202013554개발 환경 구성: 468. Nuget 패키지의 로컬 보관 폴더를 옮기는 방법 [2]
12134정성태2/5/202013626.NET Framework: 884. eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지 [5]파일 다운로드1
12133정성태2/5/202010855디버깅 기술: 161. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 - 두 번째 이야기
12132정성태1/28/202012430.NET Framework: 883. C#으로 구현하는 Win32 API 후킹(예: Sleep 호출 가로채기)파일 다운로드1
12131정성태1/27/202012496개발 환경 구성: 467. LocaleEmulator를 이용해 유니코드를 지원하지 않는(한글이 깨지는) 프로그램을 실행하는 방법 [1]
12130정성태1/26/20209995VS.NET IDE: 142. Visual Studio에서 windbg의 "Open Executable..."처럼 EXE를 직접 열어 디버깅을 시작하는 방법
12129정성태1/26/202015525.NET Framework: 882. C# - 키움 Open API+ 사용 시 Registry 등록 없이 KHOpenAPI.ocx 사용하는 방법 [3]
12128정성태1/26/202010344오류 유형: 591. The code execution cannot proceed because mfc100.dll was not found. Reinstalling the program may fix this problem.
12127정성태1/25/202010193.NET Framework: 881. C# DLL에서 제공하는 Win32 export 함수의 내부 동작 방식(VT Fix up Table)파일 다운로드1
12126정성태1/25/202011015.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석파일 다운로드1
12125정성태1/24/20208846VS.NET IDE: 141. IDE0019 - Use pattern matching
12124정성태1/23/202010666VS.NET IDE: 140. IDE1006 - Naming rule violation: These words must begin with upper case characters: ...
12123정성태1/23/202012173웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/20209121.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/20209716VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202011152.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...