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)
13526정성태1/14/20242048닷넷: 2201. C# - Facebook 연동 / 사용자 탈퇴 처리 방법
13525정성태1/13/20242018오류 유형: 891. Visual Studio - Web Application을 실행하지 못하는 IISExpress
13524정성태1/12/20242066오류 유형: 890. 한국투자증권 KIS Developers OpenAPI - GW라우팅 중 오류가 발생했습니다.
13523정성태1/12/20241888오류 유형: 889. Visual Studio - error : A project with that name is already opened in the solution.
13522정성태1/11/20242027닷넷: 2200. C# - HttpClient.PostAsJsonAsync 호출 시 "Transfer-Encoding: chunked" 대신 "Content-Length" 헤더 처리
13521정성태1/11/20242108닷넷: 2199. C# - 한국투자증권 KIS Developers OpenAPI의 WebSocket Ping, Pong 처리
13520정성태1/10/20241858오류 유형: 888. C# - Unable to resolve service for type 'Microsoft.Extensions.ObjectPool.ObjectPool`....'
13519정성태1/10/20241937닷넷: 2198. C# - Reflection을 이용한 ClientWebSocket의 Ping 호출파일 다운로드1
13518정성태1/9/20242174닷넷: 2197. C# - ClientWebSocket의 Ping, Pong 처리
13517정성태1/8/20242016스크립트: 63. Python - 공개 패키지를 이용한 위성 이미지 생성 (pystac_client, odc.stac)
13516정성태1/7/20242103닷넷: 2196. IIS - AppPool의 "Disable Overlapped Recycle" 옵션의 부작용
13515정성태1/6/20242374닷넷: 2195. async 메서드 내에서 C# 7의 discard 구문 활용 사례 [1]
13514정성태1/5/20242065개발 환경 구성: 702. IIS - AppPool의 "Disable Overlapped Recycle" 옵션
13513정성태1/5/20242004닷넷: 2194. C# - WebActivatorEx / System.Web의 PreApplicationStartMethod 특성
13512정성태1/4/20241979개발 환경 구성: 701. IIS - w3wp.exe 프로세스의 ASP.NET 런타임을 항상 Warmup 모드로 유지하는 preload Enabled 설정
13511정성태1/4/20241995닷넷: 2193. C# - ASP.NET Web Application + OpenAPI(Swashbuckle) 스펙 제공
13510정성태1/3/20241937닷넷: 2192. C# - 특정 실행 파일이 있는지 확인하는 방법 (Linux)
13509정성태1/3/20241966오류 유형: 887. .NET Core 2 이하의 프로젝트에서 System.Runtime.CompilerServices.Unsafe doesn't support netcoreapp2.0.
13508정성태1/3/20242013오류 유형: 886. ORA-28000: the account is locked
13507정성태1/2/20242696닷넷: 2191. C# - IPGlobalProperties를 이용해 netstat처럼 사용 중인 Socket 목록 구하는 방법파일 다운로드1
13506정성태12/29/20232189닷넷: 2190. C# - 닷넷 코어/5+에서 달라지는 System.Text.Encoding 지원
13505정성태12/27/20232704닷넷: 2189. C# - WebSocket 클라이언트를 닷넷으로 구현하는 예제 (System.Net.WebSockets)파일 다운로드1
13504정성태12/27/20232321닷넷: 2188. C# - ASP.NET Core SignalR로 구현하는 채팅 서비스 예제파일 다운로드1
13503정성태12/27/20232190Linux: 67. WSL 환경 + mlocate(locate) 도구의 /mnt 디렉터리 검색 문제
13502정성태12/26/20232290닷넷: 2187. C# - 다른 프로세스의 환경변수 읽는 예제파일 다운로드1
13501정성태12/25/20232087개발 환경 구성: 700. WSL + uwsgi - IPv6로 바인딩하는 방법
1  2  3  [4]  5  6  7  8  9  10  11  12  13  14  15  ...