C ++ ClrHost Invoke_3返回错误80131604 TargetInvocationException

时间:2018-07-24 16:04:19

标签: c++ .net clr clr-hosting

我需要将.net可执行文件加载到非托管C ++可执行文件中 我在C ++中使用clrhost来做到这一点 我可以简单地使用反射从C#加载.net应用程序 像这样:

namespace WindowsFormsApplication1
{
  static class Program
  {
    [STAThread]
    static void Main(string[] args)
    {
      byte[] data = { bytes of the .net executable };
      System.Reflection.Assembly.Load(data).EntryPoint.Invoke(null, null);
    } 
  }
}

但是c ++中的等效代码不起作用:

ICLRMetaHost* pMetaHost = NULL;

HRESULT hr;

/* Get ICLRMetaHost instance */

hr = CLRCreateInstance(CLSID_CLRMetaHost, IID_ICLRMetaHost, (VOID**)&pMetaHost);

if (FAILED(hr))
{
    printf("[!] CLRCreateInstance(...) failed\n");

    getchar();

    return -1;
}

printf("[+] CLRCreateInstance(...) succeeded\n");

ICLRRuntimeInfo* pRuntimeInfo = NULL;

/* Get ICLRRuntimeInfo instance */

hr = pMetaHost->GetRuntime(L"v4.0.30319", IID_ICLRRuntimeInfo, (VOID**)&pRuntimeInfo);

if (FAILED(hr))
{
    printf("[!] pMetaHost->GetRuntime(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pMetaHost->GetRuntime(...) succeeded\n");

BOOL bLoadable;

/* Check if the specified runtime can be loaded */

hr = pRuntimeInfo->IsLoadable(&bLoadable);

if (FAILED(hr) || !bLoadable)
{
    printf("[!] pRuntimeInfo->IsLoadable(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pRuntimeInfo->IsLoadable(...) succeeded\n");

ICorRuntimeHost* pRuntimeHost = NULL;

/* Get ICorRuntimeHost instance */

hr = pRuntimeInfo->GetInterface(CLSID_CorRuntimeHost, IID_ICorRuntimeHost, (VOID**)&pRuntimeHost);

if (FAILED(hr))
{
    printf("[!] pRuntimeInfo->GetInterface(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pRuntimeInfo->GetInterface(...) succeeded\n");

/* Start the CLR */

hr = pRuntimeHost->Start();

if (FAILED(hr))
{
    printf("[!] pRuntimeHost->Start() failed\n");

    getchar();

    return -1;
}

printf("[+] pRuntimeHost->Start() succeeded\n");

IUnknownPtr pAppDomainThunk = NULL;

hr = pRuntimeHost->GetDefaultDomain(&pAppDomainThunk);

if (FAILED(hr))
{
    printf("[!] pRuntimeHost->GetDefaultDomain(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pRuntimeHost->GetDefaultDomain(...) succeeded\n");

_AppDomainPtr pDefaultAppDomain = NULL;

/* Equivalent of System.AppDomain.CurrentDomain in C# */

hr = pAppDomainThunk->QueryInterface(__uuidof(_AppDomain), (VOID**)&pDefaultAppDomain);

if (FAILED(hr))
{
    printf("[!] pAppDomainThunk->QueryInterface(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pAppDomainThunk->QueryInterface(...) succeeded\n");

_AssemblyPtr pAssembly = NULL;

SAFEARRAYBOUND rgsabound[1];

rgsabound[0].cElements = RAW_ASSEMBLY_LENGTH;

rgsabound[0].lLbound = 0;

SAFEARRAY* pSafeArray = SafeArrayCreate(VT_UI1, 1, rgsabound);

void* pvData = NULL;

hr = SafeArrayAccessData(pSafeArray, &pvData);

if (FAILED(hr))
{
    printf("[!] SafeArrayAccessData(...) failed\n");

    getchar();

    return -1;
}

printf("[+] SafeArrayAccessData(...) succeeded\n");

memcpy(pvData, rawData, RAW_ASSEMBLY_LENGTH);

hr = SafeArrayUnaccessData(pSafeArray);

if (FAILED(hr))
{
    printf("[!] SafeArrayUnaccessData(...) failed\n");

    getchar();

    return -1;
}

printf("[+] SafeArrayUnaccessData(...) succeeded\n");

/* Equivalent of System.AppDomain.CurrentDomain.Load(byte[] rawAssembly) */
hr = pDefaultAppDomain->Load_3(pSafeArray, &pAssembly);

if (FAILED(hr))
{
    printf("[!] pDefaultAppDomain->Load_3(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pDefaultAppDomain->Load_3(...) succeeded\n");

_MethodInfoPtr pMethodInfo = NULL;

/* Assembly.EntryPoint Property */

hr = pAssembly->get_EntryPoint(&pMethodInfo);

if (FAILED(hr))
{
    printf("[!] pAssembly->get_EntryPoint(...) failed\n");

    getchar();

    return -1;
}

printf("[+] pAssembly->get_EntryPoint(...) succeeded\n");

VARIANT retVal;
ZeroMemory(&retVal, sizeof(VARIANT));

VARIANT obj;
ZeroMemory(&obj, sizeof(VARIANT));
obj.vt = VT_NULL;

//TODO! Change cElement to the number of Main arguments
SAFEARRAY *psaStaticMethodArgs = SafeArrayCreateVector(VT_VARIANT, 0, 0); // in c# (null,null) , main doesn't have parameters
hr = pMethodInfo->Invoke_3(obj, psaStaticMethodArgs, &retVal);
if (FAILED(hr))
    {
        _com_error err(hr);
        std::wcout << L"[!] failed with : " << err.ErrorMessage() << std::endl;
        printf("[!] pMethodInfo->Invoke_3(...) failed, hr = %X\n", hr);
        return -1;
    }
    printf("[+] pMethodInfo->Invoke_3(...) succeeded with parameters number : %X\n",i);

失败发生在Invoke_3中,该错误返回80131604 TargetInvocationException并崩溃

有什么我想念的吗?

0 个答案:

没有答案