Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 7개 있습니다.)
(시리즈 글이 5개 있습니다.)
.NET Framework: 431. .NET EXE 파일을 닷넷 프레임워크 버전에 상관없이 실행할 수 있을까요?
; https://www.sysnet.pe.kr/2/0/1659

.NET Framework: 461. .NET EXE 파일을 닷넷 프레임워크 버전에 상관없이 실행할 수 있을까요? - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1746

VC++: 129. EXE를 LoadLibrary로 로딩해 PE 헤더에 있는 EntryPoint를 직접 호출하는 방법
; https://www.sysnet.pe.kr/2/0/11858

디버깅 기술: 126. windbg - .NET x86 CLR2/CLR4 EXE의 EntryPoint
; https://www.sysnet.pe.kr/2/0/11861

디버깅 기술: 127. windbg - .NET x64 EXE의 EntryPoint
; https://www.sysnet.pe.kr/2/0/11863




.NET EXE 파일을 닷넷 프레임워크 버전에 상관없이 실행할 수 있을까요? - 두 번째 이야기

전에 쓴 글에서,

.NET EXE 파일을 닷넷 프레임워크 버전에 상관없이 실행할 수 있을까요?
; https://www.sysnet.pe.kr/2/0/1659

"대신... 이런 방법은 어떨까요? Visual C++로 exe를 만들고 그 안에 닷넷 EXE + app.config을 리소스로 포함하는 것입니다" 라고 잠깐 언급을 했었는데요. 생각해 보니 app.config을 포함할 필요는 없습니다. 즉, app.config없이 다음과 같은 supportedRuntime이 설정된 것처럼 동작시킬 수 있습니다.

<?xml version="1.0"?>
<configuration>
    <startup>
        <supportedRuntime version="v4.0.30319"/>
        <supportedRuntime version="v2.0.50727"/>
    </startup>
</configuration>

왜냐하면 이를 위한 2가지 조건이 이미 갖춰져 있기 때문입니다.

  • C++에서는 원하는 CLR을 로드할 수 있다.
  • CLR 4 환경에서도 .NET 2.0 대상의 어셈블리를 로드할 수 있다.

자.. 그럼 실제로 한번 해볼까요? ^^

호스팅 코드에 대해서는 이미 다음과 같이 CLR 2/4에 대해 각각 마이크로소프트에서 친절하게 예제 코드로 배포하고 있으니 이를 참조하겠습니다.

C++ app hosts CLR and invokes .NET assembly (CppHostCLR)
; http://code.msdn.microsoft.com/windowsdesktop/CppHostCLR-4da36165

C++ app hosts CLR 4 and invokes .NET assembly (CppHostCLR)
; http://code.msdn.microsoft.com/windowsdesktop/CppHostCLR-e6581ee0

우선, CLR 4 환경을 로드하는 시도를 해보겠습니다.

HRESULT hr;
ICLRMetaHost *pMetaHost = nullptr;
hr = CLRCreateInstance(CLSID_CLRMetaHost, IID_PPV_ARGS(&pMetaHost));

ICLRMetaHost 인터페이스는 .NET 4.0부터 지원하고 있기 때문에 .NET 2.0만 설치된 컴퓨터에서는 인스턴스 생성에 실패하게 됩니다. 오호~~~ 그렇다면 이것에 실패했을 때는 재차 CLR 2 환경을 초기화하려는 시도를 해보면 된다는 이야기입니다.

ICorRuntimeHost *pCorRuntimeHost = nullptr;
if (FAILED(hr))
{
    wprintf(L"CLRCreateInstance failed: 0x%x (.NET 4 not installed)\n", hr);

    // 실패한다면 CLR 2 로드를 시도
    PCWSTR pszFlavor = L"wks";
    PCWSTR pszVersion = L"v2.0.50727";

    hr = CorBindToRuntimeEx(
        pszVersion,                     // Runtime version 
        pszFlavor,                      // Flavor of the runtime to request 
        0,                              // Runtime startup flags 
        CLSID_CorRuntimeHost,           // CLSID of ICorRuntimeHost 
        IID_PPV_ARGS(&pCorRuntimeHost)  // Return ICorRuntimeHost 
        );

    if (FAILED(hr))
    {
        wprintf(L".NET 2.0 load failed\n");
        break;
    }
    else
    {
        wprintf(L".NET 2.0 loaded\n");
    }
}

위의 코드에서 ICorRuntimeHost 인터페이스를 구해오는데요. .NET 4.0 로드에서 성공했다면 ICLRMetaHost로부터 다음과 같은 절차를 거쳐서 동일한 ICorRuntimeHost 인터페이스를 구할 수 있습니다.

hr = pMetaHost->GetRuntime(L"v4.0.30319", IID_PPV_ARGS(&pRuntimeInfo));
if (FAILED(hr))
{
    wprintf(L".NET 4.0 load failed\n");
    break;
}
else
{
    wprintf(L".NET 4.0 loaded\n");
}

hr = pRuntimeInfo->GetInterface(CLSID_CorRuntimeHost, IID_PPV_ARGS(&pCorRuntimeHost));
if (FAILED(hr))
{
    wprintf(L"GetInterface(CLSID_CorRuntimeHost) failed\n");
    break;
}

여기까지 되었으면 게임 끝입니다. 컴퓨터에 설치된 CLR을 로드했으므로 Default AppDomain을 구하고,

hr = pCorRuntimeHost->Start(); // CLR 구동
if (FAILED(hr))
{
    wprintf(L"CLR failed to start\n");
    break;
}

{
    IUnknownPtr spAppDomainThunk = nullptr;
    _AssemblyPtr spAssembly = nullptr;

    hr = pCorRuntimeHost->GetDefaultDomain(&spAppDomainThunk);
    if (FAILED(hr))
    {
        wprintf(L"GetDefaultDomain failed\n");
        break;
    }

    _AppDomainPtr spDefaultAppDomain = spAppDomainThunk;
    if (spDefaultAppDomain == nullptr)
    {
        wprintf(L"Failed to get default _AppDomainPtr\n");
        break;
    }

    // AppDomain을 구했으므로 어셈블리를 로드하고, 실행하는 코드를 추가
    // ... [생략: 아래에서 설명]...
}

pCorRuntimeHost->Stop();

그 중간에 우리가 실행할 .NET EXE를 로드하고 실행하는 코드를 추가하면 됩니다. 우리가 의도하는 것은 .NET 2.0/4.0에 상관없이 실행할 수 있는 단일 EXE 파일만 배포하는 것이므로 .NET 2.0 대상으로 다음의 코드를 포함하는 C# 프로젝트를 하나 만들고,

using System;

class Program
{
    static void Main()
    {
        Console.WriteLine("Sample .NET App - running...");
    }
}

컴파일된 EXE 파일을 C++ 프로젝트에 리소스로 포함시킵니다. 포함된 리소스 바이너리는 다음과 같이 구할 수 있고,

HMODULE hModule = ::GetModuleHandle(NULL);
HRSRC hResource = FindResource(hModule, MAKEINTRESOURCE(IDR_EXEFILE1), L"EXEFILE");
HGLOBAL hMemory = LoadResource(hModule, hResource);
DWORD dwSize = SizeofResource(hModule, hResource);
LPVOID lpAddress = LockResource(hMemory);

이를 SafeArray에 담아 Main 메서드를 찾아 실행하면 됩니다.

SAFEARRAYBOUND rgsabound[] = { dwSize, 0 };
SAFEARRAY *pSafeArray = SafeArrayCreate(VT_UI1, 1, rgsabound);
pSafeArray->pvData = lpAddress;

hr = spDefaultAppDomain->Load_3(pSafeArray, &spAssembly);
/*
Critical error detected c0000374
ConsoleApplication1.exe has triggered a breakpoint.
*/
// SafeArrayDestroy(pSafeArray);

if (FAILED(hr))
{
    wprintf(L"Failed to load the assembly\n");
    break;
}

_MethodInfoPtr mainMethod;
hr = spAssembly->get_EntryPoint(&mainMethod);

if (FAILED(hr))
{
    wprintf(L"No Entry method\n");
    break;
}

VARIANT vtEmpty;
VariantInit(&vtEmpty);

BindingFlags flags = (BindingFlags)(BindingFlags::BindingFlags_InvokeMethod | BindingFlags::BindingFlags_Static);
hr = mainMethod->Invoke_2(vtEmpty, flags,
    nullptr, nullptr, nullptr, nullptr);

생각보다 어렵지 않지요? ^^

이제 C++ 프로젝트를 빌드한 단일 EXE 파일을 .NET 2.0만 설치된 컴퓨터에 복사해서 실행하면 다음과 같은 출력 결과를 얻을 수 있고,

D:\temp>ConsoleApplication1.exe
CLRCreateInstance failed: 0x80004001 (.NET 4 not installed)
.NET 2.0 loaded
Sample .NET App - running...

.NET 4.0만 설치된 컴퓨터에서도 마찬가지로 잘 실행이 되는 것을 확인할 수 있습니다.

D:\temp>ConsoleApplication1.exe
.NET 4.0 loaded
Sample .NET App - running...

오~~~ 멋집니다. ^^

게다가 .NET Framework이 아예 설치되어 있지 않다면 이후의 원하는 동작도 자유롭게 제어할 수 있는 권한도 얻게 된 것입니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)
(github에 예제 코드를 올려 두었습니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/24/2020]

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

비밀번호

댓글 작성자
 



2014-09-18 11시50분
[ryujh] 안녕하세요.
본문에서는 .net 4.0 부터 찾아서 실행해보고 다음에 .net 2.0 찾게 되어있는데, 즉 상위버전부터 찾는 것 같
.net 2.0 등 하위버전 부터 찾는 것도 가능한가요? 아마 로직을 바꾸면 가능할 것 같은데 단
.net 4.0 과 .net 2.0 이 공존할때 .net 2.0 으로 실행하는 것이 의미가 없는 것일까요?

감사합니다.
[guest]
2014-09-19 12시57분
ryujh님, 당연히 코드 순서를 바꾸면 .NET 2.0 이후 4.0을 찾게 됩니다. 단지, 최근 CLR이 더 나은 성능을 보이기 때문에 먼저 찾도록 한 것 뿐입니다.
정성태
2014-09-22 07시11분
[spowner] 헐.. 대박. 감사합니다!
[guest]
2015-02-04 12시07분
[초보] 안녕하세요. CLR 버전에 고민하던중 이 글이 정말 많은 도움이 되었습니다.
다만 제가 C++에 무지한 탓에 몇가지 문제가 생겼는데 혼자 해결하기 어렵네요.
먼저 닷넷 Main 메서드에 인자 값을 받으면 작동이 안된다는 점과
WPF에서는 시작 개체를 변경하여 Main매서드에 인자값을 받지 않게 해도
Failed to call main method가 뜨는데요.
몇일 고민을 해봐도 도무지 모르겠습니다..
혹시 해결방법을 아신다면 답변부탁드려도 될까요?
감사합니다. :)
[guest]
2015-02-04 12시19분
그럼, Main 함수를 바로 부르지 말고 차라리 별도로 자신만의 .NET Assembly를 하나 만들어서 그걸 부르게 한 다음, 그 안에서 다시 원하는 닷넷 Main 메서드를 인자를 가지고 부르게 바꿔보세요. 차라리 그게 더 구현하기 쉬울 것입니다.
정성태
2023-07-17 09시41분
How small is the smallest .NET Hello World binary?
; https://blog.washi.dev/posts/tinysharp/
정성태

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13600정성태4/18/2024271닷넷: 2242. C# - 관리 스레드와 비관리 스레드
13599정성태4/17/2024287닷넷: 2241. C# - WAV 파일의 PCM 사운드 재생(Windows Multimedia)파일 다운로드1
13598정성태4/16/2024309닷넷: 2240. C# - WAV 파일 포맷 + LIST 헤더파일 다운로드1
13597정성태4/15/2024376닷넷: 2239. C# - WAV 파일의 PCM 데이터 생성 및 출력파일 다운로드1
13596정성태4/14/2024752닷넷: 2238. C# - WAV 기본 파일 포맷파일 다운로드1
13595정성태4/13/2024874닷넷: 2237. C# - Audio 장치 열기 (Windows Multimedia, NAudio)파일 다운로드1
13594정성태4/12/20241007닷넷: 2236. C# - Audio 장치 열람 (Windows Multimedia, NAudio)파일 다운로드1
13593정성태4/8/20241050닷넷: 2235. MSBuild - AccelerateBuildsInVisualStudio 옵션
13592정성태4/2/20241206C/C++: 165. CLion으로 만든 Rust Win32 DLL을 C#과 연동
13591정성태4/2/20241167닷넷: 2234. C# - WPF 응용 프로그램에 Blazor App 통합파일 다운로드1
13590정성태3/31/20241072Linux: 70. Python - uwsgi 응용 프로그램이 k8s 환경에서 OOM 발생하는 문제
13589정성태3/29/20241142닷넷: 2233. C# - 프로세스 CPU 사용량을 나타내는 성능 카운터와 Win32 API파일 다운로드1
13588정성태3/28/20241195닷넷: 2232. C# - Unity + 닷넷 App(WinForms/WPF) 간의 Named Pipe 통신파일 다운로드1
13587정성태3/27/20241154오류 유형: 900. Windows Update 오류 - 8024402C, 80070643
13586정성태3/27/20241296Windows: 263. Windows - 복구 파티션(Recovery Partition) 용량을 늘리는 방법
13585정성태3/26/20241094Windows: 262. PerformanceCounter의 InstanceName에 pid를 추가한 "Process V2"
13584정성태3/26/20241047개발 환경 구성: 708. Unity3D - C# Windows Forms / WPF Application에 통합하는 방법파일 다운로드1
13583정성태3/25/20241156Windows: 261. CPU Utilization이 100% 넘는 경우를 성능 카운터로 확인하는 방법
13582정성태3/19/20241417Windows: 260. CPU 사용률을 나타내는 2가지 수치 - 사용량(Usage)과 활용률(Utilization)파일 다운로드1
13581정성태3/18/20241586개발 환경 구성: 707. 빌드한 Unity3D 프로그램을 C++ Windows Application에 통합하는 방법
13580정성태3/15/20241136닷넷: 2231. C# - ReceiveTimeout, SendTimeout이 적용되지 않는 Socket await 비동기 호출파일 다운로드1
13579정성태3/13/20241493오류 유형: 899. HTTP Error 500.32 - ANCM Failed to Load dll
13578정성태3/11/20241628닷넷: 2230. C# - 덮어쓰기 가능한 환형 큐 (Circular queue)파일 다운로드1
13577정성태3/9/20241874닷넷: 2229. C# - 닷넷을 위한 난독화 도구 소개 (예: ConfuserEx)
13576정성태3/8/20241543닷넷: 2228. .NET Profiler - IMetaDataEmit2::DefineMethodSpec 사용법
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...