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

비밀번호

댓글 작성자
 




... 16  17  18  19  20  21  22  23  24  25  26  27  28  [29]  30  ...
NoWriterDateCnt.TitleFile(s)
12893정성태12/27/20218692.NET Framework: 1124. C# - .NET Platform Extension의 ObjectPool<T> 사용법 소개파일 다운로드1
12892정성태12/26/20216675기타: 83. unsigned 형의 이전 값이 최댓값을 넘어 0을 지난 경우, 값의 차이를 계산하는 방법
12891정성태12/23/20216602스크립트: 38. 파이썬 - uwsgi의 --master 옵션
12890정성태12/23/20216740VC++: 152. Golang - (문자가 아닌) 바이트 위치를 반환하는 strings.IndexRune 함수
12889정성태12/22/20219147.NET Framework: 1123. C# - (SharpDX + DXGI) 화면 캡처한 이미지를 빠르게 JPG로 변환하는 방법파일 다운로드1
12888정성태12/21/20217343.NET Framework: 1122. C# - ImageCodecInfo 사용 시 System.Drawing.Image와 System.Drawing.Bitmap에 따른 Save 성능 차이파일 다운로드1
12887정성태12/21/20219366오류 유형: 777. OpenCVSharp4를 사용한 프로그램 실행 시 "The type initializer for 'OpenCvSharp.Internal.NativeMethods' threw an exception." 예외 발생
12886정성태12/20/20217325스크립트: 37. 파이썬 - uwsgi의 --enable-threads 옵션 [2]
12885정성태12/20/20217578오류 유형: 776. uwsgi-plugin-python3 환경에서 MySQLdb 사용 환경
12884정성태12/20/20216651개발 환경 구성: 620. Windows 10+에서 WMI root/Microsoft/Windows/WindowsUpdate 네임스페이스 제거
12883정성태12/19/20217489오류 유형: 775. uwsgi-plugin-python3 환경에서 "ModuleNotFoundError: No module named 'django'" 오류 발생
12882정성태12/18/20216598개발 환경 구성: 619. Windows Server에서 WSL을 위한 리눅스 배포본을 설치하는 방법
12881정성태12/17/20217146개발 환경 구성: 618. WSL Ubuntu 20.04에서 파이썬을 위한 uwsgi 설치 방법 (2)
12880정성태12/16/20216916VS.NET IDE: 170. Visual Studio에서 .NET Core/5+ 역어셈블 소스코드 확인하는 방법
12879정성태12/16/202113142오류 유형: 774. Windows Server 2022 + docker desktop 설치 시 WSL 2로 선택한 경우 "Failed to deploy distro docker-desktop to ..." 오류 발생
12878정성태12/15/20218187개발 환경 구성: 617. 윈도우 WSL 환경에서 같은 종류의 리눅스를 다중으로 설치하는 방법
12877정성태12/15/20216860스크립트: 36. 파이썬 - pymysql 기본 예제 코드
12876정성태12/14/20216704개발 환경 구성: 616. Custom Sources를 이용한 Azure Monitor Metric 만들기
12875정성태12/13/20216432스크립트: 35. python - time.sleep(...) 호출 시 hang이 걸리는 듯한 문제
12874정성태12/13/20216396오류 유형: 773. shell script 실행 시 "$'\r': command not found" 오류
12873정성태12/12/20217517오류 유형: 772. 리눅스 - PATH에 등록했는데도 "command not found"가 나온다면?
12872정성태12/12/20217307개발 환경 구성: 615. GoLang과 Python 빌드가 모두 가능한 docker 이미지 만들기
12871정성태12/12/20217457오류 유형: 771. docker: Error response from daemon: OCI runtime create failed
12870정성태12/9/20216060개발 환경 구성: 614. 파이썬 - PyPI 패키지 만들기 (4) package_data 옵션
12869정성태12/8/20218253개발 환경 구성: 613. git clone 실행 시 fingerprint 묻는 단계를 생략하는 방법
12868정성태12/7/20216825오류 유형: 770. twine 업로드 시 "HTTPError: 400 Bad Request ..." 오류 [1]
... 16  17  18  19  20  21  22  23  24  25  26  27  28  [29]  30  ...