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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  [37]  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12714정성태7/16/20218272오류 유형: 735. VCRUNTIME140.dll, MSVCP140.dll, VCRUNTIME140.dll, VCRUNTIME140_1.dll이 없어 exe 실행이 안 되는 경우
12713정성태7/16/20218794.NET Framework: 1077. C# - 동기 방식이면서 비동기 규약을 따르게 만드는 Task.FromResult파일 다운로드1
12712정성태7/15/20218234개발 환경 구성: 579. Azure - 리눅스 호스팅의 Site Extension 제작 방법
12711정성태7/15/20218605개발 환경 구성: 578. Azure - Java Web App Service를 위한 Site Extension 제작 방법
12710정성태7/15/202110393개발 환경 구성: 577. MQTT - emqx.io 서비스 소개
12709정성태7/14/20216983Linux: 42. 실행 중인 docker 컨테이너에 대한 구동 시점의 docker run 명령어를 확인하는 방법
12708정성태7/14/202110391Linux: 41. 리눅스 환경에서 디스크 용량 부족 시 원인 분석 방법
12707정성태7/14/202177663오류 유형: 734. MySQL - Authentication method 'caching_sha2_password' not supported by any of the available plugins.
12706정성태7/14/20218823.NET Framework: 1076. C# - AsyncLocal 기능을 CallContext만으로 구현하는 방법 [2]파일 다운로드1
12705정성태7/13/20219002VS.NET IDE: 168. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 - 두 번째 이야기
12704정성태7/12/20218137개발 환경 구성: 576. Azure VM의 서비스를 Azure Web App Service에서만 접근하도록 NSG 설정을 제한하는 방법
12703정성태7/11/202113802개발 환경 구성: 575. Azure VM에 (ICMP) ping을 허용하는 방법
12702정성태7/11/20218917오류 유형: 733. TaskScheduler에 등록된 wacs.exe의 Let's Encrypt 인증서 업데이트 문제
12701정성태7/9/20218587.NET Framework: 1075. C# - ThreadPool의 스레드는 반환 시 ThreadStatic과 AsyncLocal 값이 초기화 될까요?파일 다운로드1
12700정성태7/8/20218963.NET Framework: 1074. RuntimeType의 메모리 누수? [1]
12699정성태7/8/20217759VS.NET IDE: 167. Visual Studio 디버깅 중 GC Heap 상태를 보여주는 "Show Diagnostic Tools" 메뉴 사용법
12698정성태7/7/202111712오류 유형: 732. Windows 11 업데이트 시 3% 또는 0%에서 다운로드가 멈춘 경우
12697정성태7/7/20217580개발 환경 구성: 574. Windows 11 (Insider Preview) 설치하는 방법
12696정성태7/6/20218193VC++: 146. 운영체제의 스레드 문맥 교환(Context Switch)을 유사하게 구현하는 방법파일 다운로드2
12695정성태7/3/20218225VC++: 145. C 언어의 setjmp/longjmp 기능을 Thread Context를 이용해 유사하게 구현하는 방법파일 다운로드1
12694정성태7/2/202110216Java: 24. Azure - Spring Boot 앱을 Java SE(Embedded Web Server)로 호스팅 시 로그 파일 남기는 방법 [1]
12693정성태6/30/20217965오류 유형: 731. Azure Web App Site Extension - Failed to install web app extension [...]. {1}
12692정성태6/30/20217886디버깅 기술: 180. Azure - Web App의 비정상 종료 시 남겨지는 로그 확인
12691정성태6/30/20218660개발 환경 구성: 573. 테스트 용도이지만 테스트에 적합하지 않은 Azure D1 공유(shared) 요금제
12690정성태6/28/20219495Java: 23. Azure - 자바(Java)로 만드는 Web App Service - Tomcat 호스팅
12689정성태6/25/202110103오류 유형: 730. Windows Forms 디자이너 - The class Form1 can be designed, but is not the first class in the file. [1]
... 31  32  33  34  35  36  [37]  38  39  40  41  42  43  44  45  ...