|
|
/*****************************************************************************\
* MODULE: registry.cpp * * PURPOSE: Implementation of COM interface for BidiSpooler * * Copyright (C) 2000 Microsoft Corporation * * History: * * 03/07/00 Weihai Chen (weihaic) Created * \*****************************************************************************/
#include "precomp.h"
#include "registry.h"
////////////////////////////////////////////////////////
//
// Internal helper functions prototypes
//
// Set the given key and its value.
////////////////////////////////////////////////////////
//
// Constants
//
// Size of a CLSID as a string
CONST DWORD TComRegistry::m_cdwClsidStringSize = 39; CONST TCHAR TComRegistry::m_cszCLSID[] = _T ("CLSID\\"); CONST TCHAR TComRegistry::m_cszCLSID2[] = _T ("CLSID"); CONST TCHAR TComRegistry::m_cszInprocServer32[] = _T ("InprocServer32"); CONST TCHAR TComRegistry::m_cszProgID[] = _T ("ProgID"); CONST TCHAR TComRegistry::m_cszVersionIndependentProgID[] = _T ("VersionIndependentProgID"); CONST TCHAR TComRegistry::m_cszCurVer[] = _T ("CurVer"); CONST TCHAR TComRegistry::m_cszThreadingModel[] = _T ("ThreadingModel"); CONST TCHAR TComRegistry::m_cszBoth[] = _T ("Both");
/////////////////////////////////////////////////////////
//
// Public function implementation
//
//
// Register the component in the registry.
//
BOOL TComRegistry::RegisterServer( IN HMODULE hModule, // DLL module handle
IN REFCLSID clsid, // Class ID
IN LPCTSTR pszFriendlyName, // Friendly Name
IN LPCTSTR pszVerIndProgID, // Programmatic
IN LPCTSTR pszProgID) // IDs
{ BOOL bRet = FALSE; // Get server location.
TCHAR szModule [MAX_PATH];
DWORD dwResult; TCHAR szCLSID[m_cdwClsidStringSize] ;
DBGMSG(DBG_TRACE,("Enter RegisterServer"));
if (GetModuleFileName(hModule, szModule,MAX_PATH) > 0) {
// Convert the CLSID into a string.
if (CLSIDtoString(clsid, szCLSID, sizeof(szCLSID))) {
// Build the key CLSID\\{...}
TCHAR szKey[64] ; StringCchCopy(szKey, COUNTOF(szKey), m_cszCLSID) ; StringCchCat(szKey, COUNTOF(szKey), szCLSID) ;
// Add the CLSID to the registry.
if (SetKeyAndValue(szKey, NULL, pszFriendlyName) &&
// Add the server filename subkey under the CLSID key.
SetKeyAndValue(szKey, m_cszInprocServer32, szModule) &&
SetKeyAndNameValue(szKey, m_cszInprocServer32, m_cszThreadingModel, m_cszBoth) &&
// Add the ProgID subkey under the CLSID key.
SetKeyAndValue(szKey, m_cszProgID, pszProgID) &&
// Add the version-independent ProgID subkey under CLSID key.
SetKeyAndValue(szKey, m_cszVersionIndependentProgID, pszVerIndProgID) &&
// Add the version-independent ProgID subkey under HKEY_CLASSES_ROOT.
SetKeyAndValue(pszVerIndProgID, NULL, pszFriendlyName) && SetKeyAndValue(pszVerIndProgID, m_cszCLSID2, szCLSID) && SetKeyAndValue(pszVerIndProgID, m_cszCurVer, pszProgID) &&
// Add the versioned ProgID subkey under HKEY_CLASSES_ROOT.
SetKeyAndValue(pszProgID, NULL, pszFriendlyName) && SetKeyAndValue(pszProgID, m_cszCLSID2, szCLSID) ) {
bRet = TRUE; } } }
DBGMSG(DBG_TRACE,("Leave RegisterServer (Ret = %d)\n", bRet));
return bRet; }
//
// Remove the component from the registry.
//
BOOL TComRegistry::UnregisterServer( IN REFCLSID clsid, // Class ID
IN LPCTSTR pszVerIndProgID, // Programmatic
IN LPCTSTR pszProgID) // IDs
{ BOOL bRet = FALSE; // Convert the CLSID into a char.
TCHAR szCLSID[m_cdwClsidStringSize] ;
DBGMSG(DBG_TRACE,("Enter UnregisterServer\n", bRet));
if (CLSIDtoString(clsid, szCLSID, sizeof(szCLSID))) {
TCHAR szKey[64] ; StringCchCopy(szKey, COUNTOF(szKey), m_cszCLSID) ; StringCchCat(szKey, COUNTOF(szKey), szCLSID) ;
if (RecursiveDeleteKey(HKEY_CLASSES_ROOT, szKey) && RecursiveDeleteKey(HKEY_CLASSES_ROOT, pszVerIndProgID) && RecursiveDeleteKey(HKEY_CLASSES_ROOT, pszProgID)) {
bRet = TRUE;
} }
DBGMSG(DBG_TRACE,("Leave UnregisterServer (Ret = %d)\n", bRet));
return bRet; }
///////////////////////////////////////////////////////////
//
// Internal helper functions
//
// Convert a CLSID to a string.
BOOL TComRegistry::CLSIDtoString( IN REFCLSID clsid, IN OUT LPTSTR pszCLSID, IN DWORD dwLength)
{ BOOL bRet = FALSE; HRESULT hr = E_FAIL; LPWSTR pwszCLSID = NULL ;
if (dwLength >= m_cdwClsidStringSize ) { // Get CLSID
hr = StringFromCLSID(clsid, &pwszCLSID);
if (SUCCEEDED (hr)) { hr = StringCbCopy( pszCLSID, dwLength, pwszCLSID); if (SUCCEEDED(hr)) bRet = TRUE; else SetLastError (HRESULTTOWIN32 (hr));
// Free memory.
CoTaskMemFree(pwszCLSID) ;
} else SetLastError (HRESULTTOWIN32 (hr)); }
return bRet; }
//
// Delete a key and all of its descendents.
//
BOOL TComRegistry::RecursiveDeleteKey( IN HKEY hKeyParent, // Parent of key to delete
IN LPCTSTR lpszKeyChild) // Key to delete
{ BOOL bRet = FALSE; // Open the child.
HKEY hKeyChild = NULL; LONG lResult = 0; FILETIME time ; TCHAR szBuffer[MAX_PATH] ; DWORD dwSize;
lResult = RegOpenKeyEx (hKeyParent, lpszKeyChild, 0, KEY_ALL_ACCESS, &hKeyChild) ;
if (lResult == ERROR_SUCCESS) {
// Enumerate all of the decendents of this child.
for (;;) {
dwSize = MAX_PATH ;
lResult = RegEnumKeyEx(hKeyChild, 0, szBuffer, &dwSize, NULL, NULL, NULL, &time); if (lResult == ERROR_NO_MORE_ITEMS) { break; } else if (lResult == ERROR_SUCCESS) { // Delete the decendents of this child.
if (!RecursiveDeleteKey (hKeyChild, szBuffer)) { goto Cleanup; } } else { goto Cleanup; } }
// Close the child.
RegCloseKey(hKeyChild) ; hKeyChild = NULL;
// Delete this child.
if (ERROR_SUCCESS == RegDeleteKey(hKeyParent, lpszKeyChild)) { bRet = TRUE; }
}
Cleanup: // Cleanup before exiting.
if (hKeyChild) RegCloseKey(hKeyChild) ;
if (!bRet && lResult) SetLastError (lResult);
return bRet;
}
//
// Create a key and set its value.
// - This helper function was borrowed and modifed from
// Kraig Brockschmidt's book Inside OLE.
//
BOOL TComRegistry::SetKeyAndValue( IN LPCTSTR pszKey, IN LPCTSTR pszSubkey, IN LPCTSTR pszValue) { return SetKeyAndNameValue (pszKey, pszSubkey, NULL, pszValue); }
BOOL TComRegistry::SetKeyAndNameValue( IN LPCTSTR pszKey, IN LPCTSTR pszSubkey, IN LPCTSTR pszName, IN LPCTSTR pszValue) { BOOL bRet = FALSE; HKEY hKey = NULL; LPTSTR pszKeyBuf = NULL; long lResult; DWORD dwLen = lstrlen (pszKey) + 1;
if (pszSubkey) { dwLen += lstrlen (pszSubkey) + 1; }
pszKeyBuf = new TCHAR [dwLen];
if (pszKeyBuf) { // Copy keyname into buffer.
StringCchCopy(pszKeyBuf, dwLen, pszKey) ;
// Add subkey name to buffer.
if (pszSubkey != NULL) { StringCchCat(pszKeyBuf, dwLen, _T ("\\")) ; StringCchCat(pszKeyBuf, dwLen, pszSubkey ) ; }
// Create and open key and subkey.
lResult = RegCreateKeyEx(HKEY_CLASSES_ROOT , pszKeyBuf, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL);
if (ERROR_SUCCESS == lResult ) {
// Set the Value.
if (pszValue != NULL) {
lResult = RegSetValueEx (hKey, pszName, 0, REG_SZ, (PBYTE) pszValue, sizeof (TCHAR) * lstrlen(pszValue));
if (ERROR_SUCCESS == lResult ) {
bRet = TRUE; } } else bRet = TRUE; }
if (hKey) { RegCloseKey(hKey) ; }
delete [] pszKeyBuf; }
return bRet; }
|