Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 3개 있습니다.)

MASM + CRT 함수를 사용하는 경우 발생하는 컴파일 오류 정리

C/C++ 함수를 사용하는 경우 발생하는 링커 오류를 정리합니다.

이전 글에서 마지막 hello_cpp.asm을 컴파일할 때, 그냥 Win32 API 호출하던 예제를 컴파일하듯이 명령어를 주면,

ml64 hello_cpp.asm /link /subsystem:console kernel32.lib /entry:main

당연히 printf 함수를 찾을 수 없다는 오류가 발생합니다.

hello_cpp.obj : error LNK2019: unresolved external symbol printf referenced in function main

그래서 단순히 이에 대한 정의만 포함하고 있는 legacy_stdio_definitions.lib를 추가하면,

ml64 hello_cpp.asm /link /subsystem:console kernel32.lib legacy_stdio_definitions.lib /entry:main

이제는 더 많은 오류가 발생합니다.

legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __acrt_iob_func referenced in function _vwprintf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfwprintf referenced in function _vfwprintf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfwprintf_s referenced in function _vfwprintf_s_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfwprintf_p referenced in function _vfwprintf_p_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfwscanf referenced in function _vfwscanf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vswprintf referenced in function _vsnwprintf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vswprintf_s referenced in function _vswprintf_s_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vsnwprintf_s referenced in function _vsnwprintf_s_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vswprintf_p referenced in function _vswprintf_p_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vswscanf referenced in function _vswscanf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfprintf referenced in function _vfprintf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfprintf_s referenced in function _vfprintf_s_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfprintf_p referenced in function _vfprintf_p_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vfscanf referenced in function _vfscanf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vsprintf referenced in function _vsnprintf_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vsprintf_s referenced in function _vsprintf_s_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vsnprintf_s referenced in function _vsnprintf_s_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vsprintf_p referenced in function _vsprintf_p_l
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2019: unresolved external symbol __stdio_common_vsscanf referenced in function _vsscanf_l
hello_cpp.exe : fatal error LNK1120: 19 unresolved externals

이 함수들에 대한 정의를 포함한 ucrt.lib를 다시 한 번 더 추가해야 합니다.

ml64 hello_cpp.asm /link /subsystem:console kernel32.lib ucrt.lib legacy_stdio_definitions.lib /entry:main

이제는 빌드까지는 잘 되지만 찜찜한 링크 경고가 하나 남습니다.

legacy_stdio_wide_specifiers.lib(legacy_stdio_wide_specifiers.obj) : warning LNK4210: .CRT section exists; there may be unhandled static initializers or terminators

즉, CRT 초기화를 위한 진입점을 호출해야 하는 건데요, 따라서 진입점을 우리가 만든 main이 아니라 CRT의 main에 해당하는 mainCRTStartup로 /entry 옵션을 잡아줘야 합니다.

ml64 hello_cpp.asm /link /subsystem:console kernel32.lib ucrt.lib legacy_stdio_definitions.lib /entry:main /entry:mainCRTStartup

하지만 ^^; 그래도 오류가 발생합니다.

LINK : error LNK2001: unresolved external symbol mainCRTStartup

그래서 mainCRTStartup를 정의한 libcmt.lib를 추가하면,

ml64 hello_cpp.asm /link /subsystem:console kernel32.lib libcmt.lib ucrt.lib legacy_stdio_definitions.lib /entry:main /entry:mainCRTStartup

될 듯도 한데, 분명히 완벽한 데도 불구하고 오류가 발생합니다.

libucrt.lib(exit.obj) : error LNK2005: _c_exit already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(exit.obj) : error LNK2005: _cexit already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(exit.obj) : error LNK2005: _exit already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(exit.obj) : error LNK2005: _register_thread_local_exe_atexit_callback already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(exit.obj) : error LNK2005: exit already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(onexit.obj) : error LNK2005: _crt_at_quick_exit already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(onexit.obj) : error LNK2005: _crt_atexit already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(onexit.obj) : error LNK2005: _execute_onexit_table already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(onexit.obj) : error LNK2005: _initialize_onexit_table already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(onexit.obj) : error LNK2005: _register_onexit_function already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(argv_data.obj) : error LNK2005: __p___argc already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(argv_data.obj) : error LNK2005: __p___argv already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(matherr.obj) : error LNK2005: __setusermatherr already defined in ucrt.lib(api-ms-win-crt-math-l1-1-0.dll)
libucrt.lib(environment_initialization.obj) : error LNK2005: _get_initial_narrow_environment already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(environment_initialization.obj) : error LNK2005: _initialize_narrow_environment already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(wsetlocale.obj) : error LNK2005: _configthreadlocale already defined in ucrt.lib(api-ms-win-crt-locale-l1-1-0.dll)
libucrt.lib(exception_filter.obj) : error LNK2005: _seh_filter_dll already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(exception_filter.obj) : error LNK2005: _seh_filter_exe already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
libucrt.lib(new_mode.obj) : error LNK2005: _set_new_mode already defined in ucrt.lib(api-ms-win-crt-heap-l1-1-0.dll)
libucrt.lib(report_runtime_error.obj) : error LNK2005: _set_app_type already defined in ucrt.lib(api-ms-win-crt-runtime-l1-1-0.dll)
hello_cpp.exe : fatal error LNK1169: one or more multiply defined symbols found

이 오류는, 명령행의 인자로 지정한 라이브러리 순서가 libcmt.lib + ucrt.lib인 것을 거꾸로 ucrt.lib + libcmt.lib 순으로 해주면 정상적으로 빌드가 됩니다. ^^;

ml64 hello_cpp.asm /link /subsystem:console kernel32.lib ucrt.lib libcmt.lib legacy_stdio_definitions.lib /entry:main /entry:mainCRTStartup

자, 그럼 이 상태에서 nodefaultlib 옵션을 주면 어떨까요?

ml64 hello_cpp.asm /link /nodefaultlib /subsystem:console kernel32.lib ucrt.lib libcmt.lib legacy_stdio_definitions.lib /entry:main /entry:mainCRTStartup

이런 오류들이 쏟아져 나오는데,

libcmt.lib(exe_main.obj) : error LNK2001: unresolved external symbol __C_specific_handler
libcmt.lib(utility.obj) : error LNK2001: unresolved external symbol __C_specific_handler
libcmt.lib(tncleanup.obj) : error LNK2019: unresolved external symbol __std_type_info_destroy_list referenced in function "void __cdecl __scrt_uninitialize_type_info(void)" (?__scrt_uninitialize_type_info@@YAXXZ)
libcmt.lib(utility_desktop.obj) : error LNK2019: unresolved external symbol __current_exception referenced in function __scrt_unhandled_exception_filter
libcmt.lib(utility_desktop.obj) : error LNK2019: unresolved external symbol __current_exception_context referenced in function __scrt_unhandled_exception_filter
libcmt.lib(utility_desktop.obj) : error LNK2019: unresolved external symbol memset referenced in function __scrt_fastfail
legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2001: unresolved external symbol __PLEASE_LINK_WITH_legacy_stdio_wide_specifiers.lib

vcruntime.lib와 링크시키면 됩니다.

ml64 hello_cpp.asm /link /nodefaultlib /subsystem:console kernel32.lib vcruntime.lib ucrt.lib libcmt.lib legacy_stdio_definitions.lib /entry:main /entry:mainCRTStartup

그래도 여전히 오류가 발생합니다. ^^

legacy_stdio_definitions.lib(legacy_stdio_definitions.obj) : error LNK2001: unresolved external symbol __PLEASE_LINK_WITH_legacy_stdio_wide_specifiers.lib

다행히 이번에는 오류 메시지에서 legacy_stdio_wide_specifiers.lib를 포함해야 한다는 것을 알 수 있습니다. ^^

ml64 hello_cpp.asm /link /nodefaultlib /subsystem:console kernel32.lib vcruntime.lib ucrt.lib libcmt.lib legacy_stdio_definitions.lib legacy_stdio_wide_specifiers.lib /entry:main /entry:mainCRTStartup

이제야 끝이 났습니다. ^^ 정리해 보면, C 런타임을 사용하는 경우 기본적으로 다음과 같은 라이브러리를 함께 링크시켜야 합니다 ^^

  • vcruntime.lib
  • ucrt.lib
  • libcmt.lib
  • legacy_stdio_definitions.lib
  • legacy_stdio_wide_specifiers.lib




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/5/2022]

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

비밀번호

댓글 작성자
 



2024-10-07 10시19분
Understanding the classical model for linking, groundwork: The algorithm
; https://devblogs.microsoft.com/oldnewthing/20130107-00/?p=5633

Understanding the classical model for linking: Taking symbols along for the ride
; https://devblogs.microsoft.com/oldnewthing/20130108-00/?p=5623

Understanding the classical model for linking: You can override an LIB with another LIB, and a LIB with an OBJ, but you can’t override an OBJ
; https://devblogs.microsoft.com/oldnewthing/20130109-00/?p=5613

Understanding the classical model for linking: Sometimes you don’t want a symbol to come along for a ride
; https://devblogs.microsoft.com/oldnewthing/20130110-00/?p=5593

Understanding errors in classical linking: The delay-load catch-22
; https://devblogs.microsoft.com/oldnewthing/20130111-00/?p=5583

Using the classical model for linking to provide unit test overrides
; https://devblogs.microsoft.com/oldnewthing/20250416-00/?p=111077
정성태

... 136  137  138  [139]  140  141  142  143  144  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1669정성태5/15/201448847Windows: 93. 윈도우 시스템 디스크 용량 확보를 위한 $PatchCache$ 폴더 삭제 [2]
1668정성태5/10/201427514.NET Framework: 434. Microsoft.SqlServer.Types.SqlGeography 형변환 시 null 반환하는 문제
1667정성태5/5/201428731개발 환경 구성: 221. Azure 데이터베이스를 로컬 DB로 이전하는 방법 [2]
1666정성태5/2/201445540기타: 45. 윈도우 계정의 암호를 알아내는 mimikatz 도구 [5]
1665정성태5/1/201428718.NET Framework: 433. C# - 간단한 HyperLogLog 자료 구조 테스트파일 다운로드1
1664정성태4/28/201426351오류 유형: 227. Process Explorer의 프로세스 뷰가 트리 형식으로 보이지 않는 문제
1663정성태4/28/201422489오류 유형: 226. Visual Studio - We were unable to establish the connection because it is configured for user
1662정성태4/28/201427456개발 환경 구성: 220. supportedRuntime 설정을 위한 app.config Transformation [1]
1661정성태4/26/201423271.NET Framework: 432. WPF - System.Windows.Data Error: 47 : XmlDataProvider has inline XML that does not explicitly set its XmlNamespace (xmlns="").
1660정성태4/25/201431368VC++: 77. C++ 숫자형 값이 범위를 벗어나는 경우의 출력 사례 모음
1659정성태4/17/201431949.NET Framework: 431. .NET EXE 파일을 닷넷 프레임워크 버전에 상관없이 실행할 수 있을까요? [5]
1658정성태4/17/201422921.NET Framework: 430. C#에서 사용자 정의 예약어가 가능할까요? [1]
1657정성태4/10/201443779.NET Framework: 429. C# - 유니코드 한글 문자열을 ks_c_5601-1987로 변환하는 방법 [3]파일 다운로드1
1656정성태3/19/201426628오류 유형: 225. regsvcs 등록 시 0x80040153 오류
1655정성태3/19/201427029Windows: 92. Thumbs.db 파일이 삭제 안 되는 문제
1654정성태3/19/201429159개발 환경 구성: 219. SOS.dll 확장 모듈을 버전 별로 구하는 방법 [4]
1653정성태3/13/201423666.NET Framework: 428. .NET Reflection으로 다차원/Jagged 배열을 구분하는 방법
1652정성태3/12/201424265VC++: 76. Direct Show를 사용하는 다른 프로그램의 필터 그래프를 graphedt.exe에서 확인하는 방법파일 다운로드1
1651정성태3/11/201428894.NET Framework: 427. C# 컴파일러는 변수를 초기화시키지 않을까요?
1650정성태3/6/201429578VC++: 75. Visual C++ 컴파일 오류 - Cannot use __try in functions that require object unwinding [1]파일 다운로드1
1649정성태3/5/201423887기타: 44. BTN 스토어 앱 개인정보 보호 정책 안내
1648정성태3/5/201424337개발 환경 구성: 218. 스토어 앱 인증 실패 - no privacy statement
1647정성태3/3/201425822오류 유형: 224. 스카이드라이브 비정상 종료 - Error 0x80040A41: No error description available
1646정성태3/3/201435217오류 유형: 223. Microsoft-Windows-DistributedCOM 10016 이벤트 로그 에러 [1]
1645정성태3/1/201424656기타: 43. 마이크로소프트 MVP들이 모여 전국 세미나를 엽니다.
1644정성태2/26/201431942.NET Framework: 426. m3u8 스트리밍 파일을 윈도우 8.1 Store App에서 재생하는 방법파일 다운로드1
... 136  137  138  [139]  140  141  142  143  144  145  146  147  148  149  150  ...