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

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




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

[연관 글]


donaricano-btn



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

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

비밀번호

댓글 쓴 사람
 




1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
12740정성태7/29/2021190오류 유형: 747. SharePoint - InvalidOperationException 0x80131509
12739정성태7/28/2021174오류 유형: 746. Azure Active Directory - IDW10106: The 'ClientId' option must be provided.
12738정성태7/28/2021231오류 유형: 745. Azure Active Directory - Client credential flows must have a scope value with /.default suffixed to the resource identifier (application ID URI).
12737정성태7/28/2021185오류 유형: 744. Azure Active Directory - The resource principal named api://...[client_id]... was not found in the tenant
12736정성태7/28/2021150오류 유형: 743. Active Azure Directory에서 "API permissions"의 권한 설정이 "Not granted for ..."로 나오는 문제
12735정성태7/27/2021238.NET Framework: 1081. C# - Azure AD 인증을 지원하는 데스크톱 애플리케이션 예제(Windows Forms)파일 다운로드1
12734정성태7/26/2021310스크립트: 20. 특정 단어로 시작하거나/끝나는 문자열을 포함/제외하는 정규 표현식 - Look-around
12733정성태7/23/2021192.NET Framework: 1081. Self-Contained/SingleFile 유형의 .NET Core/5+ 실행 파일을 임베딩한다면?파일 다운로드2
12732정성태7/23/2021128오류 유형: 742. SharePoint - The super user account utilized by the cache is not configured.
12731정성태7/23/2021138개발 환경 구성: 584. Add Internal URLs 화면에서 "Save" 버튼이 비활성화 된 경우
12730정성태7/23/2021230개발 환경 구성: 583. Visual Studio Code - Go 코드에서 입력을 받는 경우
12729정성태7/22/2021178.NET Framework: 1080. xUnit 단위 테스트에 메서드/클래스 수준의 문맥 제공 - Fixture
12728정성태7/22/2021153.NET Framework: 1079. MSTestv2 단위 테스트에 메서드/클래스/어셈블리 수준의 문맥 제공
12727정성태7/21/2021267.NET Framework: 1078. C# 단위 테스트 - MSTestv2/NUnit의 Assert.Inconclusive 사용법(?)
12726정성태7/21/2021253VS.NET IDE: 169. 비주얼 스튜디오 - 단위 테스트 선택 시 MSTestv2 외의 xUnit, NUnit 사용법
12725정성태7/21/2021275오류 유형: 741. Failed to find the "go" binary in either GOROOT() or PATH
12724정성태7/21/2021364개발 환경 구성: 582. 윈도우 환경에서 Visual Studio Code + Go (Zip) 개발 환경 [1]
12723정성태7/21/2021186오류 유형: 740. SharePoint - Alternate access mappings have not been configured 경고
12722정성태7/20/2021200오류 유형: 739. MSVCR110.dll이 없어 exe 실행이 안 되는 경우
12721정성태7/20/2021142오류 유형: 738. The trust relationship between this workstation and the primary domain failed. - 세 번째 이야기
12720정성태7/19/2021191Linux: 43. .NET Core/5+ 응용 프로그램의 Ubuntu (Debian) 패키지 준비
12719정성태7/19/2021129오류 유형: 737. SharePoint 설치 시 "0x800710D8 The object identifier does not represent a valid object." 오류 발생
12718정성태7/19/2021191개발 환경 구성: 581. Windows에서 WSL로 파일 복사 시 root 소유권으로 적용되는 문제파일 다운로드1
12717정성태7/18/2021190Windows: 195. robocopy에서 파일의 ADS(Alternate Data Stream) 정보 복사를 제외하는 방법
12716정성태7/17/2021222개발 환경 구성: 580. msbuild의 Exec Task에 robocopy를 사용하는 방법파일 다운로드1
12715정성태7/17/2021897오류 유형: 736. Windows - MySQL zip 파일 버전의 "mysqld --skip-grant-tables" 실행 시 비정상 종료
1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...