Microsoft MVP성태의 닷넷 이야기
VC++: 25. Microsoft National Language Support Downlevel APIs 1.0 사용 방법 [링크 복사], [링크+제목 복사],
조회: 21173
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 151  152  153  154  155  156  157  [158]  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1096정성태8/15/201125657디버깅 기술: 42. Watson Bucket 정보를 이용한 CLR 응용 프로그램 예외 분석 - (2)
1095정성태8/14/201126101디버깅 기술: 41. Windbg - 비정상 종료된 닷넷 프로그램의 StackTrace에서 보이는 offset 값 의미
1094정성태8/14/201130458오류 유형: 131. Fiddler가 강제 종료된 경우, 웹 사이트 방문이 안되는 현상
1093정성태7/27/201124092오류 유형: 130. Unable to connect to the Microsoft Visual Studio Remote Debugging Monitor ... Access is denied.
1092정성태7/22/201126468Team Foundation Server: 46. 코드 이외의 파일에 대해 소스 제어에서 제외시키는 방법
1091정성태7/21/201125473개발 환경 구성: 128. WP7 Emulator 실행 시 audiodg.exe의 CPU 소모율 증가 [2]
1089정성태7/18/201131051.NET Framework: 234. 왜? Button 컨트롤에는 MouseDown/MouseUp 이벤트가 발생하지 않을까요?파일 다운로드1
1088정성태7/16/201124195.NET Framework: 233. Entity Framework 4.1 - 윈도우 폰 7에서의 CodeFirst 순환 참조 문제파일 다운로드1
1087정성태7/15/201126817.NET Framework: 232. Entity Framework 4.1 - CodeFirst 개체의 직렬화 시 순환 참조 해결하는 방법 - 두 번째 이야기파일 다운로드1
1086정성태7/14/201128278.NET Framework: 231. Entity Framework 4.1 - CodeFirst 개체의 직렬화 시 순환 참조 해결하는 방법 [1]파일 다운로드1
1085정성태7/14/201128694.NET Framework: 230. Entity Framework 4.1 - Code First + WCF 서비스 시 EndpointNotFoundException 오류 - 두 번째 이야기파일 다운로드1
1084정성태7/11/201134003.NET Framework: 229. SQL 서버 - DB 테이블의 데이터 변경에 대한 알림 처리 [4]파일 다운로드1
1083정성태7/11/201128058.NET Framework: 228. Entity Framework 4.1 - Code First + WCF 서비스 시 EndpointNotFoundException 오류
1082정성태7/10/201127622.NET Framework: 227. basicHttpBinding + 사용자 정의 인증 구현 [2]파일 다운로드1
1081정성태7/9/201126939VC++: 53. Windows 7에서 gcc.exe 실행 시 Access denied 오류 [2]
1080정성태7/8/201125423웹: 23. Sysnet 웹 사이트의 HTML5 변환 기록 - 두 번째 이야기파일 다운로드1
1079정성태7/6/201129875오류 유형: 129. Hyper-V + Realtek 랜카드가 설치된 시스템의 BSOD 현상 [2]
1078정성태7/5/201137429VC++: 52. Chromium 컴파일하는 방법 [2]
1077정성태6/24/201135047.NET Framework: 226. HttpWebRequest 타입의 HaveResponse 속성 이야기파일 다운로드1
1076정성태6/23/201129176오류 유형: 128. SQL Express - User Instance 옵션을 사용한 경우 발생하는 오류 메시지 유형 2가지
1075정성태6/21/201124798VS.NET IDE: 69. 윈폰 프로젝트에서 WCF 서비스 참조할 때 Reference.cs 파일이 비어있는 경우
1074정성태6/20/201124874.NET Framework: 225. 닷넷 네트워크 라이브러리의 트레이스 기능파일 다운로드1
1073정성태6/20/201127103오류 유형: 127. Visual Studio에서 WCF 서비스의 이름 변경 시 발생할 수 있는 오류
1072정성태6/19/201126565.NET Framework: 224. EF 4.1 Code First에서 Identity 칼럼 생성하는 방법파일 다운로드1
1071정성태6/19/201130087.NET Framework: 223. Entity Framework 4.1의 Code First를 이용한 SQL Azure 데이터베이스 생성 [3]파일 다운로드1
1070정성태6/19/201127607.NET Framework: 222. Windows Azure - VM Role 베타 프로그램 참여 [2]
... 151  152  153  154  155  156  157  [158]  159  160  161  162  163  164  165  ...