You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
722 lines
21 KiB
722 lines
21 KiB
/*
|
|
* oleutil.c - OLE utility functions module.
|
|
*/
|
|
|
|
|
|
/* Headers
|
|
**********/
|
|
|
|
#include "project.h"
|
|
#pragma hdrstop
|
|
|
|
#include "oleutil.h"
|
|
|
|
|
|
/* Macros
|
|
*********/
|
|
|
|
/* macro for translating an HRESULT to a TWINRESULT */
|
|
|
|
#define HRESULTToTWINRESULT(hr, TR) case hr: tr = TR; break
|
|
|
|
|
|
/* Constants
|
|
************/
|
|
|
|
/* maximum allowed registry lengths */
|
|
|
|
#define MAX_REG_KEY_LEN MAX_PATH_LEN
|
|
#define MAX_REG_VALUE_LEN MAX_PATH_LEN
|
|
|
|
/* subkeys and associated lengths */
|
|
|
|
#define CLSID_SUBKEY TEXT("CLSID")
|
|
/* CLSID subkey length in bytes, including null terminator */
|
|
#define CLSID_SUBKEY_LEN (5 + 1)
|
|
|
|
#define IN_PROC_SERVER_SUBKEY TEXT("InProcServer32")
|
|
/* InProcServer32 subkey length in bytes, including null terminator */
|
|
#define IN_PROC_SERVER_SUBKEY_LEN (14 + 1)
|
|
|
|
#define LOCAL_SERVER_SUBKEY TEXT("LocalServer32")
|
|
/* LocalServer32 subkey length in bytes, including null terminator */
|
|
#define LOCAL_SERVER_SUBKEY_LEN (13 + 1)
|
|
|
|
#define RECONCILER_SUBKEY TEXT("Roles\\Reconciler")
|
|
/* InProcServer32 subkey length in bytes, including null terminator */
|
|
#define RECONCILER_SUBKEY_LEN (5 + 1 + 10 + 1)
|
|
|
|
#define NOTIFY_SUBKEY TEXT("Roles\\NotifyReplica")
|
|
/* InProcServer32 subkey length in bytes, including null terminator */
|
|
#define NOTIFY_SUBKEY_LEN (5 + 1 + 13 + 1)
|
|
|
|
#define COPY_SUBKEY TEXT("SingleChangeHook")
|
|
/* copy subkey length in bytes, including null terminator */
|
|
#define COPY_SUBKEY_LEN (16 + 1)
|
|
|
|
/*
|
|
* length of GUID subkey string in bytes, including null terminator
|
|
*
|
|
* "{12345678-1234-1234-1234-123456789012}"
|
|
*/
|
|
|
|
#define GUID_SUBKEY_LEN (1 + 8 + 1 + 4 + 1 + 4 + 1 + 4 + 1 + 12 + 1 + 1)
|
|
|
|
/*
|
|
* class ID key length in bytes, including null terminator
|
|
*
|
|
* "CLSID\{12345678-1234-1234-1234-123456789012}"
|
|
*/
|
|
|
|
#define CLSID_REG_KEY_LEN (CLSID_SUBKEY_LEN + GUID_SUBKEY_LEN)
|
|
|
|
/*
|
|
* InProcServer32 key length in bytes, including null terminator
|
|
*
|
|
* "CLSID\{12345678-1234-1234-1234-123456789012}\InProcServer32"
|
|
*/
|
|
|
|
#define IN_PROC_SERVER_REG_KEY_LEN (CLSID_REG_KEY_LEN + IN_PROC_SERVER_SUBKEY_LEN)
|
|
|
|
/*
|
|
* LocalServer32 key length in bytes, including null terminator
|
|
*
|
|
* "CLSID\{12345678-1234-1234-1234-123456789012}\LocalServer32"
|
|
*/
|
|
|
|
#define LOCAL_SERVER_REG_KEY_LEN (CLSID_REG_KEY_LEN + LOCAL_SERVER_SUBKEY_LEN)
|
|
|
|
/*
|
|
* reconciler key length in bytes, including null terminator
|
|
*
|
|
* "CLSID\{12345678-1234-1234-1234-123456789012}\Roles\Reconciler"
|
|
*/
|
|
|
|
#define RECONCILER_REG_KEY_LEN (CLSID_REG_KEY_LEN + RECONCILER_SUBKEY_LEN)
|
|
|
|
/*
|
|
* notify replica key length in bytes, including null terminator
|
|
*
|
|
* "CLSID\{12345678-1234-1234-1234-123456789012}\Roles\NotifyReplica"
|
|
*/
|
|
|
|
#define NOTIFY_REG_KEY_LEN (CLSID_REG_KEY_LEN + NOTIFY_SUBKEY_LEN)
|
|
|
|
/*
|
|
* copy key length in bytes, including null terminator
|
|
*
|
|
* "CLSID\{12345678-1234-1234-1234-123456789012}\SingleChangeHook"
|
|
*/
|
|
|
|
#define COPY_REG_KEY_LEN (CLSID_REG_KEY_LEN + COPY_SUBKEY_LEN)
|
|
|
|
|
|
/* Macros
|
|
*********/
|
|
|
|
/*
|
|
* Determine whether or not an integer value is within a given inclusive range.
|
|
*/
|
|
|
|
#define IsWithin(test, first, last) ((UINT)((test) - (first)) <= (UINT)((last) - (first)))
|
|
|
|
|
|
/***************************** Private Functions *****************************/
|
|
|
|
/* Module Prototypes
|
|
********************/
|
|
|
|
PRIVATE_CODE void MakeClsIDSubKey(PCGUID, LPCTSTR, LPTSTR, int);
|
|
PRIVATE_CODE BOOL HexStringToDWORD(LPCTSTR *, PDWORD, UINT, TCHAR);
|
|
PRIVATE_CODE BOOL StringToGUID(LPCTSTR, PGUID);
|
|
PRIVATE_CODE HRESULT GetClassID(LPCTSTR, LPCTSTR, PCLSID);
|
|
|
|
|
|
/*
|
|
** MakeClsIDSubKey()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE void MakeClsIDSubKey(PCGUID pcguid, LPCTSTR pcszSubKey,
|
|
LPTSTR pszRegKeyBuf, int cchMax)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pcguid, CGUID));
|
|
ASSERT(IS_VALID_STRING_PTR(pcszSubKey, CSTR));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszRegKeyBuf, STR, CLSID_REG_KEY_LEN + lstrlen(pcszSubKey) + 1));
|
|
|
|
/* (- 1) for null terminator. */
|
|
|
|
EVAL(wnsprintf(pszRegKeyBuf, cchMax,
|
|
TEXT("CLSID\\{%08lX-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}\\%s"),
|
|
pcguid->Data1,
|
|
pcguid->Data2,
|
|
pcguid->Data3,
|
|
pcguid->Data4[0],
|
|
pcguid->Data4[1],
|
|
pcguid->Data4[2],
|
|
pcguid->Data4[3],
|
|
pcguid->Data4[4],
|
|
pcguid->Data4[5],
|
|
pcguid->Data4[6],
|
|
pcguid->Data4[7],
|
|
pcszSubKey)
|
|
== CLSID_REG_KEY_LEN + lstrlen(pcszSubKey));
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/*
|
|
** HexStringToDWORD()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL HexStringToDWORD(LPCTSTR *ppcsz, PDWORD pdwValue, UINT ucDigits,
|
|
TCHAR chDelimiter)
|
|
{
|
|
BOOL bResult = TRUE;
|
|
UINT u;
|
|
|
|
/* chDelimiter may be any value. */
|
|
|
|
ASSERT(IS_VALID_WRITE_PTR(ppcsz, LPCTSTR));
|
|
ASSERT(IS_VALID_STRING_PTR(*ppcsz, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwValue, DWORD));
|
|
ASSERT(ucDigits <= 8);
|
|
|
|
*pdwValue = 0;
|
|
|
|
for (u = 0; u < ucDigits; u++)
|
|
{
|
|
TCHAR ch = (*ppcsz)[u];
|
|
|
|
if (IsWithin(ch, TEXT('0'), TEXT('9')))
|
|
*pdwValue = (*pdwValue << 4) + ch - TEXT('0');
|
|
else if (IsWithin((ch |= (TEXT('a') - TEXT('A'))), TEXT('a'), TEXT('f')))
|
|
*pdwValue = (*pdwValue << 4) + ch - TEXT('a') + 10;
|
|
else
|
|
{
|
|
WARNING_OUT((TEXT("HexStringToDWORD(): Found unrecognized hex digit %c."),
|
|
ch));
|
|
|
|
bResult = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (bResult)
|
|
{
|
|
if (chDelimiter)
|
|
{
|
|
bResult = ((*ppcsz)[u++] == chDelimiter);
|
|
|
|
if (! bResult)
|
|
WARNING_OUT((TEXT("HexStringToDWORD(): Character %c does not match required delimiter %c."),
|
|
(*ppcsz)[u],
|
|
chDelimiter));
|
|
}
|
|
|
|
*ppcsz += u;
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRING_PTR(*ppcsz, CSTR));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** StringToGUID()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL StringToGUID(LPCTSTR pcszGUID, PGUID pguid)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
DWORD dwValue;
|
|
LPCTSTR pcszNext = pcszGUID;
|
|
|
|
ASSERT(IS_VALID_STRING_PTR(pcszGUID, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pguid, GUID));
|
|
|
|
if (*pcszNext++ == TEXT('{') &&
|
|
HexStringToDWORD(&pcszNext, &(pguid->Data1), sizeof(DWORD) * 2, TEXT('-')) &&
|
|
HexStringToDWORD(&pcszNext, &dwValue, sizeof(WORD) * 2, TEXT('-')))
|
|
{
|
|
pguid->Data2 = (WORD)dwValue;
|
|
|
|
if (HexStringToDWORD(&pcszNext, &dwValue, sizeof(WORD) * 2, TEXT('-')))
|
|
{
|
|
UINT u;
|
|
static const TCHAR SrgcchDelimiters[] = { 0, TEXT('-'), 0, 0, 0, 0, 0, TEXT('}') };
|
|
|
|
pguid->Data3 = (WORD)dwValue;
|
|
|
|
bResult = TRUE;
|
|
|
|
for (u = 0; u < ARRAY_ELEMENTS(pguid->Data4); u++)
|
|
{
|
|
if (HexStringToDWORD(&pcszNext, &dwValue, sizeof(BYTE) * 2,
|
|
SrgcchDelimiters[u]))
|
|
pguid->Data4[u] = (BYTE)dwValue;
|
|
else
|
|
{
|
|
bResult = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (bResult)
|
|
{
|
|
ASSERT(u == ARRAY_ELEMENTS(pguid->Data4));
|
|
|
|
if (*pcszNext)
|
|
{
|
|
bResult = FALSE;
|
|
|
|
WARNING_OUT((TEXT("StringToGUID(): Found %c instead of }."),
|
|
*pcszNext));
|
|
}
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("StringToGUID(): Bad GUID string %s."),
|
|
pcszGUID));
|
|
}
|
|
}
|
|
|
|
ASSERT(! bResult ||
|
|
IS_VALID_STRUCT_PTR(pguid, CGUID));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetClassID()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE HRESULT GetClassID(LPCTSTR pcszPath, LPCTSTR pcszSubKey, PCLSID pclsid)
|
|
{
|
|
HRESULT hr;
|
|
CLSID clsidFile;
|
|
|
|
ASSERT(IS_VALID_STRING_PTR(pcszPath, CSTR));
|
|
ASSERT(IS_VALID_STRING_PTR(pcszSubKey, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pclsid, CLSID));
|
|
|
|
/* Verify string length constants. */
|
|
|
|
/* (+ 1) for each null terminator. */
|
|
|
|
ASSERT(lstrlen(CLSID_SUBKEY) + 1 == CLSID_SUBKEY_LEN);
|
|
ASSERT(lstrlen(IN_PROC_SERVER_SUBKEY) + 1 == IN_PROC_SERVER_SUBKEY_LEN);
|
|
ASSERT(lstrlen(LOCAL_SERVER_SUBKEY) + 1 == LOCAL_SERVER_SUBKEY_LEN);
|
|
ASSERT(lstrlen(RECONCILER_SUBKEY) + 1 == RECONCILER_SUBKEY_LEN);
|
|
ASSERT(lstrlen(NOTIFY_SUBKEY) + 1 == NOTIFY_SUBKEY_LEN);
|
|
ASSERT(lstrlen(COPY_SUBKEY) + 1 == COPY_SUBKEY_LEN);
|
|
|
|
ASSERT(lstrlen( TEXT("{12345678-1234-1234-1234-123456789012}")) + 1 == GUID_SUBKEY_LEN);
|
|
ASSERT(lstrlen(TEXT("CLSID\\{12345678-1234-1234-1234-123456789012}")) + 1 == CLSID_REG_KEY_LEN);
|
|
ASSERT(lstrlen(TEXT("CLSID\\{12345678-1234-1234-1234-123456789012}\\InProcServer32")) + 1 == IN_PROC_SERVER_REG_KEY_LEN);
|
|
ASSERT(lstrlen(TEXT("CLSID\\{12345678-1234-1234-1234-123456789012}\\LocalServer32")) + 1 == LOCAL_SERVER_REG_KEY_LEN);
|
|
ASSERT(lstrlen(TEXT("CLSID\\{12345678-1234-1234-1234-123456789012}\\Roles\\Reconciler")) + 1 == RECONCILER_REG_KEY_LEN);
|
|
ASSERT(lstrlen(TEXT("CLSID\\{12345678-1234-1234-1234-123456789012}\\Roles\\NotifyReplica")) + 1 == NOTIFY_REG_KEY_LEN);
|
|
ASSERT(lstrlen(TEXT("CLSID\\{12345678-1234-1234-1234-123456789012}\\SingleChangeHook")) + 1 == COPY_REG_KEY_LEN);
|
|
|
|
hr = GetClassFileByExtension(pcszPath, &clsidFile);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
TCHAR rgchRecRegKey[MAX_REG_KEY_LEN];
|
|
TCHAR rgchRecGUID[GUID_SUBKEY_LEN];
|
|
DWORD dwcbLen = sizeof(rgchRecGUID);
|
|
|
|
hr = REGDB_E_CLASSNOTREG;
|
|
|
|
MakeClsIDSubKey(&clsidFile, pcszSubKey, rgchRecRegKey, ARRAYSIZE(rgchRecRegKey));
|
|
|
|
if (GetDefaultRegKeyValue(HKEY_CLASSES_ROOT, rgchRecRegKey, rgchRecGUID,
|
|
&dwcbLen) == ERROR_SUCCESS)
|
|
{
|
|
/* (+ 1) for null terminator. */
|
|
|
|
ASSERT((DWORD)(lstrlen(rgchRecGUID) + 1) * sizeof(TCHAR) == dwcbLen);
|
|
ASSERT(dwcbLen * sizeof(TCHAR) <= sizeof(rgchRecGUID));
|
|
|
|
if (StringToGUID(rgchRecGUID, pclsid))
|
|
{
|
|
hr = S_OK;
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
TCHAR rgchInProcServerKey[IN_PROC_SERVER_REG_KEY_LEN];
|
|
TCHAR rgchInProcServerValue[MAX_REG_VALUE_LEN];
|
|
DWORD dwcbInProcLen = sizeof(rgchInProcServerValue);
|
|
|
|
/* Display the path to the registered reconciler. */
|
|
|
|
wnsprintf(rgchInProcServerKey, ARRAYSIZE(rgchInProcServerKey), TEXT("%s\\%s\\%s"), CLSID_SUBKEY,
|
|
rgchRecGUID, IN_PROC_SERVER_SUBKEY);
|
|
|
|
if (GetDefaultRegKeyValue(HKEY_CLASSES_ROOT,
|
|
rgchInProcServerKey,
|
|
rgchInProcServerValue, &dwcbInProcLen)
|
|
== ERROR_SUCCESS)
|
|
TRACE_OUT((TEXT("GetClassID(): Found registered %s reconciler %s for file %s."),
|
|
IN_PROC_SERVER_SUBKEY,
|
|
rgchInProcServerValue,
|
|
pcszPath));
|
|
else
|
|
{
|
|
dwcbInProcLen = sizeof(rgchInProcServerValue);
|
|
|
|
wnsprintf(rgchInProcServerKey, ARRAYSIZE(rgchInProcServerKey), TEXT("%s\\%s\\%s"), CLSID_SUBKEY,
|
|
rgchRecGUID, LOCAL_SERVER_SUBKEY);
|
|
|
|
if (GetDefaultRegKeyValue(HKEY_CLASSES_ROOT,
|
|
rgchInProcServerKey,
|
|
rgchInProcServerValue,
|
|
&dwcbInProcLen)
|
|
== ERROR_SUCCESS)
|
|
TRACE_OUT((TEXT("GetClassID(): Found registered %s reconciler %s for file %s."),
|
|
LOCAL_SERVER_SUBKEY,
|
|
rgchInProcServerValue,
|
|
pcszPath));
|
|
else
|
|
WARNING_OUT((TEXT("GetClassID(): Unregistered class ID %s listed as reconciler in %s for file %s."),
|
|
rgchRecGUID,
|
|
rgchRecRegKey,
|
|
pcszPath));
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetClassID(): Bad reconciler class ID %s specified for key %s."),
|
|
rgchRecGUID,
|
|
rgchRecRegKey));
|
|
}
|
|
else
|
|
TRACE_OUT((TEXT("GetClassID(): No reconciler registered for file %s."),
|
|
pcszPath));
|
|
}
|
|
else
|
|
TRACE_OUT((TEXT("GetClassID(): No class ID registered for file %s."),
|
|
pcszPath));
|
|
|
|
ASSERT(FAILED(hr) ||
|
|
IS_VALID_STRUCT_PTR(pclsid, CCLSID));
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
/****************************** Public Functions *****************************/
|
|
|
|
|
|
/*
|
|
** GetClassFileByExtension()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE HRESULT GetClassFileByExtension(LPCTSTR pcszFile, PCLSID pclsid)
|
|
{
|
|
HRESULT hr = MK_E_INVALIDEXTENSION;
|
|
LPCTSTR pcszExtension;
|
|
|
|
ASSERT(IS_VALID_STRING_PTR(pcszFile, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pclsid, CLSID));
|
|
|
|
pcszExtension = ExtractExtension(pcszFile);
|
|
|
|
if (*pcszExtension)
|
|
{
|
|
TCHAR rgchFileType[MAX_REG_VALUE_LEN];
|
|
DWORD dwcbBufLen = sizeof(rgchFileType);
|
|
|
|
ASSERT(*pcszExtension == PERIOD);
|
|
|
|
if (GetDefaultRegKeyValue(HKEY_CLASSES_ROOT, pcszExtension, rgchFileType,
|
|
&dwcbBufLen) == ERROR_SUCCESS)
|
|
{
|
|
/* dwcbBufLen includes rgchFileType's null terminator. */
|
|
|
|
if (dwcbBufLen + sizeof(CLSID_SUBKEY) <= sizeof(rgchFileType))
|
|
{
|
|
TCHAR rgchFileClsID[GUID_SUBKEY_LEN];
|
|
DWORD dwcbFileClsIDBufLen = sizeof(rgchFileClsID);
|
|
|
|
CatPath(rgchFileType, CLSID_SUBKEY, ARRAYSIZE(rgchFileType));
|
|
|
|
if (GetDefaultRegKeyValue(HKEY_CLASSES_ROOT, rgchFileType,
|
|
rgchFileClsID, &dwcbFileClsIDBufLen)
|
|
== ERROR_SUCCESS)
|
|
{
|
|
if (StringToGUID(rgchFileClsID, pclsid))
|
|
{
|
|
hr = S_OK;
|
|
|
|
TRACE_OUT((TEXT("GetClassFileByExtension(): Retrieved class ID %s for file %s."),
|
|
rgchFileClsID,
|
|
pcszFile));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetClassFileByExtension(): Invalid class ID \"%s\" in %s\\%s. No class ID will be used."),
|
|
rgchFileClsID,
|
|
rgchFileType,
|
|
CLSID_SUBKEY));
|
|
}
|
|
else
|
|
TRACE_OUT((TEXT("GetClassFileByExtension(): No %s subkey for file type key \"%s\". No class ID will be used."),
|
|
CLSID_SUBKEY,
|
|
rgchFileType));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetClassFileByExtension(): File type \"%s\" for extension %s of file %s is too long. No class ID will be used."),
|
|
rgchFileType,
|
|
pcszExtension,
|
|
pcszFile));
|
|
}
|
|
else
|
|
TRACE_OUT((TEXT("GetClassFileByExtension(): No file type for extension %s of file %s. No class ID will be used."),
|
|
pcszExtension,
|
|
pcszFile));
|
|
}
|
|
else
|
|
TRACE_OUT((TEXT("GetClassFileByExtension(): File %s has no extension. No class ID will be used."),
|
|
pcszFile));
|
|
|
|
ASSERT(FAILED(hr) ||
|
|
IS_VALID_STRUCT_PTR(pclsid, CCLSID));
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetReconcilerClassID()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE HRESULT GetReconcilerClassID(LPCTSTR pcszPath, PCLSID pclsid)
|
|
{
|
|
ASSERT(IS_VALID_STRING_PTR(pcszPath, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pclsid, CLSID));
|
|
|
|
return(GetClassID(pcszPath, RECONCILER_SUBKEY, pclsid));
|
|
}
|
|
|
|
|
|
/*
|
|
** GetCopyHandlerClassID()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE HRESULT GetCopyHandlerClassID(LPCTSTR pcszPath, PCLSID pclsid)
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT(IS_VALID_STRING_PTR(pcszPath, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pclsid, CLSID));
|
|
|
|
hr = GetReconcilerClassID(pcszPath, pclsid);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
TCHAR rgchCopyRegKey[COPY_REG_KEY_LEN];
|
|
|
|
MakeClsIDSubKey(pclsid, COPY_SUBKEY, rgchCopyRegKey, ARRAYSIZE(rgchCopyRegKey));
|
|
|
|
if (RegKeyExists(HKEY_CLASSES_ROOT, rgchCopyRegKey))
|
|
hr = S_OK;
|
|
else
|
|
hr = REGDB_E_CLASSNOTREG;
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetReplicaNotificationClassID()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE HRESULT GetReplicaNotificationClassID(LPCTSTR pcszPath, PCLSID pclsid)
|
|
{
|
|
ASSERT(IS_VALID_STRING_PTR(pcszPath, CSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pclsid, CLSID));
|
|
|
|
return(GetClassID(pcszPath, NOTIFY_SUBKEY, pclsid));
|
|
}
|
|
|
|
|
|
/*
|
|
** CompareGUIDs()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE COMPARISONRESULT CompareGUIDs(PCGUID pcguid1, PCGUID pcguid2)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pcguid1, CGUID));
|
|
ASSERT(IS_VALID_STRUCT_PTR(pcguid2, CGUID));
|
|
|
|
return(MyMemComp(pcguid1, pcguid2, sizeof(*pcguid1)));
|
|
}
|
|
|
|
|
|
/*
|
|
** TranslateHRESULTToTWINRESULT()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE TWINRESULT TranslateHRESULTToTWINRESULT(HRESULT hr)
|
|
{
|
|
TWINRESULT tr;
|
|
|
|
switch (hr)
|
|
{
|
|
HRESULTToTWINRESULT(S_OK, TR_SUCCESS);
|
|
|
|
HRESULTToTWINRESULT(REC_S_NOTCOMPLETE, TR_MERGE_INCOMPLETE);
|
|
HRESULTToTWINRESULT(REC_S_NOTCOMPLETEBUTPROPAGATE, TR_MERGE_INCOMPLETE);
|
|
|
|
HRESULTToTWINRESULT(E_ABORT, TR_ABORT);
|
|
HRESULTToTWINRESULT(E_OUTOFMEMORY, TR_OUT_OF_MEMORY);
|
|
HRESULTToTWINRESULT(E_FAIL, TR_RH_LOAD_FAILED);
|
|
|
|
HRESULTToTWINRESULT(REC_E_ABORTED, TR_ABORT);
|
|
HRESULTToTWINRESULT(REC_E_TOODIFFERENT, TR_TOO_DIFFERENT);
|
|
|
|
default:
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
tr = TR_SUCCESS;
|
|
|
|
WARNING_OUT((TEXT("TranslateHRESULTToTWINRESULT(): Translating unlisted success HRESULT %s into TWINRESULT %s."),
|
|
GetHRESULTString(hr),
|
|
GetTWINRESULTString(tr)));
|
|
}
|
|
else
|
|
{
|
|
tr = TR_RH_LOAD_FAILED;
|
|
|
|
WARNING_OUT((TEXT("TranslateHRESULTToTWINRESULT(): Translating unlisted failure HRESULT %s into TWINRESULT %s."),
|
|
GetHRESULTString(hr),
|
|
GetTWINRESULTString(tr)));
|
|
}
|
|
break;
|
|
}
|
|
|
|
return(tr);
|
|
}
|
|
|
|
|
|
#if defined(DEBUG) || defined(VSTF)
|
|
|
|
/*
|
|
** IsValidPCINotifyReplica()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL IsValidPCINotifyReplica(PCINotifyReplica pcinr)
|
|
{
|
|
return(IS_VALID_READ_PTR(pcinr, CINotifyReplica) &&
|
|
IS_VALID_READ_PTR(pcinr->lpVtbl, sizeof(*(pcinr->lpVtbl))) &&
|
|
IS_VALID_STRUCT_PTR((PCIUnknown)pcinr, CIUnknown) &&
|
|
IS_VALID_CODE_PTR(pcinr->lpVtbl->YouAreAReplica, YouAreAReplica));
|
|
}
|
|
|
|
|
|
/*
|
|
** IsValidPCIReconcileInitiator()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL IsValidPCIReconcileInitiator(PCIReconcileInitiator pciri)
|
|
{
|
|
return(IS_VALID_READ_PTR(pciri, CIReconcileInitiator) &&
|
|
IS_VALID_READ_PTR(pciri->lpVtbl, sizeof(*(pciri->lpVtbl))) &&
|
|
IS_VALID_STRUCT_PTR((PCIUnknown)pciri, CIUnknown) &&
|
|
IS_VALID_CODE_PTR(pciri->lpVtbl->SetAbortCallback, SetAbortCallback) &&
|
|
IS_VALID_CODE_PTR(pciri->lpVtbl->SetProgressFeedback, SetProgressFeedback));
|
|
}
|
|
|
|
#endif
|
|
|