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

C/C++ - Windows 10 Version 1607부터 지원하는 /DEPENDENTLOADFLAG 옵션

아래의 글에,

How can I try to escape the disease-ridden hot-tubs known as the TEMP and Downloads directories?
; https://devblogs.microsoft.com/oldnewthing/20230328-00/?p=107978

새로운 옵션이 소개됐군요. ^^

/DEPENDENTLOADFLAG (Set default dependent load flags)
; https://learn.microsoft.com/en-us/cpp/build/reference/dependentloadflag

이 옵션은 LoadLibraryEx의 dwFlags 인자와 동일한 값을 취하고, 그 역할도 같습니다. 단지 차이점이라면, LoadLibraryEx는 사용자가 직접 호출하는 코드에 flags를 지정하게 되는 반면, /DEPENDENTLOADFLAG는 정적으로 링크된 DLL들에 대해 자동으로 적용된다는 점입니다.

일반적으로는 DLL은 다음의 문서에 따라,

Dynamic-link library search order
; https://learn.microsoft.com/en-us/windows/win32/dlls/dynamic-link-library-search-order

제법 복잡한 규칙으로 찾게 됩니다. 자세한 것은 저 문서를 읽어보시고 여기서는 ^^ 저 옵션을 간단한 예제로 테스트해 보겠습니다.




Visual Studio로 C/C++ Console Application과 Dynamic-Link Library 유형의 프로젝트를 각각 생성한 다음, DLL 프로젝트에는 다음과 같이 테스트용 함수를 하나 추가하고,

// Dll1.h

#pragma once

#ifdef DLL1LIB_EXPORTS
#define DLL1LIBRARY_API __declspec(dllexport)
#else
#define DLL1LIBRARY_API __declspec(dllimport)
#endif

extern "C" DLL1LIBRARY_API void test_func();

// pch.cpp

#include "pch.h"
#include "Dll1.h"
#include <stdio.h>

void DLL1LIBRARY_API test_func()
{
    printf("test_func");
}

Console Application에서는 위의 함수를 사용하는 코드를 다음과 같이 넣어둡니다.

#include <iostream>

#include "..\\Dll1\\Dll1.h"
#pragma comment(lib, "..\\x64\\debug\\Dll1.lib")

int main()
{
    std::cout << "Hello World!\n";

    test_func();
}

빌드 하면 ./x64/Debug/ 디렉터리에 각각 다음과 같은 파일들이 생성되고,

ConsoleApplication1.exe
Dll1.dll
...[필요 없는 파일 생략]...

당연히 저 디렉터리에서 실행하면 화면에는 Hello World와 test_func가 출력됩니다. 자, 그럼 여기서 Dll1.dll 파일을 그 부모 디렉터리로 이동한 다음, 그 x64 디렉터리에서 ConsoleApplication1.exe를 실행해 보면 어떻게 될까요?

C:\ConsoleApplication1\x64> dir /b /s
C:\ConsoleApplication1\x64\Dll1.dll
C:\ConsoleApplication1\x64\Debug\ConsoleApplication1.exe
...[필요 없는 파일 생략]...

C:\ConsoleApplication1\x64> .\Debug\ConsoleApplication1.exe

ConsoleApplication1.exe가 위치한 디렉터리에 Dll1.dll 파일이 없는데도 잘 실행이 될 것입니다. 왜냐하면, "Dynamic-link library search order" 문서에 나오듯이 "Current Directory"에서도, 즉 해당 응용 프로그램을 실행한 경로에서도 dll을 찾기 때문입니다.

이제 약간씩 변형을 줘볼까요? ^^ 우선, DEPENDENTLOADFLAG 옵션으로 0xa00 값을 줄 텐데요,

// Linker / Command Line의 "Additional Options"에 명시

/DEPENDENTLOADFLAG:0xa00

문서에 0xa00은 다음의 2개 플래그를 OR 연산한 것입니다.

LOAD_LIBRARY_SEARCH_APPLICATION_DIR 0x00000200
LOAD_LIBRARY_SEARCH_SYSTEM32 0x00000800

그러니까, DLL을 c:\windows\system32 디렉터리와 exe 파일이 있는 디렉터리에서만 찾으라는 의미입니다. 이렇게 빌드하고 다시 위와 같은 상황으로 실행하면 이번에는 다음과 같은 오류가 발생합니다.

ConsoleApplication1.exe - System Error
The code execution cannot proceed because Dll1.dll was not found. Reinstalling the program may fix this problem. 

또한 이벤트 로그에는 다음과 같은 "정보" 항목이 남습니다.

Log Name:      System
Source:        Application Popup
...[생략]...
Event ID:      26
Task Category: None
Level:         Information
User:          SYSTEM
Description:
Application popup: ConsoleApplication1.exe - System Error : The code execution cannot proceed because Dll1.dll was not found. Reinstalling the program may fix this problem. 

왜냐하면, Dll1.dll 파일이 exe와 같은 디렉터리이거나, system32에만 있어야 하기 때문입니다. 대충 느낌이 오시죠? ^^ 그렇다면, 다음의 옵션(LOAD_LIBRARY_SEARCH_SYSTEM32)으로 주고 빌드하면 어떻게 될까요?

/DEPENDENTLOADFLAG:0x800

그럼 dll 파일을 오직 system32 디렉터리에서만 찾기 때문에 exe와 dll 파일이 같은 디렉터리에 있어도 실행되지 않습니다. 따라서 위의 옵션을 주었다면 의존하는 dll을 모두 system32로 복사해야만 합니다.




해보는 김에, 혹시 LOAD_LIBRARY_SEARCH_APPLICATION_DIR만 주면 어떻게 될까요?

/DEPENDENTLOADFLAG:0x200

현재 예제에서는 Visual C++ Debug 빌드를 했기 때문에 다음의 DLL들에 대한 의존성이 있습니다.

msvcp140d.dll
msvcp140_1d.dll
ucrtbased.dll
vcruntime140d.dll
vcruntime140_1d.dll

Visual Studio를 설치한 경우 위의 DLL들은 모두 c:\windows\system32 디렉터리에 있기 때문에 LOAD_LIBRARY_SEARCH_APPLICATION_DIR 옵션만으로 빌드하게 되면 위의 DLL들을 찾을 수 없어 System Error가 발생합니다. 따라서 해당 DLL들을 exe와 같은 디렉터리에 복사해,

C:\ConsoleApplication1\x64\Debug> dir /b
ConsoleApplication1.exe
Dll1.dll
msvcp140d.dll
msvcp140_1d.dll
ucrtbased.dll
vcruntime140d.dll
vcruntime140_1d.dll
...[필요 없는 파일 생략]...

실행하면 정상적으로 구동됩니다. 여기서 재미있는 것은, windows의 시스템 DLL(예를 들어, combase.dll, gdi32.dll, imm32.dll 등)은 이전처럼 system32 디렉터리에서 로드한다는 것입니다. 다시 말해 LOAD_LIBRARY_SEARCH_APPLICATION_DIR 옵션을 줘도 시스템 DLL들은 예외인 것입니다. (사실, 편의상 예외로 할 수밖에 없었을 것입니다. ^^)




비록 이 옵션의 지원은 Windows 10 Version 1607부터지만 개발자에게 있어 지금도 충분한 의미가 있습니다. 위의 설명을 이해하셨다면 눈치채셨을 텐데요, ^^ 보통, 개발된 응용 프로그램을 다른 PC에서 실행할 때 의존성 문제로 고생하는 경우가 있습니다. 바로 그럴 때, 저 옵션을 주고 실행해 보면 되는 것입니다. 최종적으로 실행이 잘 되면, 그 상태의 출력 디렉터리를 xcopy로 복사해 배포하면 끝입니다.

마지막으로 유의해야 할 점은, 저 옵션은 정적 링크된 DLL에 대해서만 통용된다는 점입니다. 코드상에서 LoadLibrary로 DLL 로딩을 시도하면, 그에 대해서는 "Dynamic-link library search order" 문서의 규칙을 적용받습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/29/2023]

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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12602정성태4/16/20219881VS.NET IDE: 161. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 [1]
12601정성태4/15/20218921.NET Framework: 1040. C# - REST API 대신 github 클라이언트 라이브러리를 통해 프로그래밍으로 접근
12600정성태4/15/20219110.NET Framework: 1039. C# - Kubeconfig의 token 설정 및 인증서 구성을 자동화하는 프로그램
12599정성태4/14/20219846.NET Framework: 1038. C# - 인증서 및 키 파일로부터 pfx/p12 파일을 생성하는 방법파일 다운로드1
12598정성태4/14/20219965.NET Framework: 1037. openssl의 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 (2)파일 다운로드1
12597정성태4/13/202110029개발 환경 구성: 569. csproj의 내용을 공통 설정할 수 있는 Directory.Build.targets / Directory.Build.props 파일
12596정성태4/12/20219771개발 환경 구성: 568. Windows의 80 포트 점유를 해제하는 방법
12595정성태4/12/20219116.NET Framework: 1036. SQL 서버 - varbinary 타입에 대한 문자열의 CAST, CONVERT 변환을 C# 코드로 구현
12594정성태4/11/20218577.NET Framework: 1035. C# - kubectl 명령어 또는 REST API 대신 Kubernetes 클라이언트 라이브러리를 통해 프로그래밍으로 접근 [1]파일 다운로드1
12593정성태4/10/20219786개발 환경 구성: 567. Docker Desktop for Windows - kubectl proxy 없이 k8s 대시보드 접근 방법
12592정성태4/10/20219636개발 환경 구성: 566. Docker Desktop for Windows - k8s dashboard의 Kubeconfig 로그인 및 Skip 방법
12591정성태4/9/202112907.NET Framework: 1034. C# - byte 배열을 Hex(16진수) 문자열로 고속 변환하는 방법 [2]파일 다운로드1
12590정성태4/9/20219373.NET Framework: 1033. C# - .NET 4.0 이하에서 Console.IsInputRedirected 구현 [1]
12589정성태4/8/202110769.NET Framework: 1032. C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법 [1]
12588정성태4/7/202111315개발 환경 구성: 565. PowerShell - New-SelfSignedCertificate를 사용해 CA 인증서 생성 및 인증서 서명 방법
12587정성태4/6/202112175개발 환경 구성: 564. Windows 10 - ClickOnce 배포처럼 사용할 수 있는 MSIX 설치 파일 [1]
12586정성태4/5/20219800오류 유형: 710. Windows - Restart-Computer / shutdown 명령어 수행 시 Access is denied(E_ACCESSDENIED)
12585정성태4/5/20219497개발 환경 구성: 563. 기본 생성된 kubeconfig 파일의 내용을 새롭게 생성한 인증서로 구성하는 방법
12584정성태4/1/202110216개발 환경 구성: 562. kubeconfig 파일 없이 kubectl 옵션만으로 실행하는 방법
12583정성태3/29/202111699개발 환경 구성: 561. kubectl 수행 시 다른 k8s 클러스터로 접속하는 방법
12582정성태3/29/202110428오류 유형: 709. Visual C++ - 컴파일 에러 error C2059: syntax error: '__stdcall'
12581정성태3/28/202110372.NET Framework: 1031. WinForm/WPF에서 Console 창을 띄워 출력하는 방법 (2) - Output 디버깅 출력을 AllocConsole로 우회 [2]
12580정성태3/28/20219015오류 유형: 708. SQL Server Management Studio - Execution Timeout Expired.
12579정성태3/28/20219150오류 유형: 707. 중첩 가상화(Nested Virtualization) - The virtual machine could not be started because this platform does not support nested virtualization.
12578정성태3/27/20219391개발 환경 구성: 560. Docker Desktop for Windows 기반의 Kubernetes 구성 (2) - WSL 2 인스턴스에 kind가 구성한 k8s 서비스 위치
12577정성태3/26/202111432개발 환경 구성: 559. Docker Desktop for Windows 기반의 Kubernetes 구성 - WSL 2 인스턴스에 kind 도구로 k8s 클러스터 구성
... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...