Click here to Skip to main content
Click here to Skip to main content

Unmanaged to Managed calls (C++ to C#) without Regasm

, 10 Jan 2006
Rate this:
Please Sign up or sign in to vote.
A simple way to call a managed method from the unmanaged world.

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:

// ExposeManageCode.cpp : Defines the entry point
//                        for the console application.
//
// REF : MSDN : Accessing Members Through IDispatch

#include "stdafx.h"
#include <atlbase.h>
#include <mscoree.h>
#include <comutil.h>

// Need to be modified as your directory settings.
#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;
    //Calling manageddll.dll Add() method.
    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
    {
        //Retrieve a pointer to the ICorRuntimeHost interface
        HRESULT hr = CorBindToRuntimeEx(
            NULL,    //Specify the version 
                     //of the runtime that will be loaded. 
            L"wks",  //Indicate whether the server
                     // or workstation build should be loaded.
            //Control whether concurrent
            //or non-concurrent garbage collection
            //Control whether assemblies are loaded as domain-neutral. 
            STARTUP_LOADER_SAFEMODE | STARTUP_CONCURRENT_GC, 
            CLSID_CorRuntimeHost,
            IID_ICorRuntimeHost,
            //Obtain an interface pointer to ICorRuntimeHost 
            (void**)&pRuntimeHost
            );
        
        if (FAILED(hr)) return hr;
        
        //Start the CLR
        hr = pRuntimeHost->Start();
        
        CComPtr<IUnknown> pUnknown;
        
        //Retrieve the IUnknown default AppDomain
        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);
        
        //Creates an instance of the Assembly
        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;
        
        //Invoke the method on the Dispatch Interface
        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)
    {
        //Exception handling.
    }
}

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.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

Share

About the Author

Chakrabarty Rajib
Web Developer
United States United States
Rajib is one of the many altruist guy working with Cognizant Technology Solution ... Smile | :)

Comments and Discussions

 
QuestionThank you PinmemberMember 81680188-Aug-12 15:51 
QuestionHi, Can you help me with my issue? PinmemberSteve Ween20-Jul-12 3:38 
Questionthis is stupid way to call the c# dll just add the c# dll as a reference to the win32 project with /clr option Pinmembera ahole26-Jun-11 16:47 
Questionhow to pass a string to c# from vc++ using the above function Pinmemberkindi19-Jul-10 0:44 
AnswerRe: how to pass a string to c# from vc++ using the above function Pinmemberbaek hum kyung17-Aug-12 19:43 
Questionhow would the code change if the constructor for the C# app took arguments? Pinmemberronn kling21-Apr-10 10:34 
QuestionHow to call C# DLL functions from Delphi Pinmembervijay.victory14-Apr-10 3:01 
GeneralWhy it doesn't work if the C# method is static Pinmembermarvinmenaleal11-Jun-09 4:25 
GeneralInteresting information on arguments. PinmemberDoug071620-Mar-09 9:51 
GeneralRe: Interesting information on arguments. PinmemberHabib Ahmed2-Aug-12 4:57 
Generalgetting errors - please help urgent Pinmembersanjeevkum5-Mar-09 4:44 
GeneralRe: getting errors - please help urgent PinmemberdrQyH16j876K23-Jun-09 15:01 
GeneralQuestion of calling managed dll from unmanaged c++ [modified] - Please help- Pinmembersanjeevkum4-Mar-09 17:38 
GeneralUnmanaged to Managed, how to pass data by reference PinmemberMustafa Alahwel19-Feb-09 4:03 
Hi Rajib,
 
Very helpful example, good job.
 
I have a C# DLL that has the following function:
 
void XferData( ref byte[] pdata, ref int len)
{
}
 
In my unmanaged C++ app, I have the following byte buffer:
 
int len = 256;
BYTE* pBuf = new BYTE[len];
 
How do I pass this to the managed C# DLL using CallManagedFunction()in your unmanaged C++ example?
 
Thanks for the help.
GeneralRe: Unmanaged to Managed, how to pass data by reference PinmemberDoug071620-Mar-09 9:30 
QuestionHow about performance? Pinmemberaiflame330-Jan-09 5:28 
QuestionHow to do the same thing with BOOL without parameter? PinmemberMohd Audy12-Jan-09 15:50 
GeneralCool! [modified] PinmemberRabbit6331-Oct-08 9:41 
Generalcompilation error with this code PinmemberMember 448196331-Oct-08 6:58 
GeneralRe: compilation error with this code Pinmembermenakerman1-Dec-08 21:33 
QuestionHow to do the same thing but with byte[]? Pinmembermringholm22-Sep-08 5:18 
QuestionExample works but... [modified] Pinmemberapalomba15-Sep-08 10:43 
AnswerRe: Example works but... PinmemberChakrabarty Rajib5-Oct-08 4:04 
QuestionThanks and a remark - release pointers? Pinmembertb200011-Mar-08 9:16 
AnswerRe: Thanks and a remark - release pointers? PinmemberChakrabarty Rajib5-Oct-08 4:08 
QuestionDifferent Directory PinmemberCSpicer21-Feb-08 10:17 
GeneralGood Work! Pinmemberadmcse8-Jan-08 5:01 
QuestionWhy it doesn't work? [modified] Pinmembercaimodorro12-Feb-07 0:15 
AnswerRe: Why it doesn't work? PinmemberHristo Konstantinov2-Mar-07 9:22 
GeneralRe: Why it doesn't work? Pinmembercaimodorro4-Mar-07 19:49 
GeneralUse unmenage C lib on C# PinmemberBandoleiro11-Dec-06 0:24 
GeneralDebuggin... PinmembertheDarkBrianer30-Jul-06 5:58 
GeneralSay thanks PinmemberAlexEvans22-Jan-06 18:17 
GeneralUnmanged to managed calls PinmemberSeverian@Severian.org19-Jan-06 20:38 
GeneralRe: Unmanged to managed calls PinmemberClaus Brod22-Jan-06 6:03 
GeneralRe: Unmanged to managed calls PinmemberAntony M Kancidrowski13-Jun-06 6:38 
Generalperformance? and what about managed C++ Pinmembernuno8918-Jan-06 16:55 
GeneralRe: performance? and what about managed C++ Pinmemberischen_s118-Jan-06 20:53 
GeneralRe: performance? and what about managed C++ PinmemberAndromeda Shun18-Jan-06 20:53 
GeneralExtending this to Visual Basic 6 Pinmembergoldsun18-Jan-06 13:48 
GeneralThis is nuts PinmemberAndrei Dumitrache17-Jan-06 22:19 
GeneralRe: This is nuts PinmemberDasbose19-Jan-06 4:17 
GeneralRe: This is nuts PinmemberNagy Pityu19-Jan-06 6:26 
GeneralRe: This is nuts PinmemberAndrei Dumitrache19-Jan-06 22:34 
QuestionMind boggling! More on pointers? PinmemberKoushik Biswas11-Jan-06 15:52 
AnswerRe: Mind boggling! More on pointers? PinmemberChakrabarty Rajib13-Jan-06 2:47 
GeneralInteresting PinmemberStephen Hewitt10-Jan-06 15:56 
GeneralRe: Interesting PinmemberChakrabarty Rajib10-Jan-06 19:17 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

| Advertise | Privacy | Mobile
Web04 | 2.8.140814.1 | Last Updated 10 Jan 2006
Article Copyright 2006 by Chakrabarty Rajib
Everything else Copyright © CodeProject, 1999-2014
Terms of Service
Layout: fixed | fluid