Introduction
As .NET is penetrating the development environment day by day, we are having unmanaged and managed code running parallel. These days, managed to unmanaged calls have become very popular. But unmanaged to managed calls are still tedious. So my aim was to make such calls as simple as possible.
Background
Unmanaged to managed calls using Regasm are very common. I have tried a straightforward call from unmanaged C++ code to a managed C# code. The main funda which I worked on are:
- Each type of .NET Framework application requires a piece of code called a Runtime host to start it. The runtime host loads the runtime into a process, creates the application domains within the process, and loads and executes user code within those application domains. This section explains how to write a runtime host that performs several fundamental tasks.
- Operating systems and runtime environments typically provide some form of isolation between applications. This isolation is necessary to ensure that code running in one application cannot adversely affect other, unrelated applications.
Application domains provide a more secure and versatile unit of processing that the common language runtime can use to provide isolation between applications. Application domains are typically created by runtime hosts, which are responsible for bootstrapping the common language runtime before an application is run.
- Unmanaged hosting code is used to configure the common language runtime, load it in the process, and transition the program into managed code. The managed portion of the hosting code creates the application domains in which the user code will run and dispatches user requests to the created application domains.
In my project, I have made an attempt to create a CLRHost to call methods of a managed DLL or EXE. I have done the steps described in the following section.
Using the code
Suppose we have a .NET DLL with a method declaration like this:
using System;
namespace ManagedDll
{
public class ManagedClass
{
public ManagedClass()
{
}
public int Add(int i, int j)
{
return(i+j);
}
}
}
Now, the C++ code which directly calls this DLL without using Regasm would be:
#include "stdafx.h"
#include <atlbase.h>
#include <mscoree.h>
#include <comutil.h>
#import "C:\\WINNT\\Microsoft.NET\\Framework\\"
"v1.1.4322\\Mscorlib.tlb" raw_interfaces_only
using namespace mscorlib;
int CallManagedFunction(char*, char*, BSTR, int,
VARIANT *, VARIANT *);
int main(int argc, char* argv[])
{
VARIANT varArgs[2] ;
varArgs[0].vt = VT_INT;
varArgs[0].intVal = 1;
varArgs[1].vt = VT_INT;
varArgs[1].intVal = 2;
VARIANT varRet;
varRet.vt = VT_INT;
int iRet = CallManagedFunction("ManagedDll",
"ManagedDll.ManagedClass",L"Add",
2,varArgs,&varRet);
if(!iRet)
printf("\nSum = %d\n",varRet.intVal);
return 0;
}
int CallManagedFunction(char* szAsseblyName,
char* szClassNameWithNamespace,BSTR szMethodName,
int iNoOfParams, VARIANT * pvArgs, VARIANT * pvRet)
{
CComPtr<ICorRuntimeHost> pRuntimeHost;
CComPtr<_AppDomain> pDefAppDomain;
try
{
HRESULT hr = CorBindToRuntimeEx(
NULL, L"wks", STARTUP_LOADER_SAFEMODE | STARTUP_CONCURRENT_GC,
CLSID_CorRuntimeHost,
IID_ICorRuntimeHost,
(void**)&pRuntimeHost
);
if (FAILED(hr)) return hr;
hr = pRuntimeHost->Start();
CComPtr<IUnknown> pUnknown;
hr = pRuntimeHost->GetDefaultDomain(&pUnknown);
if (FAILED(hr)) return hr;
hr = pUnknown->QueryInterface(&pDefAppDomain.p);
if (FAILED(hr)) return hr;
CComPtr<_ObjectHandle> pObjectHandle;
_bstr_t _bstrAssemblyName(szAsseblyName);
_bstr_t _bstrszClassNameWithNamespace(szClassNameWithNamespace);
hr = pDefAppDomain->CreateInstance(
_bstrAssemblyName,
_bstrszClassNameWithNamespace,
&pObjectHandle
);
if (FAILED(hr)) return hr;
CComVariant VntUnwrapped;
hr = pObjectHandle->Unwrap(&VntUnwrapped);
if (FAILED(hr)) return hr;
if (VntUnwrapped.vt != VT_DISPATCH)
return E_FAIL;
CComPtr<IDispatch> pDisp;
pDisp = VntUnwrapped.pdispVal;
DISPID dispid;
DISPPARAMS dispparamsArgs = {NULL, NULL, 0, 0};
dispparamsArgs.cArgs = iNoOfParams;
dispparamsArgs.rgvarg = pvArgs;
hr = pDisp->GetIDsOfNames (
IID_NULL,
&szMethodName,
1,
LOCALE_SYSTEM_DEFAULT,
&dispid
);
if (FAILED(hr)) return hr;
hr = pDisp->Invoke (
dispid,
IID_NULL,
LOCALE_SYSTEM_DEFAULT,
DISPATCH_METHOD,
&dispparamsArgs,
pvRet,
NULL,
NULL
);
if (FAILED(hr)) return hr;
pRuntimeHost->Stop();
return ERROR_SUCCESS;
}
catch(_com_error e)
{
}
}
We need to be careful about some settings like:
- The managed DLL should be in the right path, i.e., any mapped path, or inside the debug/release folder of the calling C++ project.
- In the C++ project, we need to include the path C:\PROGRAM FILES\MICROSOFT VISUAL STUDIO .NET 2003\SDK\V1.1\BIN, and C:\PROGRAM FILES\MICROSOFT VISUAL STUDIO .NET 2003\SDK\V1.1\LIB for MSCOREE.H and MSCOREE.LIB.
MSCOREE.H contains the definition of the interfaces used in this program.
Points of Interest
So it is done. Unmanaged to managed call becomes very simple, we just need to pass the namespace, class name and the method name with the arguments to be passed. If you want more fundas, click here.