Реализация поставщиков MSSCCI в .Net - PullRequest
3 голосов
/ 06 июля 2010

Я хотел бы реализовать поставщика MSSCCI, однако, если это вообще возможно, я хотел бы реализовать его в .Net (поэтому мой поставщик MSSCCI фактически является тонкой оболочкой для реализации .Net)

  • Возможно ли это?
  • Это хорошая идея?

Я знаю, что реализация этого в .Net будет означать, что любой, использующий мой поставщик MSSCCI, будет вынужден принять.Net Framework внутри их процесса - это необоснованный запрос?Кроме того, какие еще ограничения мне нужно учитывать, если я собираюсь реализовать это в .Net?

1 Ответ

5 голосов
/ 14 ноября 2010

Это возможно и относительно просто.Я разработал один некоторое время назад, и он отлично работает.Я использовал COM-совместимость от C ++ до C # .

Итак, вы создадите две библиотеки DLL. C ++ one является только оболочкой, которая реализует API, передающий вызовы COM в C # .C # one должен быть зарегистрирован как COM компонент с regasm / codebase mycomlibrary.dll

Вот несколько рекомендаций по его реализации.В примере кода я только реализую функцию SccInitialize в качестве примера.Надеюсь, что это поможет.

Это компонент C ++ :

#include <comutil.h>

/**********************************************************************************************************/
// Imports the COM object that implements the SCC API in .NET
/**********************************************************************************************************/
#import "SccCOMServer.tlb" no_namespace named_guids

static int s_nInitializedCount = 0;

/**********************************************************************************************************/
// Starting point of the dll
/**********************************************************************************************************/
BOOL APIENTRY DllMain( 
                      HANDLE hModule, 
                      DWORD  ul_reason_for_call, 
                      LPVOID lpReserved
                      )
{
    switch (ul_reason_for_call)
    {
        case DLL_PROCESS_ATTACH:
        case DLL_THREAD_ATTACH:
        case DLL_THREAD_DETACH:
        case DLL_PROCESS_DETACH:
        break;
    }
    return TRUE;
}

/**********************************************************************************************************/
// Variable with a instance of the COM object
/**********************************************************************************************************/

ISccCOMServer *mCpi = NULL;


/**********************************************************************************************************/
// Utility functions
/**********************************************************************************************************/

void BSTR2T(BSTR s1, LPSTR s2)
{
    _bstr_t s(s1, false);
    strcpy(s2, s);
}

char* ConvertBSTRToLPSTR (BSTR bstrIn)
{
    LPSTR pszOut = NULL;
    if (bstrIn != NULL)
    {
        int nInputStrLen = SysStringLen (bstrIn);

        // Double NULL Termination
        int nOutputStrLen = WideCharToMultiByte(CP_ACP, 0, bstrIn, nInputStrLen, NULL, 0, 0, 0) + 2; 
        pszOut = new char [nOutputStrLen];

        if (pszOut)
        {
            memset (pszOut, 0x00, sizeof (char)*nOutputStrLen);
            WideCharToMultiByte (CP_ACP, 0, bstrIn, nInputStrLen, pszOut, nOutputStrLen, 0, 0);
        }
    }
    return pszOut;
}

 /**********************************************************************************************************/
//                                      IMPLEMENTATION OF THE FUNCTIONS
/**********************************************************************************************************/


/**********************************************************************************************************/
// Initialization and Housekeepeng Functions
/**********************************************************************************************************/

SCCEXTERNC SCCRTN EXTFUN __cdecl SccInitialize(
    LPVOID * ppContext, 
    HWND hWnd, 
    LPCSTR lpCallerName,
    LPSTR lpSccName, // [In, out]
    LPLONG lpSccCaps, // [Out]
    LPSTR lpAuxPathLabel, // [In, out]
    LPLONG pnCheckoutCommentLen, // [Out]
    LPLONG pnCommentLen //[Out]
    )
{

    // Initialize COM the first time the function is called
    CoInitialize(0);
    s_nInitializedCount++;
    HRESULT hr = CoCreateInstance(CLSID_ISccCOMServerImpl,
        NULL, CLSCTX_INPROC_SERVER,
        IID_ISccCOMServer, reinterpret_cast<void**>(&mCpi));

    long response;

    // We need auxiliar strings because out string in COM are BSTR *  
    BSTR bstrSccName;
    BSTR bstrAuxPathLabel;

    bstrSccName = T2BSTR(lpSccName);
    bstrAuxPathLabel = T2BSTR(lpAuxPathLabel);

    Context *CC = new Context;
    // Calling to the COM equivalent Function

    response = mCpi->Initialize(CC, (long) hWnd, lpCallerName, &bstrSccName, lpSccCaps, &bstrAuxPathLabel, 
        pnCheckoutCommentLen, pnCommentLen);

    *ppContext = (void *)CC;

    // Converting the strings
    BSTR2T(bstrSccName, lpSccName);
    BSTR2T(bstrAuxPathLabel, lpAuxPathLabel);
    return response;

}

И тогда часть C # проще:

[Guid("C6659361-1625-4746-931C-36014B146679")]
public class ISccCOMServerImpl : ISccCOMServer
{
    public int Initialize(
        out Context ppContext,
        IntPtr hWnd,
        string lpCallerName,
        ref string lpSccName, // out
        out int lpSccCaps, // out
        ref string lpAuxPathLabel, // out
        out int pnCheckoutCommentLen, // out
        out int pnCommentLen //out
        )
    {
       //your manage code here!
    }

}

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...