|
|
/*++
Copyright (c) Microsoft Corporation. All rights reserved.
Module Name:
registry.c
Abstract:
Registry interface routines for Windows NT Setup API Dll.
Author:
Ted Miller (tedm) 6-Feb-1995
Revision History:
Jamie Hunter (JamieHun) Mar-05-2002 Security code review
--*/
#include "precomp.h"
#pragma hdrstop
static BOOL _RegistryDelnodeWorker( IN HKEY ParentKeyHandle, IN PCTSTR KeyName, IN DWORD Flags, OUT PDWORD ErrorCode )
/*++
Routine Description:
Delete all subkeys of a key whose name and parent's handle was passed as parameter. The algorithm used in this function guarantees that the maximum number of descendent keys will be deleted.
Arguments:
ParentKeyHandle - Handle to the parent of the key that is currently being examined.
KeyName - Name of the key that is currently being examined. This name can be an empty string (but not a NULL pointer), and in this case ParentKeyHandle refers to the key that is being examined.
ErrorCode - Pointer to a variable that will contain an Win32 error code if the function fails.
Return Value:
BOOL - Returns TRUE if the opearation succeeds.
--*/
{ HKEY CurrentKeyTraverseAccess; DWORD iSubKey; TCHAR SubKeyName[MAX_PATH+1]; DWORD SubKeyNameLength; FILETIME ftLastWriteTime; LONG Status; LONG StatusEnum; LONG SavedStatus;
//
// NOTICE-2002/03/11-JamieHun Recursive delete of registry requires special care
// particularly if running under raised security permissions
// we have to watch out for symbolic links, and if found, delete the
// link not what the link refers to
//
//
// Do not accept NULL pointer for ErrorCode
//
if(ErrorCode == NULL) { return(FALSE); } //
// Do not accept NULL pointer for KeyName.
//
if(KeyName == NULL) { *ErrorCode = ERROR_INVALID_PARAMETER; return(FALSE); }
//
// Open a handle to the key whose subkeys are to be deleted.
// Since we need to delete its subkeys, the handle must have
// KEY_ENUMERATE_SUB_KEYS access.
//
Status = RegOpenKeyEx( ParentKeyHandle, KeyName, REG_OPTION_OPEN_LINK, // don't follow links, delete them
#ifdef _WIN64
(( Flags & FLG_DELREG_32BITKEY ) ? KEY_WOW64_32KEY:0) | #else
(( Flags & FLG_DELREG_64BITKEY ) ? KEY_WOW64_64KEY:0) | #endif
KEY_ENUMERATE_SUB_KEYS | DELETE, &CurrentKeyTraverseAccess );
if(Status != ERROR_SUCCESS) { //
// If unable to enumerate the subkeys, return error.
//
*ErrorCode = Status; return(FALSE); }
//
// Traverse the key
//
iSubKey = 0; SavedStatus = ERROR_SUCCESS; do { //
// Get the name of a subkey
//
SubKeyNameLength = SIZECHARS(SubKeyName); StatusEnum = RegEnumKeyEx( CurrentKeyTraverseAccess, iSubKey, SubKeyName, &SubKeyNameLength, NULL, NULL, NULL, &ftLastWriteTime );
if(StatusEnum == ERROR_SUCCESS) { //
// Delete all children of the subkey.
// Just assume that the children will be deleted, and don't check
// for failure.
//
_RegistryDelnodeWorker(CurrentKeyTraverseAccess,SubKeyName,0,&Status); //
// Now delete the subkey, and check for failure.
//
Status = RegDeleteKey(CurrentKeyTraverseAccess,SubKeyName); //
// If unable to delete the subkey, then save the error code.
// Note that the subkey index is incremented only if the subkey
// was not deleted.
//
if(Status != ERROR_SUCCESS) { iSubKey++; SavedStatus = Status; } } else { //
// If unable to get a subkey name due to ERROR_NO_MORE_ITEMS,
// then the key doesn't have subkeys, or all subkeys were already
// enumerated. Otherwise, an error has occurred, so just save
// the error code.
//
if(StatusEnum != ERROR_NO_MORE_ITEMS) { SavedStatus = StatusEnum; } } //if((StatusEnum != ERROR_SUCCESS ) && (StatusEnum != ERROR_NO_MORE_ITEMS)) {
// printf( "RegEnumKeyEx() failed, Key Name = %ls, Status = %d, iSubKey = %d \n",KeyName,StatusEnum,iSubKey);
//}
} while(StatusEnum == ERROR_SUCCESS);
//
// Close the handle to the key whose subkeys were deleted, and return
// the result of the operation.
//
RegCloseKey(CurrentKeyTraverseAccess);
if(SavedStatus != ERROR_SUCCESS) { *ErrorCode = SavedStatus; return(FALSE); } return(TRUE); }
DWORD pSetupRegistryDelnodeEx( IN HKEY RootKey, IN PCTSTR SubKeyName, IN DWORD ExtraFlags ) /*++
Routine Description:
This routine deletes a registry key and gets rid of everything under it recursively.
Arguments:
RootKey - Supplies handle to open registry key..ex. HKLM etc.
SubKeyName - Name of the SubKey that we wish to recursively delete.
ExtraFlags - Flags that are specified in the DelReg section of the INF.
Return Value:
If successful, the return value is NO_ERROR, otherwise, it is an error code.
--*/ { DWORD d,err,Status; HKEY hKey; PTSTR p; PTSTR TempKey = NULL;
d = _RegistryDelnodeWorker(RootKey,SubKeyName,ExtraFlags,&err) ? NO_ERROR : err;
if((d == ERROR_FILE_NOT_FOUND) || (d == ERROR_PATH_NOT_FOUND)) { d = NO_ERROR; }
if(d == NO_ERROR) { //
// Delete top-level key
//
#ifdef _WIN64
if( ExtraFlags & FLG_DELREG_32BITKEY ) { #else
if( ExtraFlags & FLG_DELREG_64BITKEY ) { #endif
//
// For handling the WOW64 case:
// deleting RootKey\SubKeyName by itself won't work
// split subkeyname into parent\final
// open parent for 32-bit access, and delete final
//
TempKey = pSetupDuplicateString(SubKeyName); if(TempKey) {
p = _tcsrchr(TempKey, TEXT('\\')); if(p){ *p++ = TEXT('\0');
d = RegOpenKeyEx( RootKey, TempKey, 0, #ifdef _WIN64
KEY_WOW64_32KEY | #else
KEY_WOW64_64KEY | #endif
DELETE, &hKey );
d = RegDeleteKey(hKey, p);
}else{
d = NO_ERROR;
} pSetupFree( TempKey );
}else{ d = ERROR_NOT_ENOUGH_MEMORY; } }else{ //
// native case
//
d = RegDeleteKey(RootKey, SubKeyName); }
if((d == ERROR_FILE_NOT_FOUND) || (d == ERROR_PATH_NOT_FOUND)) { //
// FUTURE-2002/03/13-JamieHun Logging
// At a verbose level, log that this key wasn't found
//
d = NO_ERROR; } }
return(d); }
DWORD pSetupRegistryDelnode( IN HKEY RootKey, IN PCTSTR SubKeyName ) { // Calls into Ex Function
return pSetupRegistryDelnodeEx( RootKey, SubKeyName, 0);
}
|