Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 30. 64비트 OS에서의 ChartFX 라이선스 문제 [링크 복사], [링크+제목 복사],
조회: 36725
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 3개 있습니다.)

64비트 OS에서의 ChartFX 라이선스 문제


기억하실 분들이 계실지 모르겠습니다. 지난번에, 이 문제를 설명하면서 "땜빵" 식으로 처리를 했었지요.

오류 유형 : 42. System.ComponentModel.LicenseException 
; https://www.sysnet.pe.kr/2/0/518

그런데, 위의 사건이 있은 이후로 ChartFX가 얹혀진 간단한 예제 프로그램을 만들면서 다시 문제가 발생하기 시작했습니다. 이번에는, 아예 Runtime License를 얻을 수 없다면서 다음과 같은 오류 메시지를 내뱉기 시작했습니다.

x64_runtime_license_error_4.png

{"Couldn't get Run Time license for 'ChartFX.WinForms.Chart'"}

System.ComponentModel.LicenseException was unhandled
  Message="Couldn't get Run Time license for 'ChartFX.WinForms.Chart'"
  Source="ChartFX.WinForms"
  StackTrace:
       at ChartFX.WinForms.Internal.h.a(LicenseContext A_0, Type A_1, Object A_2, Boolean A_3, LicenseProvider A_4)
       at ChartFX.WinForms.Internal.j.a(LicenseContext A_0, Type A_1, Object A_2, Boolean A_3)
       at System.ComponentModel.LicenseManager.ValidateInternalRecursive(LicenseContext context, Type type, Object instance, Boolean allowExceptions, License& license, String& licenseKey)
...  [중간생략] ...       
       at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
       at System.Threading.ThreadHelper.ThreadStart()

휴... 이쯤 되면, 참을만큼 참았습니다. 컴파일 시에 문제가 발생하는 거야 어떻게 해본다고 쳐도, 실행 시에도 이렇게 오류를 내버리니 ^^ 더 이상 방법이 없더군요.




자... 이제 문제를 해결해야 하는데,,, 어떤 방법이 좋을까 가만히 생각해 봤습니다. 대강 2가지 방법이 떠오르더군요.

  • WinDBG / SOS.dll을 이용한 디버깅
  • Sysinternals의 Regmon.exe 또는 ProcMon.exe를 이용한 관련 레지스트리 또는 파일의 문제 추적.

WinDBG를 이용하는 방법은 왠지 하기가 싫더군요. 이유는 난독 처리가 되어 있는 어셈블리 때문이었습니다. 그래서 일단, Procmon.exe를 이용해 보고 나서도 해결이 안되면 WinDBG를 사용하기로 마음을 굳히고, Procmon.exe를 통해서 문제 해결에 나섰습니다.

그나저나, 라이선스 문제와 procmon.exe가 무슨 관계가 있을까... 하고 생각하시는 분들이 계시겠지요.

사실, 로컬 PC에서 라이선스 운운하기 위해서는 필히 로컬 PC의 어딘가에 관련 정보를 저장해야 합니다. 그 과정에서 이용할 수 있는 자원이라고는 "파일" 또는 "레지스트리"뿐이 없고. 그냥 그렇게 ^^ 간단한 문제입니다.

자,,, 이제 비교를 시작해 봤습니다. 우선 기존에 개발 중인 exe 파일은 실행까지는 잘 되었기 때문에 그것을 실행시키는 과정의 레지스트리 Read/Write 작업을 모니터링하고 그 결과물을 파일로 저장했습니다. 이어서, 신규 예제 프로젝트를 실행시키고 오류가 발생하는 시점까지의 레지스트리 Read/Write 작업을 모니터링하고 역시 결과물을 저장해 두었습니다.

오... 그러다 보니, 흥미로운 사실을 하나 발견했습니다.

기존 개발중인 exe 파일은 VC++과의 연동 작업으로 인해 대상 프로세서 유형을 확실히 정해 두어야 했던 탓으로 x86으로 플랫폼이 맞춰져 있던 반면, 새롭게 제작한 예제 응용 프로그램은 "AnyCPU" 유형으로 인해 자연스럽게 x64 유형으로 JIT 컴파일링 되어 실행되어졌다는 사실이었습니다.

그 사실과 함께, 2가지 레지스트리 작업 결과물을 비교해 가던 중에 다음의 라인에서 차이가 나는 것을 발견할 수 있었습니다.

[그림: x64 유형 - 런타임 라이선스 오류가 발생]
x64_runtime_license_error_1.png

[그림: x86 유형 - 정상 실행]
x64_runtime_license_error_2.png

오호... "HKCU\Software\Classes\Licenses\{...GUID...}\(Default)" 값이 x86에서는 읽혀졌던 반면, x64에서는 읽혀지지 않은 것을 확인할 수 있습니다. 실제로, RegEdit.exe로 확인해 보니, x86의 "Wow6432Node" 하위에는 라이선스 문자열이 있었지만 x64 하위에는 존재하지 않았습니다.

아하... 답이 나왔습니다. ChartFX 설치 프로그램이 32bit 응용 프로그램이었기 때문에 관련 레지스트를 Wow6432Node 하위에만 기록을 했을 뿐, 64비트 관련 레지스트리에는 기록을 하지 않았기 때문에 이러한 문제가 발생한 것이었습니다. 그렇게 따져 보니, 지난번 문제의 LC.exe가 정상적으로 실행되지 않았던 의문이 풀리기 시작했습니다. 즉, 64비트용의 LC.exe가 실행되었기 때문에 그때도 역시 라이선스 문자열을 레지스트리에서 찾을 수가 없어 문제가 발생한 것이었습니다.

그렇다면... ^^ Wow6432Node 하위에 있던 그 라이선스 문자열을 64비트 레지스트리 하위에 복사해 주면 문제는 해결되겠군요.




가벼운 마음으로 ^^ Copy & Paste를 하고는 실행시켜 보았습니다.
헛... 그런데 이게 웬일입니까?

오류 유형이 다르긴 하지만, 이번에도 역시 ChartFX 생성자에서 오류가 발생했습니다. 아니... 다른 PC도 아니고, 동일한 PC에서 (64비트라는 점을 감안해서) 동등한 레지스트리 경로의 값에 대해 어떻게 이런 반응을 보일 수가 있을까???

역시나, 가만히 살펴보니 문제가 있습니다.
이해할 수는 없지만, ChartFX 측에서 REG_SZ 유형임에도 불구하고 중간에 "\r\n" 문자를 넣어둔 것이었습니다.

x64_runtime_license_error_3.png

당연히 그러한 개행문자가 포함된 문자열을 regedit.exe의 편집 상자에서는 첫 번째 라인의 라이선스 문자열만 입력시킬 수가 있었고, 암호 해독 과정에서 정상적인 처리를 하지 못하게 만드는 결과를 초래한 것이었습니다.

휴... 귀찮군요. ^^ 그냥 다음과 같이 간단하게 프로그램을 만들어서 실행시켜 주었습니다.

private void Form1_Load(object sender, EventArgs e)
{
  string key = "HKEY_CLASSES_ROOT\\Licenses\\1E1D9DB5-1D37-4a8f-B55E-8315007CACA9";

  string value = "nRamFftbAAAfbAABUNORjcwC1NGW... [중간 생략] ...UIxNENBNz8AAAAA" + "\r\n" +
    "NnB+OqDTCdLLgvev9uwRj74HeWSTUuHEpk... [중간 생략] ...OtGrNQ2WlkkA=";

  Registry.SetValue(key, string.Empty, value);
}

자, 이제 모두 해결되었습니다. (물론, 잊지 않고 지난번에 주석 처리해 두었던 <LC />도 풀어주고. ^^)




문제를 다 해결하고 나서, 혹시나 ChartFX 사이트에 이와 관련된 문의가 있었을까 싶어서 찾아보았더니 다음의 게시물이 검색되었습니다.

Chart FX DevStudio 설치가 안됩니다.
; http://www.chartfxkorea.com/community/com_read.asp?cl_no=6087&CurPage=1&cl_type=NQ

지금이라도 도움이 되려나 싶어, 답변을 달아 놓을라고 했더니... 로그인하라고 해서 포기. ^^;



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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/26/2021]

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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  [112]  113  114  115  116  117  118  119  120  ...
NoWriterDateCnt.TitleFile(s)
11158정성태3/22/201719370디버깅 기술: 85. Windbg - SOS 디버깅 사례 System.NullReferenceException 예외 추적
11157정성태3/22/201722698.NET Framework: 648. Dictionary<TKey, TValue>를 deep copy하는 방법파일 다운로드1
11156정성태3/21/201723982.NET Framework: 647. 닷넷(C#) 코드로 인증서 요청 코드 만드는 방법파일 다운로드1
11155정성태3/21/201724269.NET Framework: 646. SslStream의 CipherAlgorithm 선택이 가능할까요?파일 다운로드1
11154정성태3/5/201730599VC++: 109. DLL에서 STL 객체를 인자/반환값으로 갖는 함수를 제공할 때, 그 함수를 외부에서 사용하는 경우 비정상 종료한다면? [2]파일 다운로드1
11153정성태3/5/201730179VC++: 108. DLL에 정의된 C++ template 클래스의 복사 생성자 문제파일 다운로드1
11152정성태3/4/201724333VC++: 107. VirtualAlloc, HeapAlloc, GlobalAlloc, LocalAlloc, malloc, new의 차이점 [1]파일 다운로드1
11151정성태3/3/201724513VC++: 106. DLL 개발자가 주의해야 할 Secure CRT 함수 사용 [1]파일 다운로드1
11150정성태2/21/201720471.NET Framework: 645. Visual Studio Fakes 기능에서 Shim... 클래스가 생성되지 않는 경우 [5]
11149정성태2/21/201724308오류 유형: 378. A 64-bit test cannot run in a 32-bit process. Specify platform as X64 to force test run in X64 mode on X64 machine.
11148정성태2/20/201723414.NET Framework: 644. AppDomain에 대한 단위 테스트 시 알아야 할 사항
11147정성태2/19/201721844오류 유형: 377. Windows 10에서 Fake 어셈블리를 생성하는 경우 빌드 시 The type or namespace name '...' does not exist in the namespace 컴파일 오류 발생
11146정성태2/19/201721119오류 유형: 376. Error VSP1033: The file '...' does not contain a recognized executable image. [2]
11145정성태2/16/201722585.NET Framework: 643. 작업자 프로세스(w3wp.exe)가 재시작되는 시점을 알 수 있는 방법 - 두 번째 이야기 [4]파일 다운로드1
11144정성태2/6/201726003.NET Framework: 642. C# 개발자를 위한 Win32 DLL export 함수의 호출 규약 (부록 1) - CallingConvention.StdCall, CallingConvention.Cdecl에 상관없이 왜 호출이 잘 될까요?파일 다운로드1
11143정성태2/5/201723536.NET Framework: 641. [Out] 형식의 int * 인자를 가진 함수에 대한 P/Invoke 호출 방법파일 다운로드1
11142정성태2/5/201731409.NET Framework: 640. 닷넷 - 배열 크기의 한계 [2]파일 다운로드1
11141정성태1/31/201725873.NET Framework: 639. C# 개발자를 위한 Win32 DLL export 함수의 호출 규약 (4) - CLR JIT 컴파일러의 P/Invoke 호출 규약 [1]파일 다운로드1
11140정성태1/27/201721263.NET Framework: 638. RSAParameters와 RSA파일 다운로드1
11139정성태1/22/201724148.NET Framework: 637. C# 개발자를 위한 Win32 DLL export 함수의 호출 규약 (3) - x64 환경의 __fastcall과 Name mangling [1]파일 다운로드1
11138정성태1/20/201722576VS.NET IDE: 113. 프로젝트 생성 시부터 "Enable the Visual Studio hosting process" 옵션을 끄는 방법 - 두 번째 이야기 [3]
11137정성태1/20/201720717Windows: 135. AD에 참여한 컴퓨터로 RDP 연결 시 배경 화면을 못 바꾸는 정책
11136정성태1/20/201720351오류 유형: 375. Hyper-V 내에 구성한 Active Directory 환경의 시간 구성 방법 - 두 번째 이야기
11135정성태1/20/201721186Windows: 134. Windows Server 2016의 작업 표시줄에 있는 시계가 사라졌다면? [1]
11134정성태1/20/201728785.NET Framework: 636. System.Threading.Timer를 이용해 타이머 작업을 할 때 유의할 점 [5]파일 다운로드1
11133정성태1/20/201724816.NET Framework: 635. C# 개발자를 위한 Win32 DLL export 함수의 호출 규약 (2) - x86 환경의 __fastcall [1]파일 다운로드1
... 106  107  108  109  110  111  [112]  113  114  115  116  117  118  119  120  ...