Microsoft MVP성태의 닷넷 이야기
VC++: 25. Microsoft National Language Support Downlevel APIs 1.0 사용 방법 [링크 복사], [링크+제목 복사],
조회: 21239
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
[TestDlg.zip]    

2006/06/30일자로 MSDN 다운로드 센터에 간단한 유틸리티 함수를 제공하는 "Microsoft National Language Support Downlevel APIs 1.0"이 올라왔습니다.

다음의 기사를 참고하십시오.
Microsoft National Language Support Downlevel APIs 1.0
; https://www.sysnet.pe.kr/2/1/353

그런데, 잠시 테스트해 볼려고 VS.NET 2005에서 예제로 MFC Dialog 유형의 프로젝트를 만들어서 다음과 같은 코드를 컴파일 해보았습니다.

#include "..\include\Nlsdl.h"
#pragma comment( lib, "..\\lib\\x86\\Nlsdl.lib" )

LCID lcid = DownlevelLocaleNameToLCID( L"ko-KR", DOWNLEVEL_LOCALE_NAME );
// lcid == 1024

LCID parentLcid = DownlevelGetParentLocaleLCID( lcid );
// lcid == 18

wchar_t name[ 1024 ] = { 0 };
DownlevelLCIDToLocaleName( lcid, name, 1024, 0 );
// name == "ko-KR"

wchar_t parentName[ 1024 ] = { 0 };
DownlevelGetParentLocaleName( lcid, parentName, 1024 );
// parentName == "ko"

그런데 ^^; 이게 웬일입니까?
다음과 같은 오류가 나는 것입니다.

TestDlgDlg.obj : error LNK2019: unresolved external symbol "int __stdcall DownlevelGetParentLocaleName(unsigned long,wchar_t *,int)" (?DownlevelGetParentLocaleName@@YGHKPA_WH@Z) referenced in function "protected: virtual int __thiscall CTestDlgDlg::OnInitDialog(void)" (?OnInitDialog@CTestDlgDlg@@MAEHXZ)
TestDlgDlg.obj : error LNK2019: unresolved external symbol "int __stdcall DownlevelLCIDToLocaleName(unsigned long,wchar_t *,int,unsigned long)" (?DownlevelLCIDToLocaleName@@YGHKPA_WHK@Z) referenced in function "protected: virtual int __thiscall CTestDlgDlg::OnInitDialog(void)" (?OnInitDialog@CTestDlgDlg@@MAEHXZ)
TestDlgDlg.obj : error LNK2019: unresolved external symbol "unsigned long __stdcall DownlevelLocaleNameToLCID(wchar_t const *,unsigned long)" (?DownlevelLocaleNameToLCID@@YGKPB_WK@Z) referenced in function "protected: virtual int __thiscall CTestDlgDlg::OnInitDialog(void)" (?OnInitDialog@CTestDlgDlg@@MAEHXZ)
C:\Microsoft NLS Downlevel APIs\TestDlg\Debug\TestDlg.exe : fatal error LNK1120: 3 unresolved externals

분명히, Nlsdl.h 파일에 정의된 대로 정확하게 사용을 했는데 위와 같은 에러가 나는 것입니다.
오호... Microsoft가 뭔가 실수를 했을까요?

우선, Nlsdl.h 파일에 있는 정의 대로 LIB 파일에 있는지 검사를 해야 했습니다. 여러 가지 툴을 사용해 볼 수 있겠지만, 단순히 export 된 함수에 대한 signature만 확인하면 되기 때문에 그냥 LIB 파일을 VS.NET IDE로 끌어다 놓아서 binary 형식으로 보았습니다.

아래와 같이 export 되어 있는 것을 확인해 볼 수 있는데요.
CRIPTOR_Nlsdl__NULL_IMPORT_DESCRIPTOR
Nlsdl_NULL_THUNK_DATA
?DownlevelLocaleNameToLCID@@YGKPBGK@Z__imp_
?DownlevelLocaleNameToLCID@@YGKPBGK@Z
?DownlevelLCIDToLocaleName@@YGHKPAGHK@Z__imp_
?DownlevelLCIDToLocaleName@@YGHKPAGHK@Z
?DownlevelGetParentLocaleLCID@@YGKK@Z__imp_
?DownlevelGetParentLocaleLCID@@YGKK@Z
?DownlevelGetParentLocaleName@@YGHKPAGH@Z__imp_
?DownlevelGetParentLocaleName@@YGHKPAGH@Z        

그런데, 한 가지 이상한 점이 있습니다. 함수 이름 이후의 "@@" 문자뒤에 오는 signature 값이 VS.NET 컴파일 화면에 있는 값과 일치하지 않았습니다.

컴파일 오류 : ?DownlevelLocaleNameToLCID@@YGKPB_WK@Z
LIB export : ?DownlevelLocaleNameToLCID@@YGKPBGK@Z

자, 그럼 위의 signature를 우리가 읽기 편하게 변환을 하면 될 텐데요. 이를 위해 VC++ 툴에 있는 "undname.exe"를 이용해 보겠습니다. (자세한 사항은 https://www.sysnet.pe.kr/2/0/151을 참조하십시오.)

결국, 다음과 같은 결과를 얻어냈습니다.

컴파일 오류 : ?DownlevelLocaleNameToLCID@@YGKPB_WK@Z
             - unsigned long __stdcall DownlevelLocaleNameToLCID(wchar_t const *,unsigned long)
LIB export : ?DownlevelLocaleNameToLCID@@YGKPBGK@Z
             - unsigned long __stdcall DownlevelLocaleNameToLCID(unsigned short const *, unsigned long)"

확실히 다르군요. ^^ 그런데, Nlsdl.h 파일에는 해당 함수에 대해서 다음과 같이 선언되어져 있습니다.

LCID WINAPI DownlevelLocaleNameToLCID(
    __in                    LPCWSTR lpName, 
    __in                    DWORD   dwFlags);

LPCWSTR : CONST WCHAR *
WCHAR : wchar_t

어쨌든, 문제 파악은 제대로 된 것 같습니다. 이제 이러한 불일치를 해결하기 위해서 2가지 방식으로 풀어볼 수 있습니다.

1. 헤더 파일 및 소스의 강제 포인터 형변환
예를 들면 더욱 설명이 빠르겠지요.

=== 헤더 파일에 다음의 선언 추가 ===
LCID WINAPI DownlevelLocaleNameToLCID(
    __in                    unsigned short const *lpName, 
    __in                    DWORD   dwFlags);

=== 소스 파일에서의 강제 형변환 ===
	LCID lcid = DownlevelLocaleNameToLCID( (unsigned short*)L"ko-KR", DOWNLEVEL_LOCALE_NAME );

나머지 2개의 함수에 대해서도 위와 같은 방법으로 처리해 줄 수 있습니다.

2. 컴파일러 스위치 사용
프로젝트 대화 상자에서 아래와 같이 "Treat wchar_t as Build-in Type"을 "No (/Zc:wchar_t-)"로 주어서 해결할 수 있습니다. 이렇게 해주면, wchar_t는 하나의 내장 형식이 아닌, typedef을 통해서 unsigned short로 바뀌게 됩니다.
wchar_t 처리
첨부된 파일은 제가 사용한 간단한 예제 프로젝트입니다. 4개의 함수 사용 및 unsigned short로 변환된 헤더 정의를 포함하고 있습니다. Nlsdl.lib와 Nlsdl.dll을 포함시켜 두었는데 모두 x86 버전입니다. iax6/x64 버전으로 테스트하시려면 "Microsoft National Language Support Downlevel APIs 1.0"을 직접 다운로드하면 그에 대한 재배포 파일을 구하실 수 있습니다.






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

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

비밀번호

댓글 작성자
 




... 136  137  138  139  140  141  142  [143]  144  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1479정성태8/14/201325117오류 유형: 183. IIS - 바인딩 추가 시 Object reference not set to an instance of an object 오류 [5]
1478정성태8/14/201328454오류 유형: 182. 윈도우 정품 활성화 오류 - 0x80070426
1477정성태8/14/201327292VC++: 71. codeplex의 Project Austin - 실감나게 책장 넘기는 표현
1476정성태8/13/201335768디버깅 기술: 55. Windbg - 윈도우 핸들 테이블 (2)
1475정성태8/12/201334875.NET Framework: 377. 프로세스가 종료된 후에도 소켓이 살아있다면?파일 다운로드1
1474정성태8/10/201330929오류 유형: 181. 윈도우 8 - WmiPrvSE.exe 프로세스가 CPU 소비하는 현상
1473정성태8/8/201327772VC++: 70. Win32 socket이 Thread-safe할까? [1]파일 다운로드1
1472정성태8/7/201326172.NET Framework: 376. .NET 2.0의 유니코드 관련 문자열 비교 오류
1471정성태8/7/201330990개발 환경 구성: 193. .aspx 확장자 대신 .html 확장자를 사용하는 방법
1470정성태8/6/201326957오류 유형: 180. DISM.exe 0xc1510111 실행 오류
1469정성태8/6/201323964.NET Framework: 375. System.Net.Sockets.Socket이 Thread-safe할까? [2]파일 다운로드1
1468정성태8/6/201322115오류 유형: 179. IIS - No connection could be made because the target machine actively refused it 127.0.0.1:80
1467정성태8/5/201325555Java: 16. IE에 로드된 Java Applet의 다운로드 위치를 확인하는 방법
1466정성태7/27/201331152.NET Framework: 374. C#과 비교한 C++ STL vector 성능 [7]파일 다운로드1
1465정성태7/18/201334468기타: 33. C:\Windows\Installer 폴더의 용량 줄이기 [3]
1464정성태7/15/201322727오류 유형: 178. Visual Studio 2012 Express - ImportCardinalityMismatchException
1463정성태7/15/201323418오류 유형: 177. [DBNETLIB][ConnectionOpen (Connect()).]SQL Server does not exist or access denied.
1462정성태7/5/201326700VC++: 69. geek스러운 C/C++ 퀴즈 문제 [2]
1461정성태6/27/201343269.NET Framework: 373. C# 문자열의 인코딩이란?
1460정성태6/17/201325101.NET Framework: 372. PerformanceCounter - Category does not exist. [1]
1459정성태6/15/201328782Windows: 74. 한글 키가 아닌 영문 키를 기본으로 선택하는 방법 [5]
1458정성태6/13/201329588.NET Framework: 371. CAS Lock 방식이 과연 성능에 얼마나 도움이 될까요? [1]파일 다운로드1
1457정성태6/13/201325779개발 환경 구성: 192. "Probabilistic Programming and Bayesian Methods for Hackers" 예제 코드 실행 방법
1456정성태6/5/201334422.NET Framework: 370. C# - WebKit .NET 사용 [2]파일 다운로드1
1455정성태6/1/201328253.NET Framework: 369. ThreadPool.QueueUserWorkItem의 실행 지연 [4]파일 다운로드1
1454정성태5/31/201326255Java: 15. Java 7 Control Panel 실행시키는 방법
... 136  137  138  139  140  141  142  [143]  144  145  146  147  148  149  150  ...