|
|
//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1995 - 1999
//
// File: csdisp.cpp
//
// Contents: IDispatch helper functions
//
// History: 09-Dec-96 vich created
//
//--------------------------------------------------------------------------
#include <pch.cpp>
#pragma hdrstop
#include <stdlib.h>
#include "csdisp.h"
#define __dwFILE__ __dwFILE_CERTLIB_DISPATCH_CPP__
#ifndef DBG_DISPATCH
# define DBG_DISPATCH 0
#endif
TCHAR szRegKeyClsidValue[] = TEXT("");
HRESULT DispatchGetReturnValue( VARIANT *pvar, IN LONG Type, OUT VOID *pretval) { HRESULT hr = S_OK;
if (VT_EMPTY == pvar->vt) { hr = CERTSRV_E_PROPERTY_EMPTY; _JumpError2(hr, error, "VT_EMPTY", hr); } if (Type != pvar->vt) { DBGPRINT(( DBG_SS_CERTLIB, "pvar->vt=%x, expected %x\n", pvar->vt, Type)); hr = TYPE_E_WRONGTYPEKIND; _JumpError(hr, error, "pvar->vt != Type"); } switch (Type) { case VT_I4: *(LONG *) pretval = pvar->lVal; break;
case VT_DATE: *(DATE *) pretval = pvar->date; break;
case VT_BSTR: *(BSTR *) pretval = pvar->bstrVal; pvar->vt = VT_EMPTY; break;
case VT_DISPATCH: *(IDispatch **) pretval = pvar->pdispVal; pvar->vt = VT_EMPTY; break;
default: hr = E_INVALIDARG; goto error; }
error: return(hr); }
HRESULT DispatchSetErrorInfoSub( IN HRESULT hrError, OPTIONAL IN WCHAR const *pwszIDispatchMethod, OPTIONAL IN WCHAR const *pwszDescription, OPTIONAL IN WCHAR const *pwszSource, OPTIONAL IN IID const *piid, OPTIONAL IN WCHAR const *pwszHelpFile, IN DWORD dwHelpFileContext) { HRESULT hr; WCHAR const *pwszError = NULL; ICreateErrorInfo *pCreateErrorInfo = NULL; IErrorInfo *pErrorInfo = NULL;
#ifdef DBG_CERTSRV_DEBUG_PRINT
if (NULL != pwszIDispatchMethod || DbgIsSSActive(DBG_SS_CERTLIBI)) #else
if (NULL != pwszIDispatchMethod) #endif
{ pwszError = myGetErrorMessageText(hrError, TRUE); if (NULL != pwszIDispatchMethod) { CONSOLEPRINT1(( MAXDWORD, "IDispatch->Invoke(%ws) Exception:\n", pwszIDispatchMethod)); } else { CONSOLEPRINT0((MAXDWORD, "COM Error:\n")); }
CONSOLEPRINT5(( MAXDWORD, " Source=%ws\n Description=%ws\n HelpFile=%ws[%x]\n %ws\n", pwszSource, pwszDescription, pwszHelpFile, dwHelpFileContext, pwszError)); }
hr = CreateErrorInfo(&pCreateErrorInfo); _JumpIfError(hr, error, "CreateErrorInfo");
if (NULL != piid) { hr = pCreateErrorInfo->SetGUID(*piid); _PrintIfError(hr, "SetGUID"); } if (NULL != pwszSource) { hr = pCreateErrorInfo->SetSource(const_cast<WCHAR *>(pwszSource)); _PrintIfError(hr, "SetSource"); } if (NULL != pwszDescription) { hr = pCreateErrorInfo->SetDescription( const_cast<WCHAR *>(pwszDescription)); _PrintIfError(hr, "SetDescription"); } if (NULL != pwszHelpFile) { hr = pCreateErrorInfo->SetHelpFile(const_cast<WCHAR *>(pwszHelpFile)); _PrintIfError(hr, "SetHelpFile");
hr = pCreateErrorInfo->SetHelpContext(dwHelpFileContext); _PrintIfError(hr, "SetHelpContext"); }
hr = pCreateErrorInfo->QueryInterface( IID_IErrorInfo, (VOID **) &pErrorInfo); _JumpIfError(hr, error, "QueryInterface");
SetErrorInfo(0, pErrorInfo); hr = S_OK;
error: if (NULL != pwszError) { LocalFree(const_cast<WCHAR *>(pwszError)); } if (NULL != pErrorInfo) { pErrorInfo->Release(); } if (NULL != pCreateErrorInfo) { pCreateErrorInfo->Release(); } return(hr); }
HRESULT DispatchInvoke( IN DISPATCHINTERFACE *pDispatchInterface, IN LONG MethodIndex, IN DWORD cvar, IN VARIANT avar[], IN LONG Type, OUT VOID *pretval) { HRESULT hr; DISPATCHTABLE const *pdt; EXCEPINFO excepinfo; VARIANT varResult; DISPPARAMS parms; unsigned iArgErr; DISPID *adispid;
VariantInit(&varResult); CSASSERT(NULL != pDispatchInterface->pDispatchTable); CSASSERT((DWORD) MethodIndex < pDispatchInterface->m_cDispatchTable); pdt = &pDispatchInterface->pDispatchTable[MethodIndex];
adispid = &pDispatchInterface->m_adispid[pdt->idispid]; CSASSERT(NULL != pDispatchInterface->m_adispid); CSASSERT(pdt->idispid + pdt->cdispid <= pDispatchInterface->m_cdispid);
parms.rgvarg = avar; parms.rgdispidNamedArgs = &adispid[1]; parms.cArgs = pdt->cdispid - 1; parms.cNamedArgs = parms.cArgs;
if (parms.cArgs != cvar) { CSASSERT(parms.cArgs == cvar); hr = HRESULT_FROM_WIN32(ERROR_INTERNAL_ERROR); _JumpError(hr, error, "cvar"); }
CSASSERT(NULL != pDispatchInterface->pDispatch);
hr = pDispatchInterface->pDispatch->Invoke( adispid[0], IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_METHOD, &parms, &varResult, &excepinfo, &iArgErr);
if (S_OK != hr) { _PrintErrorStr(hr, "Invoke", pdt->apszNames[0]);
if (DISP_E_EXCEPTION == hr) { HRESULT hr2;
if (FAILED(excepinfo.scode)) { hr = excepinfo.scode; } hr2 = DispatchSetErrorInfoSub( hr, pdt->apszNames[0], excepinfo.bstrDescription, excepinfo.bstrSource, pDispatchInterface->GetIID(), excepinfo.bstrHelpFile, excepinfo.dwHelpContext); _PrintIfError(hr2, "DispatchSetErrorInfoSub"); } goto error; }
if (CLSCTX_INPROC_SERVER != pDispatchInterface->m_ClassContext) { myRegisterMemAlloc(&varResult, 0, CSM_VARIANT); } if (NULL != pretval) { hr = DispatchGetReturnValue(&varResult, Type, pretval); _JumpIfErrorStr2( hr, error, "DispatchGetReturnValue", pdt->apszNames[0], CERTSRV_E_PROPERTY_EMPTY); }
error: VariantClear(&varResult); return(hr); }
HRESULT DispatchGetIds( IN IDispatch *pDispatch, IN DWORD cDispatchTable, IN OUT DISPATCHTABLE *pDispatchTable, IN OUT DISPATCHINTERFACE *pDispatchInterface) { HRESULT hr = S_OK; DISPATCHTABLE *pdt; DISPATCHTABLE *pdtEnd; DISPID *adispid = NULL; DWORD idispid;
pdtEnd = &pDispatchTable[cDispatchTable]; pDispatchInterface->m_cdispid = 0; for (pdt = pDispatchTable; pdt < pdtEnd; pdt++) { if (0 == pdt->idispid) { pdt->idispid = pDispatchInterface->m_cdispid; } CSASSERT(pdt->idispid == pDispatchInterface->m_cdispid); pDispatchInterface->m_cdispid += pdt->cdispid; }
adispid = (DISPID *) LocalAlloc( LMEM_FIXED, pDispatchInterface->m_cdispid * sizeof(adispid[0])); if (NULL == adispid) { hr = E_OUTOFMEMORY; _JumpError(hr, error, "LocalAlloc"); }
idispid = 0; for (pdt = pDispatchTable; pdt < pdtEnd; pdt++) { CSASSERT(idispid + pdt->cdispid <= pDispatchInterface->m_cdispid); hr = pDispatch->GetIDsOfNames( IID_NULL, pdt->apszNames, pdt->cdispid, LOCALE_SYSTEM_DEFAULT, &adispid[idispid]); #if DBG_CERTSRV
if (S_OK != hr || CSExpr(DBG_DISPATCH)) { DWORD i;
DBGPRINT(( MAXDWORD, "GetIDsOfNames(%ws) --> %x, dispid=%x" szLPAREN, pdt->apszNames[0], hr, adispid[idispid])); for (i = 1; i < pdt->cdispid; i++) { DBGPRINT(( MAXDWORD, "%ws%x", i == 1? L"" : L", ", adispid[idispid + i])); } DBGPRINT((MAXDWORD, szRPAREN "\n")); } #endif
_JumpIfError(hr, error, "GetIDsOfNames");
idispid += pdt->cdispid; } pDispatchInterface->m_cDispatchTable = cDispatchTable; pDispatchInterface->pDispatchTable = pDispatchTable; pDispatchInterface->m_adispid = adispid; adispid = NULL;
error: if (NULL != adispid) { LocalFree(adispid); } return(hr); }
HRESULT DispatchSetup( IN DWORD Flags, IN DWORD ClassContext, OPTIONAL IN TCHAR const *pszProgID, // for IDispatch
OPTIONAL IN CLSID const *pclsid, // for COM
OPTIONAL IN IID const *piid, // for COM
IN DWORD cDispatchTable, // for IDispatch
IN OUT DISPATCHTABLE *pDispatchTable, IN OUT DISPATCHINTERFACE *pDispatchInterface) { HRESULT hr; CLSID clsid; pDispatchInterface->SetIID(piid); for (;;) { pDispatchInterface->pDispatch = NULL; pDispatchInterface->pUnknown = NULL; pDispatchInterface->m_adispid = NULL;
CSASSERT(NULL != pszProgID || DISPSETUP_COM == Flags); CSASSERT(NULL != pclsid || DISPSETUP_IDISPATCH == Flags); CSASSERT(NULL != piid || DISPSETUP_IDISPATCH == Flags);
if (DISPSETUP_IDISPATCH == Flags) { // use win32 version, not our own hack
hr = CLSIDFromProgID(pszProgID, &clsid);
_JumpIfError2( hr, error, "ClassNameToCLSID", HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND));
pclsid = &clsid; piid = &IID_IDispatch; }
hr = CoCreateInstance( *pclsid, NULL, // pUnkOuter
ClassContext, *piid, DISPSETUP_IDISPATCH == Flags? (VOID **) &pDispatchInterface->pDispatch : (VOID **) &pDispatchInterface->pUnknown); #if DBG_DISPATCH
printf("CoCreateInstance(%x) --> %x\n", Flags, hr); #endif
if (S_OK != hr) { _PrintError2(hr, "CoCreateInstance", E_NOINTERFACE); if (DISPSETUP_COMFIRST != Flags) { _JumpError2(hr, error, "CoCreateInstance", E_NOINTERFACE); } Flags = DISPSETUP_IDISPATCH; continue; } pDispatchInterface->m_ClassContext = ClassContext; break; } if (DISPSETUP_IDISPATCH == Flags) { hr = DispatchGetIds( pDispatchInterface->pDispatch, cDispatchTable, pDispatchTable, pDispatchInterface); if (S_OK != hr) { DispatchRelease(pDispatchInterface); _JumpError(hr, error, "DispatchGetIds"); } }
error: return(hr); }
HRESULT DispatchSetup2( IN DWORD Flags, IN DWORD ClassContext, IN WCHAR const *pwszClass, // wszRegKeyAdminClsid
IN CLSID const *pclsid, IN DWORD cver, IN IID const * const *ppiid, // cver elements
IN DWORD const *pcDispatch, // cver elements
IN OUT DISPATCHTABLE *pDispatchTable, IN OUT DISPATCHINTERFACE *pDispatchInterface) { HRESULT hr = E_INVALIDARG;
CSASSERT(0 < cver); for ( ; 0 < cver; cver--, ppiid++, pcDispatch++) { hr = DispatchSetup( Flags, ClassContext, pwszClass, pclsid, *ppiid, *pcDispatch, pDispatchTable, pDispatchInterface); if (S_OK == hr) { pDispatchInterface->m_dwVersion = cver; pDispatchInterface->pDispatchTable = pDispatchTable; break; } if (1 == cver) { _JumpErrorStr(hr, error, "DispatchSetup", pwszClass); } _PrintErrorStr2(hr, "DispatchSetup", pwszClass, hr); }
error: return(hr); }
VOID DispatchRelease( IN OUT DISPATCHINTERFACE *pDispatchInterface) { if (NULL != pDispatchInterface->pDispatch) { pDispatchInterface->pDispatch->Release(); pDispatchInterface->pDispatch = NULL; } if (NULL != pDispatchInterface->pUnknown) { pDispatchInterface->pUnknown->Release(); pDispatchInterface->pUnknown = NULL; } if (NULL != pDispatchInterface->m_adispid) { LocalFree(pDispatchInterface->m_adispid); pDispatchInterface->m_adispid = NULL; } }
BOOL myConvertWszToBstr( OUT BSTR *pbstr, IN WCHAR const *pwc, IN LONG cb) { HRESULT hr; BSTR bstr;
bstr = NULL; if (NULL != pwc) { if (-1 == cb) { cb = wcslen(pwc) * sizeof(WCHAR); } bstr = SysAllocStringByteLen((char const *) pwc, cb); if (NULL == bstr) { hr = E_OUTOFMEMORY; _JumpError(hr, error, "SysAllocStringByteLen"); } } if (NULL != *pbstr) { SysFreeString(*pbstr); } *pbstr = bstr; hr = S_OK;
error: if (S_OK != hr) { SetLastError(hr); } return(S_OK == hr); }
BOOL myConvertSzToBstr( OUT BSTR *pbstr, IN CHAR const *pch, IN LONG cch) { HRESULT hr; BSTR bstr = NULL; LONG cwc = 0;
if (-1 == cch) { cch = strlen(pch); } CSASSERT(0 != cch); for (;;) { cwc = MultiByteToWideChar(GetACP(), 0, pch, cch, bstr, cwc); if (0 >= cwc) { hr = myHLastError(); _PrintError(hr, "MultiByteToWideChar"); if (NULL != bstr) { SysFreeString(bstr); } break; } if (NULL != bstr) { bstr[cwc] = L'\0'; if (NULL != *pbstr) { SysFreeString(*pbstr); } *pbstr = bstr; hr = S_OK; break; } bstr = SysAllocStringLen(NULL, cwc); if (NULL == bstr) { hr = E_OUTOFMEMORY; break; } } if (S_OK != hr) { SetLastError(hr); } return(S_OK == hr); }
HRESULT DecodeCertString( IN BSTR const bstrIn, IN DWORD Flags, OUT BYTE **ppbOut, OUT DWORD *pcbOut) { HRESULT hr; DWORD dwSize;
*ppbOut = NULL;
CSASSERT(CSExpr(CR_IN_BASE64HEADER == CRYPT_STRING_BASE64HEADER)); CSASSERT(CSExpr(CR_IN_BASE64 == CRYPT_STRING_BASE64)); CSASSERT(CSExpr(CR_IN_BINARY == CRYPT_STRING_BINARY));
switch (Flags) { case CRYPT_STRING_BASE64HEADER: case CRYPT_STRING_BASE64: case CRYPT_STRING_BASE64REQUESTHEADER: case CRYPT_STRING_HEX: case CRYPT_STRING_HEXASCII: case CRYPT_STRING_BASE64_ANY: case CRYPT_STRING_HEX_ANY: case CRYPT_STRING_BASE64X509CRLHEADER: case CRYPT_STRING_HEXADDR: case CRYPT_STRING_HEXASCIIADDR: CSASSERT(sizeof(WCHAR) * wcslen(bstrIn) == SysStringByteLen(bstrIn)); // FALLTHROUGH
case CRYPT_STRING_ANY: dwSize = (SysStringByteLen(bstrIn) + sizeof(WCHAR) - 1) / sizeof(WCHAR); break;
case CRYPT_STRING_BINARY: dwSize = SysStringByteLen(bstrIn); break;
default: hr = E_INVALIDARG; _JumpError(hr, error, "Flags"); } hr = myCryptStringToBinary( bstrIn, dwSize, Flags, ppbOut, pcbOut, NULL, NULL); _JumpIfError(hr, error, "myCryptStringToBinary");
if (CRYPT_STRING_ANY == Flags && (SysStringByteLen(bstrIn) & (sizeof(WCHAR) - 1))) { (*pcbOut)--; CSASSERT(SysStringByteLen(bstrIn) == *pcbOut); } hr = S_OK;
error: return(hr); }
HRESULT EncodeCertString( IN BYTE const *pbIn, IN DWORD cbIn, IN DWORD Flags, OUT BSTR *pbstrOut) { HRESULT hr = E_INVALIDARG; WCHAR *pwcCert = NULL; DWORD cbCert;
CSASSERT(CSExpr(CR_OUT_BASE64HEADER == CRYPT_STRING_BASE64HEADER)); CSASSERT(CSExpr(CR_OUT_BASE64 == CRYPT_STRING_BASE64)); CSASSERT(CSExpr(CR_OUT_BINARY == CRYPT_STRING_BINARY)); //CSASSERT(CSExpr(CR_OUT_BASE64REQUESTHEADER == CRYPT_STRING_BASE64REQUESTHEADER));
//CSASSERT(CSExpr(CR_OUT_HEX == CRYPT_STRING_HEX));
CSASSERT(CSExpr(CV_OUT_BASE64HEADER == CRYPT_STRING_BASE64HEADER)); CSASSERT(CSExpr(CV_OUT_BASE64 == CRYPT_STRING_BASE64)); CSASSERT(CSExpr(CV_OUT_BINARY == CRYPT_STRING_BINARY)); CSASSERT(CSExpr(CV_OUT_BASE64REQUESTHEADER == CRYPT_STRING_BASE64REQUESTHEADER)); CSASSERT(CSExpr(CV_OUT_HEX == CRYPT_STRING_HEX)); CSASSERT(CSExpr(CV_OUT_HEXASCII == CRYPT_STRING_HEXASCII)); CSASSERT(CSExpr(CV_OUT_BASE64X509CRLHEADER == CRYPT_STRING_BASE64X509CRLHEADER)); CSASSERT(CSExpr(CV_OUT_HEXADDR == CRYPT_STRING_HEXADDR)); CSASSERT(CSExpr(CV_OUT_HEXASCIIADDR == CRYPT_STRING_HEXASCIIADDR));
if (NULL == pbIn || (~CR_OUT_ENCODEMASK & Flags)) { goto error; }
switch (CR_OUT_ENCODEMASK & Flags) { case CRYPT_STRING_BASE64HEADER: case CRYPT_STRING_BASE64: case CRYPT_STRING_BASE64REQUESTHEADER: case CRYPT_STRING_HEX: case CRYPT_STRING_HEXASCII: case CRYPT_STRING_BASE64X509CRLHEADER: case CRYPT_STRING_HEXADDR: case CRYPT_STRING_HEXASCIIADDR: hr = myCryptBinaryToString( pbIn, cbIn, CR_OUT_ENCODEMASK & Flags, &pwcCert); _JumpIfError(hr, error, "myCryptBinaryToString");
cbCert = sizeof(WCHAR) * wcslen(pwcCert); break;
case CRYPT_STRING_BINARY: pwcCert = (WCHAR *) pbIn; cbCert = cbIn; hr = S_OK; break;
default: goto error; } if (!ConvertWszToBstr(pbstrOut, pwcCert, cbCert)) { hr = E_OUTOFMEMORY; }
hr = S_OK; error: if (NULL != pwcCert && pwcCert != (WCHAR *) pbIn) { LocalFree(pwcCert); } return(hr); }
HRESULT DispatchSetErrorInfo( IN HRESULT hrError, IN WCHAR const *pwszDescription, OPTIONAL IN WCHAR const *pwszProgId, OPTIONAL IN IID const *piid) { HRESULT hr; WCHAR const *pwszError = NULL; WCHAR *pwszText = NULL;
if (NULL == pwszDescription) { hr = E_POINTER; _JumpError(hr, error, "NULL pointer"); } CSASSERT(FAILED(hrError)); pwszError = myGetErrorMessageText(hrError, TRUE); if (NULL == pwszError) { _PrintError(E_OUTOFMEMORY, "myGetErrorMessageText"); } else { pwszText = (WCHAR *) LocalAlloc( LMEM_FIXED, (wcslen(pwszDescription) + 1 + wcslen(pwszError) + 1) * sizeof(WCHAR)); if (NULL == pwszText) { _PrintError(E_OUTOFMEMORY, "LocalAlloc"); } else { wcscpy(pwszText, pwszDescription); wcscat(pwszText, L" "); wcscat(pwszText, pwszError); } } hr = DispatchSetErrorInfoSub( hrError, NULL, // pwszIDispatchMethod
NULL != pwszText? pwszText : const_cast<WCHAR *>(pwszDescription), pwszProgId, piid, NULL, // pwszHelpFile
0); // dwHelpFileContext
_PrintIfError(hr, "DispatchSetErrorInfoSub");
error: if (NULL != pwszText) { LocalFree(pwszText); } if (NULL != pwszError) { LocalFree(const_cast<WCHAR *>(pwszError)); } return(hrError); // return input error!
}
|