| /* |
| * SHLWAPI registry functions |
| * |
| * Copyright 1998 Juergen Schmied |
| * Copyright 2001 Guy Albertelli |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include <stdarg.h> |
| #include <string.h> |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winuser.h" |
| #include "winreg.h" |
| #include "wine/debug.h" |
| #define NO_SHLWAPI_STREAM |
| #include "shlwapi.h" |
| #include "wine/unicode.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(shell); |
| |
| /* Key/Value names for MIME content types */ |
| static const char lpszContentTypeA[] = "Content Type"; |
| static const WCHAR lpszContentTypeW[] = { 'C','o','n','t','e','n','t',' ','T','y','p','e','\0'}; |
| |
| static const char szMimeDbContentA[] = "MIME\\Database\\Content Type\\"; |
| static const WCHAR szMimeDbContentW[] = { 'M', 'I', 'M','E','\\', |
| 'D','a','t','a','b','a','s','e','\\','C','o','n','t','e','n','t', |
| ' ','T','y','p','e','\\', 0 }; |
| static const DWORD dwLenMimeDbContent = 27; /* strlen of szMimeDbContentA/W */ |
| |
| static const char szExtensionA[] = "Extension"; |
| static const WCHAR szExtensionW[] = { 'E', 'x', 't','e','n','s','i','o','n','\0' }; |
| |
| /* internal structure of what the HUSKEY points to */ |
| typedef struct { |
| HKEY HKCUstart; /* Start key in CU hive */ |
| HKEY HKCUkey; /* Opened key in CU hive */ |
| HKEY HKLMstart; /* Start key in LM hive */ |
| HKEY HKLMkey; /* Opened key in LM hive */ |
| WCHAR lpszPath[MAX_PATH]; |
| } SHUSKEY, *LPSHUSKEY; |
| |
| INT WINAPI SHStringFromGUIDW(REFGUID,LPWSTR,INT); |
| HRESULT WINAPI SHRegGetCLSIDKeyW(REFGUID,LPCWSTR,BOOL,BOOL,PHKEY); |
| |
| |
| #define REG_HKCU TRUE |
| #define REG_HKLM FALSE |
| /************************************************************************* |
| * REG_GetHKEYFromHUSKEY |
| * |
| * Function: Return the proper registry key from the HUSKEY structure |
| * also allow special predefined values. |
| */ |
| static HKEY WINAPI REG_GetHKEYFromHUSKEY(HUSKEY hUSKey, BOOL which) |
| { |
| HKEY test = (HKEY) hUSKey; |
| LPSHUSKEY mihk = (LPSHUSKEY) hUSKey; |
| |
| if ((test == HKEY_CLASSES_ROOT) || |
| (test == HKEY_CURRENT_CONFIG) || |
| (test == HKEY_CURRENT_USER) || |
| (test == HKEY_DYN_DATA) || |
| (test == HKEY_LOCAL_MACHINE) || |
| (test == HKEY_PERFORMANCE_DATA) || |
| /* FIXME: need to define for Win2k, ME, XP |
| * (test == HKEY_PERFORMANCE_TEXT) || |
| * (test == HKEY_PERFORMANCE_NLSTEXT) || |
| */ |
| (test == HKEY_USERS)) return test; |
| if (which == REG_HKCU) return mihk->HKCUkey; |
| return mihk->HKLMkey; |
| } |
| |
| |
| /************************************************************************* |
| * SHRegOpenUSKeyA [SHLWAPI.@] |
| * |
| * Open a user-specific registry key. |
| * |
| * PARAMS |
| * Path [I] Key name to open |
| * AccessType [I] Access type |
| * hRelativeUSKey [I] Relative user key |
| * phNewUSKey [O] Destination for created key |
| * fIgnoreHKCU [I] TRUE=Don't check HKEY_CURRENT_USER |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from RegOpenKeyExA(). |
| */ |
| LONG WINAPI SHRegOpenUSKeyA(LPCSTR Path, REGSAM AccessType, HUSKEY hRelativeUSKey, |
| PHUSKEY phNewUSKey, BOOL fIgnoreHKCU) |
| { |
| WCHAR szPath[MAX_PATH]; |
| |
| if (Path) |
| MultiByteToWideChar(CP_ACP, 0, Path, -1, szPath, MAX_PATH); |
| |
| return SHRegOpenUSKeyW(Path ? szPath : NULL, AccessType, hRelativeUSKey, |
| phNewUSKey, fIgnoreHKCU); |
| } |
| |
| /************************************************************************* |
| * SHRegOpenUSKeyW [SHLWAPI.@] |
| * |
| * See SHRegOpenUSKeyA. |
| */ |
| LONG WINAPI SHRegOpenUSKeyW(LPCWSTR Path, REGSAM AccessType, HUSKEY hRelativeUSKey, |
| PHUSKEY phNewUSKey, BOOL fIgnoreHKCU) |
| { |
| LONG ret2, ret1 = ~ERROR_SUCCESS; |
| LPSHUSKEY hKey; |
| |
| TRACE("(%s,0x%x,%p,%p,%d)\n", debugstr_w(Path),(LONG)AccessType, |
| hRelativeUSKey, phNewUSKey, fIgnoreHKCU); |
| |
| if (phNewUSKey) |
| *phNewUSKey = NULL; |
| |
| /* Create internal HUSKEY */ |
| hKey = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*hKey)); |
| lstrcpynW(hKey->lpszPath, Path, sizeof(hKey->lpszPath)/sizeof(WCHAR)); |
| |
| if (hRelativeUSKey) |
| { |
| hKey->HKCUstart = SHRegDuplicateHKey(REG_GetHKEYFromHUSKEY(hRelativeUSKey, REG_HKCU)); |
| hKey->HKLMstart = SHRegDuplicateHKey(REG_GetHKEYFromHUSKEY(hRelativeUSKey, REG_HKLM)); |
| |
| /* FIXME: if either of these keys is NULL, create the start key from |
| * the relative keys start+path |
| */ |
| } |
| else |
| { |
| hKey->HKCUstart = HKEY_CURRENT_USER; |
| hKey->HKLMstart = HKEY_LOCAL_MACHINE; |
| } |
| |
| if (!fIgnoreHKCU) |
| { |
| ret1 = RegOpenKeyExW(hKey->HKCUstart, hKey->lpszPath, 0, AccessType, &hKey->HKCUkey); |
| if (ret1) |
| hKey->HKCUkey = 0; |
| } |
| |
| ret2 = RegOpenKeyExW(hKey->HKLMstart, hKey->lpszPath, 0, AccessType, &hKey->HKLMkey); |
| if (ret2) |
| hKey->HKLMkey = 0; |
| |
| if (ret1 || ret2) |
| TRACE("one or more opens failed: HKCU=%d HKLM=%d\n", ret1, ret2); |
| |
| if (ret1 && ret2) |
| { |
| /* Neither open succeeded: fail */ |
| SHRegCloseUSKey(hKey); |
| return ret2; |
| } |
| |
| TRACE("HUSKEY=%p\n", hKey); |
| if (phNewUSKey) |
| *phNewUSKey = (HUSKEY)hKey; |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegCloseUSKey [SHLWAPI.@] |
| * |
| * Close a user-specific registry key |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from RegCloseKey(). |
| */ |
| LONG WINAPI SHRegCloseUSKey( |
| HUSKEY hUSKey) /* [I] Key to close */ |
| { |
| LPSHUSKEY hKey = (LPSHUSKEY)hUSKey; |
| LONG ret = ERROR_SUCCESS; |
| |
| if (hKey->HKCUkey) |
| ret = RegCloseKey(hKey->HKCUkey); |
| if (hKey->HKCUstart && hKey->HKCUstart != HKEY_CURRENT_USER) |
| ret = RegCloseKey(hKey->HKCUstart); |
| if (hKey->HKLMkey) |
| ret = RegCloseKey(hKey->HKLMkey); |
| if (hKey->HKLMstart && hKey->HKLMstart != HKEY_LOCAL_MACHINE) |
| ret = RegCloseKey(hKey->HKCUstart); |
| |
| HeapFree(GetProcessHeap(), 0, hKey); |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegCreateUSKeyA [SHLWAPI.@] |
| * |
| * Create or open a user-specific registry key. |
| * |
| * PARAMS |
| * pszPath [I] Key name to create or open. |
| * samDesired [I] Wanted security access. |
| * hRelativeUSKey [I] Base path if pszPath is relative. NULL otherwise. |
| * phNewUSKey [O] Receives a handle to the new or openened key. |
| * dwFlags [I] Base key under which the key should be opened. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: Nonzero error code from winerror.h |
| */ |
| LONG WINAPI SHRegCreateUSKeyA(LPCSTR pszPath, REGSAM samDesired, HUSKEY hRelativeUSKey, |
| PHUSKEY phNewUSKey, DWORD dwFlags) |
| { |
| FIXME("(%s, 0x%08x, %p, %p, 0x%08x) stub\n", debugstr_a(pszPath), samDesired, |
| hRelativeUSKey, phNewUSKey, dwFlags); |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegCreateUSKeyW [SHLWAPI.@] |
| * |
| * See SHRegCreateUSKeyA. |
| */ |
| LONG WINAPI SHRegCreateUSKeyW(LPCWSTR pszPath, REGSAM samDesired, HUSKEY hRelativeUSKey, |
| PHUSKEY phNewUSKey, DWORD dwFlags) |
| { |
| FIXME("(%s, 0x%08x, %p, %p, 0x%08x) stub\n", debugstr_w(pszPath), samDesired, |
| hRelativeUSKey, phNewUSKey, dwFlags); |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegDeleteEmptyUSKeyA [SHLWAPI.@] |
| * |
| * Delete an empty user-specific registry key. |
| * |
| * PARAMS |
| * hUSKey [I] Handle to an open registry key. |
| * pszValue [I] Empty key name. |
| * delRegFlags [I] Flag that specifies the base from which to delete |
| * the key. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: Nonzero error code from winerror.h |
| */ |
| LONG WINAPI SHRegDeleteEmptyUSKeyA(HUSKEY hUSKey, LPCSTR pszValue, SHREGDEL_FLAGS delRegFlags) |
| { |
| FIXME("(%p, %s, 0x%08x) stub\n", hUSKey, debugstr_a(pszValue), delRegFlags); |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegDeleteEmptyUSKeyW [SHLWAPI.@] |
| * |
| * See SHRegDeleteEmptyUSKeyA. |
| */ |
| LONG WINAPI SHRegDeleteEmptyUSKeyW(HUSKEY hUSKey, LPCWSTR pszValue, SHREGDEL_FLAGS delRegFlags) |
| { |
| FIXME("(%p, %s, 0x%08x) stub\n", hUSKey, debugstr_w(pszValue), delRegFlags); |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegDeleteUSValueA [SHLWAPI.@] |
| * |
| * Delete a user-specific registry value. |
| * |
| * PARAMS |
| * hUSKey [I] Handle to an open registry key. |
| * pszValue [I] Specifies the value to delete. |
| * delRegFlags [I] Flag that specifies the base of the key from which to |
| * delete the value. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: Nonzero error code from winerror.h |
| */ |
| LONG WINAPI SHRegDeleteUSValueA(HUSKEY hUSKey, LPCSTR pszValue, SHREGDEL_FLAGS delRegFlags) |
| { |
| FIXME("(%p, %s, 0x%08x) stub\n", hUSKey, debugstr_a(pszValue), delRegFlags); |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegDeleteUSValueW [SHLWAPI.@] |
| * |
| * See SHRegDeleteUSValueA. |
| */ |
| LONG WINAPI SHRegDeleteUSValueW(HUSKEY hUSKey, LPCWSTR pszValue, SHREGDEL_FLAGS delRegFlags) |
| { |
| FIXME("(%p, %s, 0x%08x) stub\n", hUSKey, debugstr_w(pszValue), delRegFlags); |
| return ERROR_SUCCESS; |
| } |
| |
| /************************************************************************* |
| * SHRegEnumUSValueA [SHLWAPI.@] |
| * |
| * Enumerate values of a specified registry key. |
| * |
| * PARAMS |
| * hUSKey [I] Handle to an open registry key. |
| * dwIndex [I] Index of the value to be retrieved. |
| * pszValueName [O] Buffer to receive the value name. |
| * pcchValueNameLen [I] Size of pszValueName in characters. |
| * pdwType [O] Receives data type of the value. |
| * pvData [O] Receives value data. May be NULL. |
| * pcbData [I/O] Size of pvData in bytes. |
| * enumRegFlags [I] Flag that specifies the base key under which to |
| * enumerate values. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: Nonzero error code from winerror.h |
| */ |
| LONG WINAPI SHRegEnumUSValueA(HUSKEY hUSKey, DWORD dwIndex, LPSTR pszValueName, |
| LPDWORD pcchValueNameLen, LPDWORD pdwType, LPVOID pvData, |
| LPDWORD pcbData, SHREGENUM_FLAGS enumRegFlags) |
| { |
| FIXME("(%p, 0x%08x, %s, %p, %p, %p, %p, 0x%08x) stub\n", hUSKey, dwIndex, |
| debugstr_a(pszValueName), pcchValueNameLen, pdwType, pvData, pcbData, enumRegFlags); |
| return ERROR_INVALID_FUNCTION; |
| } |
| |
| /************************************************************************* |
| * SHRegEnumUSValueW [SHLWAPI.@] |
| * |
| * See SHRegEnumUSValueA. |
| */ |
| LONG WINAPI SHRegEnumUSValueW(HUSKEY hUSKey, DWORD dwIndex, LPWSTR pszValueName, |
| LPDWORD pcchValueNameLen, LPDWORD pdwType, LPVOID pvData, |
| LPDWORD pcbData, SHREGENUM_FLAGS enumRegFlags) |
| { |
| FIXME("(%p, 0x%08x, %s, %p, %p, %p, %p, 0x%08x) stub\n", hUSKey, dwIndex, |
| debugstr_w(pszValueName), pcchValueNameLen, pdwType, pvData, pcbData, enumRegFlags); |
| return ERROR_INVALID_FUNCTION; |
| } |
| |
| /************************************************************************* |
| * SHRegQueryUSValueA [SHLWAPI.@] |
| * |
| * Query a user-specific registry value. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from RegQueryValueExA(). |
| */ |
| LONG WINAPI SHRegQueryUSValueA( |
| HUSKEY hUSKey, /* [I] Key to query */ |
| LPCSTR pszValue, /* [I] Value name under hUSKey */ |
| LPDWORD pdwType, /* [O] Destination for value type */ |
| LPVOID pvData, /* [O] Destination for value data */ |
| LPDWORD pcbData, /* [O] Destination for value length */ |
| BOOL fIgnoreHKCU, /* [I] TRUE=Don't check HKEY_CURRENT_USER */ |
| LPVOID pvDefaultData, /* [I] Default data if pszValue does not exist */ |
| DWORD dwDefaultDataSize) /* [I] Length of pvDefaultData */ |
| { |
| LONG ret = ~ERROR_SUCCESS; |
| LONG i, maxmove; |
| HKEY dokey; |
| CHAR *src, *dst; |
| |
| /* if user wants HKCU, and it exists, then try it */ |
| if (!fIgnoreHKCU && (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKCU))) { |
| ret = RegQueryValueExA(dokey, |
| pszValue, 0, pdwType, pvData, pcbData); |
| TRACE("HKCU RegQueryValue returned %08x\n", ret); |
| } |
| |
| /* if HKCU did not work and HKLM exists, then try it */ |
| if ((ret != ERROR_SUCCESS) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKLM))) { |
| ret = RegQueryValueExA(dokey, |
| pszValue, 0, pdwType, pvData, pcbData); |
| TRACE("HKLM RegQueryValue returned %08x\n", ret); |
| } |
| |
| /* if neither worked, and default data exists, then use it */ |
| if (ret != ERROR_SUCCESS) { |
| if (pvDefaultData && (dwDefaultDataSize != 0)) { |
| maxmove = (dwDefaultDataSize >= *pcbData) ? *pcbData : dwDefaultDataSize; |
| src = (CHAR*)pvDefaultData; |
| dst = (CHAR*)pvData; |
| for(i=0; i<maxmove; i++) *dst++ = *src++; |
| *pcbData = maxmove; |
| TRACE("setting default data\n"); |
| ret = ERROR_SUCCESS; |
| } |
| } |
| return ret; |
| } |
| |
| |
| /************************************************************************* |
| * SHRegQueryUSValueW [SHLWAPI.@] |
| * |
| * See SHRegQueryUSValueA. |
| */ |
| LONG WINAPI SHRegQueryUSValueW( |
| HUSKEY hUSKey, |
| LPCWSTR pszValue, |
| LPDWORD pdwType, |
| LPVOID pvData, |
| LPDWORD pcbData, |
| BOOL fIgnoreHKCU, |
| LPVOID pvDefaultData, |
| DWORD dwDefaultDataSize) |
| { |
| LONG ret = ~ERROR_SUCCESS; |
| LONG i, maxmove; |
| HKEY dokey; |
| CHAR *src, *dst; |
| |
| /* if user wants HKCU, and it exists, then try it */ |
| if (!fIgnoreHKCU && (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKCU))) { |
| ret = RegQueryValueExW(dokey, |
| pszValue, 0, pdwType, pvData, pcbData); |
| TRACE("HKCU RegQueryValue returned %08x\n", ret); |
| } |
| |
| /* if HKCU did not work and HKLM exists, then try it */ |
| if ((ret != ERROR_SUCCESS) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKLM))) { |
| ret = RegQueryValueExW(dokey, |
| pszValue, 0, pdwType, pvData, pcbData); |
| TRACE("HKLM RegQueryValue returned %08x\n", ret); |
| } |
| |
| /* if neither worked, and default data exists, then use it */ |
| if (ret != ERROR_SUCCESS) { |
| if (pvDefaultData && (dwDefaultDataSize != 0)) { |
| maxmove = (dwDefaultDataSize >= *pcbData) ? *pcbData : dwDefaultDataSize; |
| src = (CHAR*)pvDefaultData; |
| dst = (CHAR*)pvData; |
| for(i=0; i<maxmove; i++) *dst++ = *src++; |
| *pcbData = maxmove; |
| TRACE("setting default data\n"); |
| ret = ERROR_SUCCESS; |
| } |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegGetUSValueA [SHLWAPI.@] |
| * |
| * Get a user-specific registry value. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from SHRegOpenUSKeyA() or SHRegQueryUSValueA(). |
| * |
| * NOTES |
| * This function opens pSubKey, queries the value, and then closes the key. |
| */ |
| LONG WINAPI SHRegGetUSValueA( |
| LPCSTR pSubKey, /* [I] Key name to open */ |
| LPCSTR pValue, /* [I] Value name to open */ |
| LPDWORD pwType, /* [O] Destination for the type of the value */ |
| LPVOID pvData, /* [O] Destination for the value */ |
| LPDWORD pcbData, /* [I] Destination for the length of the value **/ |
| BOOL flagIgnoreHKCU, /* [I] TRUE=Don't check HKEY_CURRENT_USER */ |
| LPVOID pDefaultData, /* [I] Default value if it doesn't exist */ |
| DWORD wDefaultDataSize) /* [I] Length of pDefaultData */ |
| { |
| HUSKEY myhuskey; |
| LONG ret; |
| |
| if (!pvData || !pcbData) return ERROR_INVALID_FUNCTION; /* FIXME:wrong*/ |
| TRACE("key '%s', value '%s', datalen %d, %s\n", |
| debugstr_a(pSubKey), debugstr_a(pValue), *pcbData, |
| (flagIgnoreHKCU) ? "Ignoring HKCU" : "Tries HKCU then HKLM"); |
| |
| ret = SHRegOpenUSKeyA(pSubKey, 0x1, 0, &myhuskey, flagIgnoreHKCU); |
| if (ret == ERROR_SUCCESS) { |
| ret = SHRegQueryUSValueA(myhuskey, pValue, pwType, pvData, |
| pcbData, flagIgnoreHKCU, pDefaultData, |
| wDefaultDataSize); |
| SHRegCloseUSKey(myhuskey); |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegGetUSValueW [SHLWAPI.@] |
| * |
| * See SHRegGetUSValueA. |
| */ |
| LONG WINAPI SHRegGetUSValueW( |
| LPCWSTR pSubKey, |
| LPCWSTR pValue, |
| LPDWORD pwType, |
| LPVOID pvData, |
| LPDWORD pcbData, |
| BOOL flagIgnoreHKCU, |
| LPVOID pDefaultData, |
| DWORD wDefaultDataSize) |
| { |
| HUSKEY myhuskey; |
| LONG ret; |
| |
| if (!pvData || !pcbData) return ERROR_INVALID_FUNCTION; /* FIXME:wrong*/ |
| TRACE("key '%s', value '%s', datalen %d, %s\n", |
| debugstr_w(pSubKey), debugstr_w(pValue), *pcbData, |
| (flagIgnoreHKCU) ? "Ignoring HKCU" : "Tries HKCU then HKLM"); |
| |
| ret = SHRegOpenUSKeyW(pSubKey, 0x1, 0, &myhuskey, flagIgnoreHKCU); |
| if (ret == ERROR_SUCCESS) { |
| ret = SHRegQueryUSValueW(myhuskey, pValue, pwType, pvData, |
| pcbData, flagIgnoreHKCU, pDefaultData, |
| wDefaultDataSize); |
| SHRegCloseUSKey(myhuskey); |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegSetUSValueA [SHLWAPI.@] |
| * |
| * Set a user-specific registry value. |
| * |
| * PARAMS |
| * pszSubKey [I] Name of key to set the value in |
| * pszValue [I] Name of value under pszSubKey to set the value in |
| * dwType [I] Type of the value |
| * pvData [I] Data to set as the value |
| * cbData [I] length of pvData |
| * dwFlags [I] SHREGSET_ flags from "shlwapi.h" |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from SHRegOpenUSKeyA() or SHRegWriteUSValueA(), or |
| * ERROR_INVALID_FUNCTION if pvData is NULL. |
| * |
| * NOTES |
| * This function opens pszSubKey, sets the value, and then closes the key. |
| */ |
| LONG WINAPI SHRegSetUSValueA(LPCSTR pszSubKey, LPCSTR pszValue, DWORD dwType, |
| LPVOID pvData, DWORD cbData, DWORD dwFlags) |
| { |
| BOOL ignoreHKCU = TRUE; |
| HUSKEY hkey; |
| LONG ret; |
| |
| TRACE("(%s,%s,%d,%p,%d,0x%08x\n", debugstr_a(pszSubKey), debugstr_a(pszValue), |
| dwType, pvData, cbData, dwFlags); |
| |
| if (!pvData) |
| return ERROR_INVALID_FUNCTION; |
| |
| if (dwFlags & SHREGSET_HKCU || dwFlags & SHREGSET_FORCE_HKCU) |
| ignoreHKCU = FALSE; |
| |
| ret = SHRegOpenUSKeyA(pszSubKey, KEY_ALL_ACCESS, 0, &hkey, ignoreHKCU); |
| if (ret == ERROR_SUCCESS) |
| { |
| ret = SHRegWriteUSValueA(hkey, pszValue, dwType, pvData, cbData, dwFlags); |
| SHRegCloseUSKey(hkey); |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegSetUSValueW [SHLWAPI.@] |
| * |
| * See SHRegSetUSValueA. |
| */ |
| LONG WINAPI SHRegSetUSValueW(LPCWSTR pszSubKey, LPCWSTR pszValue, DWORD dwType, |
| LPVOID pvData, DWORD cbData, DWORD dwFlags) |
| { |
| BOOL ignoreHKCU = TRUE; |
| HUSKEY hkey; |
| LONG ret; |
| |
| TRACE("(%s,%s,%d,%p,%d,0x%08x\n", debugstr_w(pszSubKey), debugstr_w(pszValue), |
| dwType, pvData, cbData, dwFlags); |
| |
| if (!pvData) |
| return ERROR_INVALID_FUNCTION; |
| |
| if (dwFlags & SHREGSET_HKCU || dwFlags & SHREGSET_FORCE_HKCU) |
| ignoreHKCU = FALSE; |
| |
| ret = SHRegOpenUSKeyW(pszSubKey, KEY_ALL_ACCESS, 0, &hkey, ignoreHKCU); |
| if (ret == ERROR_SUCCESS) |
| { |
| ret = SHRegWriteUSValueW(hkey, pszValue, dwType, pvData, cbData, dwFlags); |
| SHRegCloseUSKey(hkey); |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegGetBoolUSValueA [SHLWAPI.@] |
| * |
| * Get a user-specific registry boolean value. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from SHRegOpenUSKeyA() or SHRegQueryUSValueA(). |
| * |
| * NOTES |
| * This function opens pszSubKey, queries the value, and then closes the key. |
| * |
| * Boolean values are one of the following: |
| * True: YES,TRUE,non-zero |
| * False: NO,FALSE,0 |
| */ |
| BOOL WINAPI SHRegGetBoolUSValueA( |
| LPCSTR pszSubKey, /* [I] Key name to open */ |
| LPCSTR pszValue, /* [I] Value name to open */ |
| BOOL fIgnoreHKCU, /* [I] TRUE=Don't check HKEY_CURRENT_USER */ |
| BOOL fDefault) /* [I] Default value to use if pszValue is not present */ |
| { |
| LONG retvalue; |
| DWORD type, datalen, work; |
| BOOL ret = fDefault; |
| CHAR data[10]; |
| |
| TRACE("key '%s', value '%s', %s\n", |
| debugstr_a(pszSubKey), debugstr_a(pszValue), |
| (fIgnoreHKCU) ? "Ignoring HKCU" : "Tries HKCU then HKLM"); |
| |
| datalen = sizeof(data)-1; |
| if (!(retvalue = SHRegGetUSValueA( pszSubKey, pszValue, &type, |
| data, &datalen, |
| fIgnoreHKCU, 0, 0))) { |
| /* process returned data via type into bool */ |
| switch (type) { |
| case REG_SZ: |
| data[9] = '\0'; /* set end of string */ |
| if (lstrcmpiA(data, "YES") == 0) ret = TRUE; |
| if (lstrcmpiA(data, "TRUE") == 0) ret = TRUE; |
| if (lstrcmpiA(data, "NO") == 0) ret = FALSE; |
| if (lstrcmpiA(data, "FALSE") == 0) ret = FALSE; |
| break; |
| case REG_DWORD: |
| work = *(LPDWORD)data; |
| ret = (work != 0); |
| break; |
| case REG_BINARY: |
| if (datalen == 1) { |
| ret = (data[0] != '\0'); |
| break; |
| } |
| default: |
| FIXME("Unsupported registry data type %d\n", type); |
| ret = FALSE; |
| } |
| TRACE("got value (type=%d), returning <%s>\n", type, |
| (ret) ? "TRUE" : "FALSE"); |
| } |
| else { |
| ret = fDefault; |
| TRACE("returning default data <%s>\n", |
| (ret) ? "TRUE" : "FALSE"); |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegGetBoolUSValueW [SHLWAPI.@] |
| * |
| * See SHRegGetBoolUSValueA. |
| */ |
| BOOL WINAPI SHRegGetBoolUSValueW( |
| LPCWSTR pszSubKey, |
| LPCWSTR pszValue, |
| BOOL fIgnoreHKCU, |
| BOOL fDefault) |
| { |
| static const WCHAR wYES[]= {'Y','E','S','\0'}; |
| static const WCHAR wTRUE[]= {'T','R','U','E','\0'}; |
| static const WCHAR wNO[]= {'N','O','\0'}; |
| static const WCHAR wFALSE[]={'F','A','L','S','E','\0'}; |
| LONG retvalue; |
| DWORD type, datalen, work; |
| BOOL ret = fDefault; |
| WCHAR data[10]; |
| |
| TRACE("key '%s', value '%s', %s\n", |
| debugstr_w(pszSubKey), debugstr_w(pszValue), |
| (fIgnoreHKCU) ? "Ignoring HKCU" : "Tries HKCU then HKLM"); |
| |
| datalen = (sizeof(data)-1) * sizeof(WCHAR); |
| if (!(retvalue = SHRegGetUSValueW( pszSubKey, pszValue, &type, |
| data, &datalen, |
| fIgnoreHKCU, 0, 0))) { |
| /* process returned data via type into bool */ |
| switch (type) { |
| case REG_SZ: |
| data[9] = L'\0'; /* set end of string */ |
| if (lstrcmpiW(data, wYES)==0 || lstrcmpiW(data, wTRUE)==0) |
| ret = TRUE; |
| else if (lstrcmpiW(data, wNO)==0 || lstrcmpiW(data, wFALSE)==0) |
| ret = FALSE; |
| break; |
| case REG_DWORD: |
| work = *(LPDWORD)data; |
| ret = (work != 0); |
| break; |
| case REG_BINARY: |
| if (datalen == 1) { |
| ret = (data[0] != L'\0'); |
| break; |
| } |
| default: |
| FIXME("Unsupported registry data type %d\n", type); |
| ret = FALSE; |
| } |
| TRACE("got value (type=%d), returning <%s>\n", type, |
| (ret) ? "TRUE" : "FALSE"); |
| } |
| else { |
| ret = fDefault; |
| TRACE("returning default data <%s>\n", |
| (ret) ? "TRUE" : "FALSE"); |
| } |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegQueryInfoUSKeyA [SHLWAPI.@] |
| * |
| * Get information about a user-specific registry key. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from RegQueryInfoKeyA(). |
| */ |
| LONG WINAPI SHRegQueryInfoUSKeyA( |
| HUSKEY hUSKey, /* [I] Key to query */ |
| LPDWORD pcSubKeys, /* [O] Destination for number of sub keys */ |
| LPDWORD pcchMaxSubKeyLen, /* [O] Destination for the length of the biggest sub key name */ |
| LPDWORD pcValues, /* [O] Destination for number of values */ |
| LPDWORD pcchMaxValueNameLen,/* [O] Destination for the length of the biggest value */ |
| SHREGENUM_FLAGS enumRegFlags) /* [in] SHREGENUM_ flags from "shlwapi.h" */ |
| { |
| HKEY dokey; |
| LONG ret; |
| |
| TRACE("(%p,%p,%p,%p,%p,%d)\n", |
| hUSKey,pcSubKeys,pcchMaxSubKeyLen,pcValues, |
| pcchMaxValueNameLen,enumRegFlags); |
| |
| /* if user wants HKCU, and it exists, then try it */ |
| if (((enumRegFlags == SHREGENUM_HKCU) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKCU))) { |
| ret = RegQueryInfoKeyA(dokey, 0, 0, 0, |
| pcSubKeys, pcchMaxSubKeyLen, 0, |
| pcValues, pcchMaxValueNameLen, 0, 0, 0); |
| if ((ret == ERROR_SUCCESS) || |
| (enumRegFlags == SHREGENUM_HKCU)) |
| return ret; |
| } |
| if (((enumRegFlags == SHREGENUM_HKLM) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKLM))) { |
| return RegQueryInfoKeyA(dokey, 0, 0, 0, |
| pcSubKeys, pcchMaxSubKeyLen, 0, |
| pcValues, pcchMaxValueNameLen, 0, 0, 0); |
| } |
| return ERROR_INVALID_FUNCTION; |
| } |
| |
| /************************************************************************* |
| * SHRegQueryInfoUSKeyW [SHLWAPI.@] |
| * |
| * See SHRegQueryInfoUSKeyA. |
| */ |
| LONG WINAPI SHRegQueryInfoUSKeyW( |
| HUSKEY hUSKey, |
| LPDWORD pcSubKeys, |
| LPDWORD pcchMaxSubKeyLen, |
| LPDWORD pcValues, |
| LPDWORD pcchMaxValueNameLen, |
| SHREGENUM_FLAGS enumRegFlags) |
| { |
| HKEY dokey; |
| LONG ret; |
| |
| TRACE("(%p,%p,%p,%p,%p,%d)\n", |
| hUSKey,pcSubKeys,pcchMaxSubKeyLen,pcValues, |
| pcchMaxValueNameLen,enumRegFlags); |
| |
| /* if user wants HKCU, and it exists, then try it */ |
| if (((enumRegFlags == SHREGENUM_HKCU) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKCU))) { |
| ret = RegQueryInfoKeyW(dokey, 0, 0, 0, |
| pcSubKeys, pcchMaxSubKeyLen, 0, |
| pcValues, pcchMaxValueNameLen, 0, 0, 0); |
| if ((ret == ERROR_SUCCESS) || |
| (enumRegFlags == SHREGENUM_HKCU)) |
| return ret; |
| } |
| if (((enumRegFlags == SHREGENUM_HKLM) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKLM))) { |
| return RegQueryInfoKeyW(dokey, 0, 0, 0, |
| pcSubKeys, pcchMaxSubKeyLen, 0, |
| pcValues, pcchMaxValueNameLen, 0, 0, 0); |
| } |
| return ERROR_INVALID_FUNCTION; |
| } |
| |
| /************************************************************************* |
| * SHRegEnumUSKeyA [SHLWAPI.@] |
| * |
| * Enumerate a user-specific registry key. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS |
| * Failure: An error code from RegEnumKeyExA(). |
| */ |
| LONG WINAPI SHRegEnumUSKeyA( |
| HUSKEY hUSKey, /* [in] Key to enumerate */ |
| DWORD dwIndex, /* [in] Index within hUSKey */ |
| LPSTR pszName, /* [out] Name of the enumerated value */ |
| LPDWORD pcchValueNameLen, /* [in/out] Length of pszName */ |
| SHREGENUM_FLAGS enumRegFlags) /* [in] SHREGENUM_ flags from "shlwapi.h" */ |
| { |
| HKEY dokey; |
| |
| TRACE("(%p,%d,%p,%p(%d),%d)\n", |
| hUSKey, dwIndex, pszName, pcchValueNameLen, |
| *pcchValueNameLen, enumRegFlags); |
| |
| if (((enumRegFlags == SHREGENUM_HKCU) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKCU))) { |
| return RegEnumKeyExA(dokey, dwIndex, pszName, pcchValueNameLen, |
| 0, 0, 0, 0); |
| } |
| |
| if (((enumRegFlags == SHREGENUM_HKLM) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKLM))) { |
| return RegEnumKeyExA(dokey, dwIndex, pszName, pcchValueNameLen, |
| 0, 0, 0, 0); |
| } |
| FIXME("no support for SHREGENUM_BOTH\n"); |
| return ERROR_INVALID_FUNCTION; |
| } |
| |
| /************************************************************************* |
| * SHRegEnumUSKeyW [SHLWAPI.@] |
| * |
| * See SHRegEnumUSKeyA. |
| */ |
| LONG WINAPI SHRegEnumUSKeyW( |
| HUSKEY hUSKey, |
| DWORD dwIndex, |
| LPWSTR pszName, |
| LPDWORD pcchValueNameLen, |
| SHREGENUM_FLAGS enumRegFlags) |
| { |
| HKEY dokey; |
| |
| TRACE("(%p,%d,%p,%p(%d),%d)\n", |
| hUSKey, dwIndex, pszName, pcchValueNameLen, |
| *pcchValueNameLen, enumRegFlags); |
| |
| if (((enumRegFlags == SHREGENUM_HKCU) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKCU))) { |
| return RegEnumKeyExW(dokey, dwIndex, pszName, pcchValueNameLen, |
| 0, 0, 0, 0); |
| } |
| |
| if (((enumRegFlags == SHREGENUM_HKLM) || |
| (enumRegFlags == SHREGENUM_DEFAULT)) && |
| (dokey = REG_GetHKEYFromHUSKEY(hUSKey,REG_HKLM))) { |
| return RegEnumKeyExW(dokey, dwIndex, pszName, pcchValueNameLen, |
| 0, 0, 0, 0); |
| } |
| FIXME("no support for SHREGENUM_BOTH\n"); |
| return ERROR_INVALID_FUNCTION; |
| } |
| |
| |
| /************************************************************************* |
| * SHRegWriteUSValueA [SHLWAPI.@] |
| * |
| * Write a user-specific registry value. |
| * |
| * PARAMS |
| * hUSKey [I] Key to write the value to |
| * pszValue [I] Name of value under hUSKey to write the value as |
| * dwType [I] Type of the value |
| * pvData [I] Data to set as the value |
| * cbData [I] length of pvData |
| * dwFlags [I] SHREGSET_ flags from "shlwapi.h" |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. |
| * Failure: ERROR_INVALID_PARAMETER, if any parameter is invalid, otherwise |
| * an error code from RegSetValueExA(). |
| * |
| * NOTES |
| * dwFlags must have at least SHREGSET_FORCE_HKCU or SHREGSET_FORCE_HKLM set. |
| */ |
| LONG WINAPI SHRegWriteUSValueA(HUSKEY hUSKey, LPCSTR pszValue, DWORD dwType, |
| LPVOID pvData, DWORD cbData, DWORD dwFlags) |
| { |
| WCHAR szValue[MAX_PATH]; |
| |
| if (pszValue) |
| MultiByteToWideChar(CP_ACP, 0, pszValue, -1, szValue, MAX_PATH); |
| |
| return SHRegWriteUSValueW(hUSKey, pszValue ? szValue : NULL, dwType, |
| pvData, cbData, dwFlags); |
| } |
| |
| /************************************************************************* |
| * SHRegWriteUSValueW [SHLWAPI.@] |
| * |
| * See SHRegWriteUSValueA. |
| */ |
| LONG WINAPI SHRegWriteUSValueW(HUSKEY hUSKey, LPCWSTR pszValue, DWORD dwType, |
| LPVOID pvData, DWORD cbData, DWORD dwFlags) |
| { |
| DWORD dummy; |
| LPSHUSKEY hKey = (LPSHUSKEY)hUSKey; |
| LONG ret = ERROR_SUCCESS; |
| |
| TRACE("(%p,%s,%d,%p,%d,%d)\n", hUSKey, debugstr_w(pszValue), |
| dwType, pvData, cbData, dwFlags); |
| |
| if (!hUSKey || IsBadWritePtr(hUSKey, sizeof(SHUSKEY)) || |
| !(dwFlags & (SHREGSET_FORCE_HKCU|SHREGSET_FORCE_HKLM))) |
| return ERROR_INVALID_PARAMETER; |
| |
| if (dwFlags & (SHREGSET_FORCE_HKCU|SHREGSET_HKCU)) |
| { |
| if (!hKey->HKCUkey) |
| { |
| /* Create the key */ |
| ret = RegCreateKeyW(hKey->HKCUstart, hKey->lpszPath, &hKey->HKCUkey); |
| TRACE("Creating HKCU key, ret = %d\n", ret); |
| if (ret && (dwFlags & (SHREGSET_FORCE_HKCU))) |
| { |
| hKey->HKCUkey = 0; |
| return ret; |
| } |
| } |
| |
| if (!ret) |
| { |
| if ((dwFlags & SHREGSET_FORCE_HKCU) || |
| RegQueryValueExW(hKey->HKCUkey, pszValue, NULL, NULL, NULL, &dummy)) |
| { |
| /* Doesn't exist or we are forcing: Write value */ |
| ret = RegSetValueExW(hKey->HKCUkey, pszValue, 0, dwType, pvData, cbData); |
| TRACE("Writing HKCU value, ret = %d\n", ret); |
| } |
| } |
| } |
| |
| if (dwFlags & (SHREGSET_FORCE_HKLM|SHREGSET_HKLM)) |
| { |
| if (!hKey->HKLMkey) |
| { |
| /* Create the key */ |
| ret = RegCreateKeyW(hKey->HKLMstart, hKey->lpszPath, &hKey->HKLMkey); |
| TRACE("Creating HKLM key, ret = %d\n", ret); |
| if (ret && (dwFlags & (SHREGSET_FORCE_HKLM))) |
| { |
| hKey->HKLMkey = 0; |
| return ret; |
| } |
| } |
| |
| if (!ret) |
| { |
| if ((dwFlags & SHREGSET_FORCE_HKLM) || |
| RegQueryValueExW(hKey->HKLMkey, pszValue, NULL, NULL, NULL, &dummy)) |
| { |
| /* Doesn't exist or we are forcing: Write value */ |
| ret = RegSetValueExW(hKey->HKLMkey, pszValue, 0, dwType, pvData, cbData); |
| TRACE("Writing HKLM value, ret = %d\n", ret); |
| } |
| } |
| } |
| |
| return ret; |
| } |
| |
| /************************************************************************* |
| * SHRegGetPathA [SHLWAPI.@] |
| * |
| * Get a path from the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key containing path to get |
| * lpszValue [I] Name of value containing path to get |
| * lpszPath [O] Buffer for returned path |
| * dwFlags [I] Reserved |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. lpszPath contains the path. |
| * Failure: An error code from RegOpenKeyExA() or SHQueryValueExA(). |
| */ |
| DWORD WINAPI SHRegGetPathA(HKEY hKey, LPCSTR lpszSubKey, LPCSTR lpszValue, |
| LPSTR lpszPath, DWORD dwFlags) |
| { |
| DWORD dwSize = MAX_PATH; |
| |
| TRACE("(hkey=%p,%s,%s,%p,%d)\n", hKey, debugstr_a(lpszSubKey), |
| debugstr_a(lpszValue), lpszPath, dwFlags); |
| |
| return SHGetValueA(hKey, lpszSubKey, lpszValue, 0, lpszPath, &dwSize); |
| } |
| |
| /************************************************************************* |
| * SHRegGetPathW [SHLWAPI.@] |
| * |
| * See SHRegGetPathA. |
| */ |
| DWORD WINAPI SHRegGetPathW(HKEY hKey, LPCWSTR lpszSubKey, LPCWSTR lpszValue, |
| LPWSTR lpszPath, DWORD dwFlags) |
| { |
| DWORD dwSize = MAX_PATH; |
| |
| TRACE("(hkey=%p,%s,%s,%p,%d)\n", hKey, debugstr_w(lpszSubKey), |
| debugstr_w(lpszValue), lpszPath, dwFlags); |
| |
| return SHGetValueW(hKey, lpszSubKey, lpszValue, 0, lpszPath, &dwSize); |
| } |
| |
| |
| /************************************************************************* |
| * SHRegSetPathA [SHLWAPI.@] |
| * |
| * Write a path to the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key containing path to set |
| * lpszValue [I] Name of value containing path to set |
| * lpszPath [O] Path to write |
| * dwFlags [I] Reserved, must be 0. |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. |
| * Failure: An error code from SHSetValueA(). |
| */ |
| DWORD WINAPI SHRegSetPathA(HKEY hKey, LPCSTR lpszSubKey, LPCSTR lpszValue, |
| LPCSTR lpszPath, DWORD dwFlags) |
| { |
| char szBuff[MAX_PATH]; |
| |
| FIXME("(hkey=%p,%s,%s,%p,%d) - semi-stub\n",hKey, debugstr_a(lpszSubKey), |
| debugstr_a(lpszValue), lpszPath, dwFlags); |
| |
| lstrcpyA(szBuff, lpszPath); |
| |
| /* FIXME: PathUnExpandEnvStringsA(szBuff); */ |
| |
| return SHSetValueA(hKey,lpszSubKey, lpszValue, REG_SZ, szBuff, |
| lstrlenA(szBuff)); |
| } |
| |
| /************************************************************************* |
| * SHRegSetPathW [SHLWAPI.@] |
| * |
| * See SHRegSetPathA. |
| */ |
| DWORD WINAPI SHRegSetPathW(HKEY hKey, LPCWSTR lpszSubKey, LPCWSTR lpszValue, |
| LPCWSTR lpszPath, DWORD dwFlags) |
| { |
| WCHAR szBuff[MAX_PATH]; |
| |
| FIXME("(hkey=%p,%s,%s,%p,%d) - semi-stub\n",hKey, debugstr_w(lpszSubKey), |
| debugstr_w(lpszValue), lpszPath, dwFlags); |
| |
| lstrcpyW(szBuff, lpszPath); |
| |
| /* FIXME: PathUnExpandEnvStringsW(szBuff); */ |
| |
| return SHSetValueW(hKey,lpszSubKey, lpszValue, REG_SZ, szBuff, |
| lstrlenW(szBuff)); |
| } |
| |
| /************************************************************************* |
| * SHGetValueA [SHLWAPI.@] |
| * |
| * Get a value from the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key containing value to get |
| * lpszValue [I] Name of value to get |
| * pwType [O] Pointer to the values type |
| * pvData [O] Pointer to the values data |
| * pcbData [O] Pointer to the values size |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. Output parameters contain the details read. |
| * Failure: An error code from RegOpenKeyExA() or SHQueryValueExA(). |
| */ |
| DWORD WINAPI SHGetValueA(HKEY hKey, LPCSTR lpszSubKey, LPCSTR lpszValue, |
| LPDWORD pwType, LPVOID pvData, LPDWORD pcbData) |
| { |
| DWORD dwRet = 0; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p,%s,%s,%p,%p,%p)\n", hKey, debugstr_a(lpszSubKey), |
| debugstr_a(lpszValue), pwType, pvData, pcbData); |
| |
| /* lpszSubKey can be 0. In this case the value is taken from the |
| * current key. |
| */ |
| if(lpszSubKey) |
| dwRet = RegOpenKeyExA(hKey, lpszSubKey, 0, KEY_QUERY_VALUE, &hSubKey); |
| |
| if (! dwRet) |
| { |
| /* SHQueryValueEx expands Environment strings */ |
| dwRet = SHQueryValueExA(hSubKey ? hSubKey : hKey, lpszValue, 0, pwType, pvData, pcbData); |
| if (hSubKey) RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHRegGetValueA [SHLWAPI.@] |
| * |
| * Get a value from the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key containing value to get |
| * lpszValue [I] Name of value to get |
| * srrf [I] Flags for restricting returned data |
| * pwType [O] Pointer to the values type |
| * pvData [O] Pointer to the values data |
| * pcbData [O] Pointer to the values size |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. Output parameters contain the details read. |
| * Failure: An error code from RegOpenKeyExA() or SHQueryValueExA(). |
| */ |
| DWORD WINAPI SHRegGetValueA(HKEY hKey, LPCSTR lpszSubKey, LPCSTR lpszValue, DWORD srrfFlags, |
| LPDWORD pwType, LPVOID pvData, LPDWORD pcbData) |
| { |
| DWORD dwRet = 0; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p,%s,%s,%p,%p,%p)\n", hKey, debugstr_a(lpszSubKey), |
| debugstr_a(lpszValue), pwType, pvData, pcbData); |
| FIXME("Semi-Stub: Find meaning and implement handling of SRFF Flags 0x%08x\n", srrfFlags); |
| |
| dwRet = RegOpenKeyExA(hKey, lpszSubKey, 0, KEY_QUERY_VALUE, &hSubKey); |
| if (! dwRet) |
| { |
| /* SHQueryValueEx expands Environment strings */ |
| dwRet = SHQueryValueExA(hSubKey, lpszValue, 0, pwType, pvData, pcbData); |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHReg GetRegValueW [SHLWAPI.@] |
| * |
| * See SHGetValueA. |
| */ |
| DWORD WINAPI SHRegGetValueW(HKEY hKey, LPCWSTR lpszSubKey, LPCWSTR lpszValue, DWORD srrfFlags, |
| LPDWORD pwType, LPVOID pvData, LPDWORD pcbData) |
| { |
| DWORD dwRet = 0; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p,%s,%s,0x%08x, %p,%p,%p)\n", hKey, debugstr_w(lpszSubKey), |
| debugstr_w(lpszValue), srrfFlags,pwType, pvData, pcbData); |
| FIXME("Semi-Stub: Find meaning and implement handling of SRFF Flags 0x%08x\n", srrfFlags); |
| |
| dwRet = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_QUERY_VALUE, &hSubKey); |
| if (! dwRet) |
| { |
| dwRet = SHQueryValueExW(hSubKey, lpszValue, 0, pwType, pvData, pcbData); |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHGetValueW [SHLWAPI.@] |
| * |
| * See SHGetValueA. |
| */ |
| DWORD WINAPI SHGetValueW(HKEY hKey, LPCWSTR lpszSubKey, LPCWSTR lpszValue, |
| LPDWORD pwType, LPVOID pvData, LPDWORD pcbData) |
| { |
| DWORD dwRet = 0; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p,%s,%s,%p,%p,%p)\n", hKey, debugstr_w(lpszSubKey), |
| debugstr_w(lpszValue), pwType, pvData, pcbData); |
| |
| if(lpszSubKey) |
| dwRet = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_QUERY_VALUE, &hSubKey); |
| |
| if (! dwRet) |
| { |
| dwRet = SHQueryValueExW(hSubKey ? hSubKey : hKey, lpszValue, 0, pwType, pvData, pcbData); |
| if (hSubKey) RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHSetValueA [SHLWAPI.@] |
| * |
| * Set a value in the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key under hKey |
| * lpszValue [I] Name of value to set |
| * dwType [I] Type of the value |
| * pvData [I] Data of the value |
| * cbData [I] Size of the value |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. The value is set with the data given. |
| * Failure: An error code from RegCreateKeyExA() or RegSetValueExA() |
| * |
| * NOTES |
| * If lpszSubKey does not exist, it is created before the value is set. If |
| * lpszSubKey is NULL or an empty string, then the value is added directly |
| * to hKey instead. |
| */ |
| DWORD WINAPI SHSetValueA(HKEY hKey, LPCSTR lpszSubKey, LPCSTR lpszValue, |
| DWORD dwType, LPCVOID pvData, DWORD cbData) |
| { |
| DWORD dwRet = ERROR_SUCCESS, dwDummy; |
| HKEY hSubKey; |
| |
| TRACE("(hkey=%p,%s,%s,%d,%p,%d)\n", hKey, debugstr_a(lpszSubKey), |
| debugstr_a(lpszValue), dwType, pvData, cbData); |
| |
| if (lpszSubKey && *lpszSubKey) |
| dwRet = RegCreateKeyExA(hKey, lpszSubKey, 0, NULL, |
| 0, KEY_SET_VALUE, NULL, &hSubKey, &dwDummy); |
| else |
| hSubKey = hKey; |
| if (!dwRet) |
| { |
| dwRet = RegSetValueExA(hSubKey, lpszValue, 0, dwType, pvData, cbData); |
| if (hSubKey != hKey) |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHSetValueW [SHLWAPI.@] |
| * |
| * See SHSetValueA. |
| */ |
| DWORD WINAPI SHSetValueW(HKEY hKey, LPCWSTR lpszSubKey, LPCWSTR lpszValue, |
| DWORD dwType, LPCVOID pvData, DWORD cbData) |
| { |
| DWORD dwRet = ERROR_SUCCESS, dwDummy; |
| HKEY hSubKey; |
| |
| TRACE("(hkey=%p,%s,%s,%d,%p,%d)\n", hKey, debugstr_w(lpszSubKey), |
| debugstr_w(lpszValue), dwType, pvData, cbData); |
| |
| if (lpszSubKey && *lpszSubKey) |
| dwRet = RegCreateKeyExW(hKey, lpszSubKey, 0, NULL, |
| 0, KEY_SET_VALUE, NULL, &hSubKey, &dwDummy); |
| else |
| hSubKey = hKey; |
| if (!dwRet) |
| { |
| dwRet = RegSetValueExW(hSubKey, lpszValue, 0, dwType, pvData, cbData); |
| if (hSubKey != hKey) |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHQueryInfoKeyA [SHLWAPI.@] |
| * |
| * Get information about a registry key. See RegQueryInfoKeyA(). |
| * |
| * RETURNS |
| * The result of calling RegQueryInfoKeyA(). |
| */ |
| LONG WINAPI SHQueryInfoKeyA(HKEY hKey, LPDWORD pwSubKeys, LPDWORD pwSubKeyMax, |
| LPDWORD pwValues, LPDWORD pwValueMax) |
| { |
| TRACE("(hkey=%p,%p,%p,%p,%p)\n", hKey, pwSubKeys, pwSubKeyMax, |
| pwValues, pwValueMax); |
| return RegQueryInfoKeyA(hKey, NULL, NULL, NULL, pwSubKeys, pwSubKeyMax, |
| NULL, pwValues, pwValueMax, NULL, NULL, NULL); |
| } |
| |
| /************************************************************************* |
| * SHQueryInfoKeyW [SHLWAPI.@] |
| * |
| * See SHQueryInfoKeyA. |
| */ |
| LONG WINAPI SHQueryInfoKeyW(HKEY hKey, LPDWORD pwSubKeys, LPDWORD pwSubKeyMax, |
| LPDWORD pwValues, LPDWORD pwValueMax) |
| { |
| TRACE("(hkey=%p,%p,%p,%p,%p)\n", hKey, pwSubKeys, pwSubKeyMax, |
| pwValues, pwValueMax); |
| return RegQueryInfoKeyW(hKey, NULL, NULL, NULL, pwSubKeys, pwSubKeyMax, |
| NULL, pwValues, pwValueMax, NULL, NULL, NULL); |
| } |
| |
| /************************************************************************* |
| * SHQueryValueExA [SHLWAPI.@] |
| * |
| * Get a value from the registry, expanding environment variable strings. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszValue [I] Name of value to query |
| * lpReserved [O] Reserved for future use; must be NULL |
| * pwType [O] Optional pointer updated with the values type |
| * pvData [O] Optional pointer updated with the values data |
| * pcbData [O] Optional pointer updated with the values size |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. Any non NULL output parameters are updated with |
| * information about the value. |
| * Failure: ERROR_OUTOFMEMORY if memory allocation fails, or the type of the |
| * data is REG_EXPAND_SZ and pcbData is NULL. Otherwise an error |
| * code from RegQueryValueExA() or ExpandEnvironmentStringsA(). |
| * |
| * NOTES |
| * Either pwType, pvData or pcbData may be NULL if the caller doesn't want |
| * the type, data or size information for the value. |
| * |
| * If the type of the data is REG_EXPAND_SZ, it is expanded to REG_SZ. The |
| * value returned will be truncated if it is of type REG_SZ and bigger than |
| * the buffer given to store it. |
| * |
| * REG_EXPAND_SZ: |
| * case-1: the unexpanded string is smaller than the expanded one |
| * subcase-1: the buffer is too small to hold the unexpanded string: |
| * function fails and returns the size of the unexpanded string. |
| * |
| * subcase-2: buffer is too small to hold the expanded string: |
| * the function return success (!!) and the result is truncated |
| * *** This is clearly an error in the native implementation. *** |
| * |
| * case-2: the unexpanded string is bigger than the expanded one |
| * The buffer must have enough space to hold the unexpanded |
| * string even if the result is smaller. |
| * |
| */ |
| DWORD WINAPI SHQueryValueExA( HKEY hKey, LPCSTR lpszValue, |
| LPDWORD lpReserved, LPDWORD pwType, |
| LPVOID pvData, LPDWORD pcbData) |
| { |
| DWORD dwRet, dwType, dwUnExpDataLen = 0, dwExpDataLen; |
| |
| TRACE("(hkey=%p,%s,%p,%p,%p,%p=%d)\n", hKey, debugstr_a(lpszValue), |
| lpReserved, pwType, pvData, pcbData, pcbData ? *pcbData : 0); |
| |
| if (pcbData) dwUnExpDataLen = *pcbData; |
| |
| dwRet = RegQueryValueExA(hKey, lpszValue, lpReserved, &dwType, pvData, &dwUnExpDataLen); |
| |
| if (pcbData && (dwType == REG_EXPAND_SZ)) |
| { |
| DWORD nBytesToAlloc; |
| |
| /* Expand type REG_EXPAND_SZ into REG_SZ */ |
| LPSTR szData; |
| |
| /* If the caller didn't supply a buffer or the buffer is too small we have |
| * to allocate our own |
| */ |
| if ((!pvData) || (dwRet == ERROR_MORE_DATA) ) |
| { |
| char cNull = '\0'; |
| nBytesToAlloc = dwUnExpDataLen; |
| |
| szData = (LPSTR) LocalAlloc(LMEM_ZEROINIT, nBytesToAlloc); |
| RegQueryValueExA (hKey, lpszValue, lpReserved, NULL, (LPBYTE)szData, &nBytesToAlloc); |
| dwExpDataLen = ExpandEnvironmentStringsA(szData, &cNull, 1); |
| dwUnExpDataLen = max(nBytesToAlloc, dwExpDataLen); |
| LocalFree((HLOCAL) szData); |
| } |
| else |
| { |
| nBytesToAlloc = (lstrlenA(pvData)+1) * sizeof (CHAR); |
| szData = (LPSTR) LocalAlloc(LMEM_ZEROINIT, nBytesToAlloc ); |
| lstrcpyA(szData, pvData); |
| dwExpDataLen = ExpandEnvironmentStringsA(szData, pvData, *pcbData / sizeof(CHAR)); |
| if (dwExpDataLen > *pcbData) dwRet = ERROR_MORE_DATA; |
| dwUnExpDataLen = max(nBytesToAlloc, dwExpDataLen); |
| LocalFree((HLOCAL) szData); |
| } |
| } |
| |
| /* Update the type and data size if the caller wanted them */ |
| if ( dwType == REG_EXPAND_SZ ) dwType = REG_SZ; |
| if ( pwType ) *pwType = dwType; |
| if ( pcbData ) *pcbData = dwUnExpDataLen; |
| return dwRet; |
| } |
| |
| |
| /************************************************************************* |
| * SHQueryValueExW [SHLWAPI.@] |
| * |
| * See SHQueryValueExA. |
| */ |
| DWORD WINAPI SHQueryValueExW(HKEY hKey, LPCWSTR lpszValue, |
| LPDWORD lpReserved, LPDWORD pwType, |
| LPVOID pvData, LPDWORD pcbData) |
| { |
| DWORD dwRet, dwType, dwUnExpDataLen = 0, dwExpDataLen; |
| |
| TRACE("(hkey=%p,%s,%p,%p,%p,%p=%d)\n", hKey, debugstr_w(lpszValue), |
| lpReserved, pwType, pvData, pcbData, pcbData ? *pcbData : 0); |
| |
| if (pcbData) dwUnExpDataLen = *pcbData; |
| |
| dwRet = RegQueryValueExW(hKey, lpszValue, lpReserved, &dwType, pvData, &dwUnExpDataLen); |
| if (dwRet!=ERROR_SUCCESS && dwRet!=ERROR_MORE_DATA) |
| return dwRet; |
| |
| if (pcbData && (dwType == REG_EXPAND_SZ)) |
| { |
| DWORD nBytesToAlloc; |
| |
| /* Expand type REG_EXPAND_SZ into REG_SZ */ |
| LPWSTR szData; |
| |
| /* If the caller didn't supply a buffer or the buffer is too small we have |
| * to allocate our own |
| */ |
| if ((!pvData) || (dwRet == ERROR_MORE_DATA) ) |
| { |
| WCHAR cNull = '\0'; |
| nBytesToAlloc = dwUnExpDataLen; |
| |
| szData = (LPWSTR) LocalAlloc(LMEM_ZEROINIT, nBytesToAlloc); |
| RegQueryValueExW (hKey, lpszValue, lpReserved, NULL, (LPBYTE)szData, &nBytesToAlloc); |
| dwExpDataLen = ExpandEnvironmentStringsW(szData, &cNull, 1); |
| dwUnExpDataLen = max(nBytesToAlloc, dwExpDataLen); |
| LocalFree((HLOCAL) szData); |
| } |
| else |
| { |
| nBytesToAlloc = (lstrlenW(pvData) + 1) * sizeof(WCHAR); |
| szData = (LPWSTR) LocalAlloc(LMEM_ZEROINIT, nBytesToAlloc ); |
| lstrcpyW(szData, pvData); |
| dwExpDataLen = ExpandEnvironmentStringsW(szData, pvData, *pcbData/sizeof(WCHAR) ); |
| if (dwExpDataLen > *pcbData) dwRet = ERROR_MORE_DATA; |
| dwUnExpDataLen = max(nBytesToAlloc, dwExpDataLen); |
| LocalFree((HLOCAL) szData); |
| } |
| } |
| |
| /* Update the type and data size if the caller wanted them */ |
| if ( dwType == REG_EXPAND_SZ ) dwType = REG_SZ; |
| if ( pwType ) *pwType = dwType; |
| if ( pcbData ) *pcbData = dwUnExpDataLen; |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteKeyA [SHLWAPI.@] |
| * |
| * Delete a registry key and any sub keys/values present |
| * |
| * This function forwards to the unicode version directly, to avoid |
| * handling subkeys that are not representable in ASCII. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key to delete |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. The key is deleted. |
| * Failure: An error code from RegOpenKeyExA(), RegQueryInfoKeyA(), |
| * RegEnumKeyExA() or RegDeleteKeyA(). |
| */ |
| DWORD WINAPI SHDeleteKeyA(HKEY hKey, LPCSTR lpszSubKey) |
| { |
| WCHAR subkeyW[MAX_PATH]; |
| |
| MultiByteToWideChar (CP_ACP, 0, lpszSubKey, -1, subkeyW, sizeof(subkeyW)/sizeof(WCHAR)); |
| return SHDeleteKeyW(hKey, subkeyW); |
| } |
| |
| /************************************************************************* |
| * SHDeleteKeyW [SHLWAPI.@] |
| * |
| * See SHDeleteKeyA. |
| */ |
| DWORD WINAPI SHDeleteKeyW(HKEY hKey, LPCWSTR lpszSubKey) |
| { |
| DWORD dwRet, dwMaxSubkeyLen = 0, dwSize; |
| WCHAR szNameBuf[MAX_PATH], *lpszName = szNameBuf; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p,%s)\n", hKey, debugstr_w(lpszSubKey)); |
| |
| dwRet = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_READ, &hSubKey); |
| if(!dwRet) |
| { |
| /* Find the maximum subkey length so that we can allocate a buffer */ |
| dwRet = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, NULL, |
| &dwMaxSubkeyLen, NULL, NULL, NULL, NULL, NULL, NULL); |
| if(!dwRet) |
| { |
| dwMaxSubkeyLen++; |
| if (dwMaxSubkeyLen > sizeof(szNameBuf)/sizeof(WCHAR)) |
| /* Name too big: alloc a buffer for it */ |
| lpszName = HeapAlloc(GetProcessHeap(), 0, dwMaxSubkeyLen*sizeof(WCHAR)); |
| |
| if(!lpszName) |
| dwRet = ERROR_NOT_ENOUGH_MEMORY; |
| else |
| { |
| while (dwRet == ERROR_SUCCESS) |
| { |
| dwSize = dwMaxSubkeyLen; |
| dwRet = RegEnumKeyExW(hSubKey, 0, lpszName, &dwSize, NULL, NULL, NULL, NULL); |
| if (dwRet == ERROR_SUCCESS || dwRet == ERROR_MORE_DATA) |
| dwRet = SHDeleteKeyW(hSubKey, lpszName); |
| } |
| if (dwRet == ERROR_NO_MORE_ITEMS) |
| dwRet = ERROR_SUCCESS; |
| |
| if (lpszName != szNameBuf) |
| HeapFree(GetProcessHeap(), 0, lpszName); /* Free buffer if allocated */ |
| } |
| } |
| |
| RegCloseKey(hSubKey); |
| if(!dwRet) |
| dwRet = RegDeleteKeyW(hKey, lpszSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteEmptyKeyA [SHLWAPI.@] |
| * |
| * Delete a registry key with no sub keys. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key to delete |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. The key is deleted. |
| * Failure: If the key is not empty, returns ERROR_KEY_HAS_CHILDREN. Otherwise |
| * returns an error code from RegOpenKeyExA(), RegQueryInfoKeyA() or |
| * RegDeleteKeyA(). |
| */ |
| DWORD WINAPI SHDeleteEmptyKeyA(HKEY hKey, LPCSTR lpszSubKey) |
| { |
| DWORD dwRet, dwKeyCount = 0; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p,%s)\n", hKey, debugstr_a(lpszSubKey)); |
| |
| dwRet = RegOpenKeyExA(hKey, lpszSubKey, 0, KEY_READ, &hSubKey); |
| if(!dwRet) |
| { |
| dwRet = RegQueryInfoKeyA(hSubKey, NULL, NULL, NULL, &dwKeyCount, |
| NULL, NULL, NULL, NULL, NULL, NULL, NULL); |
| RegCloseKey(hSubKey); |
| if(!dwRet) |
| { |
| if (!dwKeyCount) |
| dwRet = RegDeleteKeyA(hKey, lpszSubKey); |
| else |
| dwRet = ERROR_KEY_HAS_CHILDREN; |
| } |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteEmptyKeyW [SHLWAPI.@] |
| * |
| * See SHDeleteEmptyKeyA. |
| */ |
| DWORD WINAPI SHDeleteEmptyKeyW(HKEY hKey, LPCWSTR lpszSubKey) |
| { |
| DWORD dwRet, dwKeyCount = 0; |
| HKEY hSubKey = 0; |
| |
| TRACE("(hkey=%p, %s)\n", hKey, debugstr_w(lpszSubKey)); |
| |
| dwRet = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_READ, &hSubKey); |
| if(!dwRet) |
| { |
| dwRet = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, &dwKeyCount, |
| NULL, NULL, NULL, NULL, NULL, NULL, NULL); |
| RegCloseKey(hSubKey); |
| if(!dwRet) |
| { |
| if (!dwKeyCount) |
| dwRet = RegDeleteKeyW(hKey, lpszSubKey); |
| else |
| dwRet = ERROR_KEY_HAS_CHILDREN; |
| } |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteOrphanKeyA [SHLWAPI.@] |
| * |
| * Delete a registry key with no sub keys or values. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key to possibly delete |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. The key has been deleted if it was an orphan. |
| * Failure: An error from RegOpenKeyExA(), RegQueryValueExA(), or RegDeleteKeyA(). |
| */ |
| DWORD WINAPI SHDeleteOrphanKeyA(HKEY hKey, LPCSTR lpszSubKey) |
| { |
| HKEY hSubKey; |
| DWORD dwKeyCount = 0, dwValueCount = 0, dwRet; |
| |
| TRACE("(hkey=%p,%s)\n", hKey, debugstr_a(lpszSubKey)); |
| |
| dwRet = RegOpenKeyExA(hKey, lpszSubKey, 0, KEY_READ, &hSubKey); |
| |
| if(!dwRet) |
| { |
| /* Get subkey and value count */ |
| dwRet = RegQueryInfoKeyA(hSubKey, NULL, NULL, NULL, &dwKeyCount, |
| NULL, NULL, &dwValueCount, NULL, NULL, NULL, NULL); |
| |
| if(!dwRet && !dwKeyCount && !dwValueCount) |
| { |
| dwRet = RegDeleteKeyA(hKey, lpszSubKey); |
| } |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteOrphanKeyW [SHLWAPI.@] |
| * |
| * See SHDeleteOrphanKeyA. |
| */ |
| DWORD WINAPI SHDeleteOrphanKeyW(HKEY hKey, LPCWSTR lpszSubKey) |
| { |
| HKEY hSubKey; |
| DWORD dwKeyCount = 0, dwValueCount = 0, dwRet; |
| |
| TRACE("(hkey=%p,%s)\n", hKey, debugstr_w(lpszSubKey)); |
| |
| dwRet = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_READ, &hSubKey); |
| |
| if(!dwRet) |
| { |
| /* Get subkey and value count */ |
| dwRet = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, &dwKeyCount, |
| NULL, NULL, &dwValueCount, NULL, NULL, NULL, NULL); |
| |
| if(!dwRet && !dwKeyCount && !dwValueCount) |
| { |
| dwRet = RegDeleteKeyW(hKey, lpszSubKey); |
| } |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteValueA [SHLWAPI.@] |
| * |
| * Delete a value from the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * lpszSubKey [I] Name of sub key containing value to delete |
| * lpszValue [I] Name of value to delete |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. The value is deleted. |
| * Failure: An error code from RegOpenKeyExA() or RegDeleteValueA(). |
| */ |
| DWORD WINAPI SHDeleteValueA(HKEY hKey, LPCSTR lpszSubKey, LPCSTR lpszValue) |
| { |
| DWORD dwRet; |
| HKEY hSubKey; |
| |
| TRACE("(hkey=%p,%s,%s)\n", hKey, debugstr_a(lpszSubKey), debugstr_a(lpszValue)); |
| |
| dwRet = RegOpenKeyExA(hKey, lpszSubKey, 0, KEY_SET_VALUE, &hSubKey); |
| if (!dwRet) |
| { |
| dwRet = RegDeleteValueA(hSubKey, lpszValue); |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHDeleteValueW [SHLWAPI.@] |
| * |
| * See SHDeleteValueA. |
| */ |
| DWORD WINAPI SHDeleteValueW(HKEY hKey, LPCWSTR lpszSubKey, LPCWSTR lpszValue) |
| { |
| DWORD dwRet; |
| HKEY hSubKey; |
| |
| TRACE("(hkey=%p,%s,%s)\n", hKey, debugstr_w(lpszSubKey), debugstr_w(lpszValue)); |
| |
| dwRet = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_SET_VALUE, &hSubKey); |
| if (!dwRet) |
| { |
| dwRet = RegDeleteValueW(hSubKey, lpszValue); |
| RegCloseKey(hSubKey); |
| } |
| return dwRet; |
| } |
| |
| /************************************************************************* |
| * SHEnumKeyExA [SHLWAPI.@] |
| * |
| * Enumerate sub keys in a registry key. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * dwIndex [I] Index of key to enumerate |
| * lpszSubKey [O] Pointer updated with the subkey name |
| * pwLen [O] Pointer updated with the subkey length |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. lpszSubKey and pwLen are updated. |
| * Failure: An error code from RegEnumKeyExA(). |
| */ |
| LONG WINAPI SHEnumKeyExA(HKEY hKey, DWORD dwIndex, LPSTR lpszSubKey, |
| LPDWORD pwLen) |
| { |
| TRACE("(hkey=%p,%d,%s,%p)\n", hKey, dwIndex, debugstr_a(lpszSubKey), pwLen); |
| |
| return RegEnumKeyExA(hKey, dwIndex, lpszSubKey, pwLen, NULL, NULL, NULL, NULL); |
| } |
| |
| /************************************************************************* |
| * SHEnumKeyExW [SHLWAPI.@] |
| * |
| * See SHEnumKeyExA. |
| */ |
| LONG WINAPI SHEnumKeyExW(HKEY hKey, DWORD dwIndex, LPWSTR lpszSubKey, |
| LPDWORD pwLen) |
| { |
| TRACE("(hkey=%p,%d,%s,%p)\n", hKey, dwIndex, debugstr_w(lpszSubKey), pwLen); |
| |
| return RegEnumKeyExW(hKey, dwIndex, lpszSubKey, pwLen, NULL, NULL, NULL, NULL); |
| } |
| |
| /************************************************************************* |
| * SHEnumValueA [SHLWAPI.@] |
| * |
| * Enumerate values in a registry key. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * dwIndex [I] Index of key to enumerate |
| * lpszValue [O] Pointer updated with the values name |
| * pwLen [O] Pointer updated with the values length |
| * pwType [O] Pointer updated with the values type |
| * pvData [O] Pointer updated with the values data |
| * pcbData [O] Pointer updated with the values size |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. Output parameters are updated. |
| * Failure: An error code from RegEnumValueA(). |
| */ |
| LONG WINAPI SHEnumValueA(HKEY hKey, DWORD dwIndex, LPSTR lpszValue, |
| LPDWORD pwLen, LPDWORD pwType, |
| LPVOID pvData, LPDWORD pcbData) |
| { |
| TRACE("(hkey=%p,%d,%s,%p,%p,%p,%p)\n", hKey, dwIndex, |
| debugstr_a(lpszValue), pwLen, pwType, pvData, pcbData); |
| |
| return RegEnumValueA(hKey, dwIndex, lpszValue, pwLen, NULL, |
| pwType, pvData, pcbData); |
| } |
| |
| /************************************************************************* |
| * SHEnumValueW [SHLWAPI.@] |
| * |
| * See SHEnumValueA. |
| */ |
| LONG WINAPI SHEnumValueW(HKEY hKey, DWORD dwIndex, LPWSTR lpszValue, |
| LPDWORD pwLen, LPDWORD pwType, |
| LPVOID pvData, LPDWORD pcbData) |
| { |
| TRACE("(hkey=%p,%d,%s,%p,%p,%p,%p)\n", hKey, dwIndex, |
| debugstr_w(lpszValue), pwLen, pwType, pvData, pcbData); |
| |
| return RegEnumValueW(hKey, dwIndex, lpszValue, pwLen, NULL, |
| pwType, pvData, pcbData); |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.205] |
| * |
| * Get a value from the registry. |
| * |
| * PARAMS |
| * hKey [I] Handle to registry key |
| * pSubKey [I] Name of sub key containing value to get |
| * pValue [I] Name of value to get |
| * pwType [O] Destination for the values type |
| * pvData [O] Destination for the values data |
| * pbData [O] Destination for the values size |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. Output parameters contain the details read. |
| * Failure: An error code from RegOpenKeyExA() or SHQueryValueExA(), |
| * or ERROR_INVALID_FUNCTION in the machine is in safe mode. |
| */ |
| DWORD WINAPI SHGetValueGoodBootA(HKEY hkey, LPCSTR pSubKey, LPCSTR pValue, |
| LPDWORD pwType, LPVOID pvData, LPDWORD pbData) |
| { |
| if (GetSystemMetrics(SM_CLEANBOOT)) |
| return ERROR_INVALID_FUNCTION; |
| return SHGetValueA(hkey, pSubKey, pValue, pwType, pvData, pbData); |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.206] |
| * |
| * Unicode version of SHGetValueGoodBootW. |
| */ |
| DWORD WINAPI SHGetValueGoodBootW(HKEY hkey, LPCWSTR pSubKey, LPCWSTR pValue, |
| LPDWORD pwType, LPVOID pvData, LPDWORD pbData) |
| { |
| if (GetSystemMetrics(SM_CLEANBOOT)) |
| return ERROR_INVALID_FUNCTION; |
| return SHGetValueW(hkey, pSubKey, pValue, pwType, pvData, pbData); |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.320] |
| * |
| * Set a MIME content type in the registry. |
| * |
| * PARAMS |
| * lpszSubKey [I] Name of key under HKEY_CLASSES_ROOT. |
| * lpszValue [I] Value to set |
| * |
| * RETURNS |
| * Success: TRUE |
| * Failure: FALSE |
| */ |
| BOOL WINAPI RegisterMIMETypeForExtensionA(LPCSTR lpszSubKey, LPCSTR lpszValue) |
| { |
| DWORD dwRet; |
| |
| if (!lpszValue) |
| { |
| WARN("Invalid lpszValue would crash under Win32!\n"); |
| return FALSE; |
| } |
| |
| dwRet = SHSetValueA(HKEY_CLASSES_ROOT, lpszSubKey, lpszContentTypeA, |
| REG_SZ, lpszValue, strlen(lpszValue)); |
| return dwRet ? FALSE : TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.321] |
| * |
| * Unicode version of RegisterMIMETypeForExtensionA. |
| */ |
| BOOL WINAPI RegisterMIMETypeForExtensionW(LPCWSTR lpszSubKey, LPCWSTR lpszValue) |
| { |
| DWORD dwRet; |
| |
| if (!lpszValue) |
| { |
| WARN("Invalid lpszValue would crash under Win32!\n"); |
| return FALSE; |
| } |
| |
| dwRet = SHSetValueW(HKEY_CLASSES_ROOT, lpszSubKey, lpszContentTypeW, |
| REG_SZ, lpszValue, strlenW(lpszValue)); |
| return dwRet ? FALSE : TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.322] |
| * |
| * Delete a MIME content type from the registry. |
| * |
| * PARAMS |
| * lpszSubKey [I] Name of sub key |
| * |
| * RETURNS |
| * Success: TRUE |
| * Failure: FALSE |
| */ |
| BOOL WINAPI UnregisterMIMETypeForExtensionA(LPCSTR lpszSubKey) |
| { |
| HRESULT ret = SHDeleteValueA(HKEY_CLASSES_ROOT, lpszSubKey, lpszContentTypeA); |
| return ret ? FALSE : TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.323] |
| * |
| * Unicode version of UnregisterMIMETypeForExtensionA. |
| */ |
| BOOL WINAPI UnregisterMIMETypeForExtensionW(LPCWSTR lpszSubKey) |
| { |
| HRESULT ret = SHDeleteValueW(HKEY_CLASSES_ROOT, lpszSubKey, lpszContentTypeW); |
| return ret ? FALSE : TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.328] |
| * |
| * Get the registry path to a MIME content key. |
| * |
| * PARAMS |
| * lpszType [I] Content type to get the path for |
| * lpszBuffer [O] Destination for path |
| * dwLen [I] Length of lpszBuffer |
| * |
| * RETURNS |
| * Success: TRUE. lpszBuffer contains the full path. |
| * Failure: FALSE. |
| * |
| * NOTES |
| * The base path for the key is "MIME\Database\Content Type\" |
| */ |
| BOOL WINAPI GetMIMETypeSubKeyA(LPCSTR lpszType, LPSTR lpszBuffer, DWORD dwLen) |
| { |
| TRACE("(%s,%p,%d)\n", debugstr_a(lpszType), lpszBuffer, dwLen); |
| |
| if (dwLen > dwLenMimeDbContent && lpszType && lpszBuffer) |
| { |
| size_t dwStrLen = strlen(lpszType); |
| |
| if (dwStrLen < dwLen - dwLenMimeDbContent) |
| { |
| memcpy(lpszBuffer, szMimeDbContentA, dwLenMimeDbContent); |
| memcpy(lpszBuffer + dwLenMimeDbContent, lpszType, dwStrLen + 1); |
| return TRUE; |
| } |
| } |
| return FALSE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.329] |
| * |
| * Unicode version of GetMIMETypeSubKeyA. |
| */ |
| BOOL WINAPI GetMIMETypeSubKeyW(LPCWSTR lpszType, LPWSTR lpszBuffer, DWORD dwLen) |
| { |
| TRACE("(%s,%p,%d)\n", debugstr_w(lpszType), lpszBuffer, dwLen); |
| |
| if (dwLen > dwLenMimeDbContent && lpszType && lpszBuffer) |
| { |
| DWORD dwStrLen = strlenW(lpszType); |
| |
| if (dwStrLen < dwLen - dwLenMimeDbContent) |
| { |
| memcpy(lpszBuffer, szMimeDbContentW, dwLenMimeDbContent * sizeof(WCHAR)); |
| memcpy(lpszBuffer + dwLenMimeDbContent, lpszType, (dwStrLen + 1) * sizeof(WCHAR)); |
| return TRUE; |
| } |
| } |
| return FALSE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.330] |
| * |
| * Get the file extension for a given Mime type. |
| * |
| * PARAMS |
| * lpszType [I] Mime type to get the file extension for |
| * lpExt [O] Destination for the resulting extension |
| * iLen [I] Length of lpExt in characters |
| * |
| * RETURNS |
| * Success: TRUE. lpExt contains the file extension. |
| * Failure: FALSE, if any parameter is invalid or the extension cannot be |
| * retrieved. If iLen > 0, lpExt is set to an empty string. |
| * |
| * NOTES |
| * - The extension returned in lpExt always has a leading '.' character, even |
| * if the registry Mime database entry does not. |
| * - iLen must be long enough for the file extension for this function to succeed. |
| */ |
| BOOL WINAPI MIME_GetExtensionA(LPCSTR lpszType, LPSTR lpExt, INT iLen) |
| { |
| char szSubKey[MAX_PATH]; |
| DWORD dwlen = iLen - 1, dwType; |
| BOOL bRet = FALSE; |
| |
| if (iLen > 0 && lpExt) |
| *lpExt = '\0'; |
| |
| if (lpszType && lpExt && iLen > 2 && |
| GetMIMETypeSubKeyA(lpszType, szSubKey, MAX_PATH) && |
| !SHGetValueA(HKEY_CLASSES_ROOT, szSubKey, szExtensionA, &dwType, lpExt + 1, &dwlen) && |
| lpExt[1]) |
| { |
| if (lpExt[1] == '.') |
| memmove(lpExt, lpExt + 1, strlen(lpExt + 1) + 1); |
| else |
| *lpExt = '.'; /* Supply a '.' */ |
| bRet = TRUE; |
| } |
| return bRet; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.331] |
| * |
| * Unicode version of MIME_GetExtensionA. |
| */ |
| BOOL WINAPI MIME_GetExtensionW(LPCWSTR lpszType, LPWSTR lpExt, INT iLen) |
| { |
| WCHAR szSubKey[MAX_PATH]; |
| DWORD dwlen = iLen - 1, dwType; |
| BOOL bRet = FALSE; |
| |
| if (iLen > 0 && lpExt) |
| *lpExt = '\0'; |
| |
| if (lpszType && lpExt && iLen > 2 && |
| GetMIMETypeSubKeyW(lpszType, szSubKey, MAX_PATH) && |
| !SHGetValueW(HKEY_CLASSES_ROOT, szSubKey, szExtensionW, &dwType, lpExt + 1, &dwlen) && |
| lpExt[1]) |
| { |
| if (lpExt[1] == '.') |
| memmove(lpExt, lpExt + 1, (strlenW(lpExt + 1) + 1) * sizeof(WCHAR)); |
| else |
| *lpExt = '.'; /* Supply a '.' */ |
| bRet = TRUE; |
| } |
| return bRet; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.324] |
| * |
| * Set the file extension for a MIME content key. |
| * |
| * PARAMS |
| * lpszExt [I] File extension to set |
| * lpszType [I] Content type to set the extension for |
| * |
| * RETURNS |
| * Success: TRUE. The file extension is set in the registry. |
| * Failure: FALSE. |
| */ |
| BOOL WINAPI RegisterExtensionForMIMETypeA(LPCSTR lpszExt, LPCSTR lpszType) |
| { |
| DWORD dwLen; |
| char szKey[MAX_PATH]; |
| |
| TRACE("(%s,%s)\n", debugstr_a(lpszExt), debugstr_a(lpszType)); |
| |
| if (!GetMIMETypeSubKeyA(lpszType, szKey, MAX_PATH)) /* Get full path to the key */ |
| return FALSE; |
| |
| dwLen = strlen(lpszExt) + 1; |
| |
| if (SHSetValueA(HKEY_CLASSES_ROOT, szKey, szExtensionA, REG_SZ, lpszExt, dwLen)) |
| return FALSE; |
| return TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.325] |
| * |
| * Unicode version of RegisterExtensionForMIMETypeA. |
| */ |
| BOOL WINAPI RegisterExtensionForMIMETypeW(LPCWSTR lpszExt, LPCWSTR lpszType) |
| { |
| DWORD dwLen; |
| WCHAR szKey[MAX_PATH]; |
| |
| TRACE("(%s,%s)\n", debugstr_w(lpszExt), debugstr_w(lpszType)); |
| |
| /* Get the full path to the key */ |
| if (!GetMIMETypeSubKeyW(lpszType, szKey, MAX_PATH)) /* Get full path to the key */ |
| return FALSE; |
| |
| dwLen = (lstrlenW(lpszExt) + 1) * sizeof(WCHAR); |
| |
| if (SHSetValueW(HKEY_CLASSES_ROOT, szKey, szExtensionW, REG_SZ, lpszExt, dwLen)) |
| return FALSE; |
| return TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.326] |
| * |
| * Delete a file extension from a MIME content type. |
| * |
| * PARAMS |
| * lpszType [I] Content type to delete the extension for |
| * |
| * RETURNS |
| * Success: TRUE. The file extension is deleted from the registry. |
| * Failure: FALSE. The extension may have been removed but the key remains. |
| * |
| * NOTES |
| * If deleting the extension leaves an orphan key, the key is removed also. |
| */ |
| BOOL WINAPI UnregisterExtensionForMIMETypeA(LPCSTR lpszType) |
| { |
| char szKey[MAX_PATH]; |
| |
| TRACE("(%s)\n", debugstr_a(lpszType)); |
| |
| if (!GetMIMETypeSubKeyA(lpszType, szKey, MAX_PATH)) /* Get full path to the key */ |
| return FALSE; |
| |
| if (!SHDeleteValueA(HKEY_CLASSES_ROOT, szKey, szExtensionA)) |
| return FALSE; |
| |
| if (!SHDeleteOrphanKeyA(HKEY_CLASSES_ROOT, szKey)) |
| return FALSE; |
| return TRUE; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.327] |
| * |
| * Unicode version of UnregisterExtensionForMIMETypeA. |
| */ |
| BOOL WINAPI UnregisterExtensionForMIMETypeW(LPCWSTR lpszType) |
| { |
| WCHAR szKey[MAX_PATH]; |
| |
| TRACE("(%s)\n", debugstr_w(lpszType)); |
| |
| if (!GetMIMETypeSubKeyW(lpszType, szKey, MAX_PATH)) /* Get full path to the key */ |
| return FALSE; |
| |
| if (!SHDeleteValueW(HKEY_CLASSES_ROOT, szKey, szExtensionW)) |
| return FALSE; |
| |
| if (!SHDeleteOrphanKeyW(HKEY_CLASSES_ROOT, szKey)) |
| return FALSE; |
| return TRUE; |
| } |
| |
| /************************************************************************* |
| * SHRegDuplicateHKey [SHLWAPI.@] |
| * |
| * Create a duplicate of a registry handle. |
| * |
| * PARAMS |
| * hKey [I] key to duplicate. |
| * |
| * RETURNS |
| * A new handle pointing to the same key as hKey. |
| */ |
| HKEY WINAPI SHRegDuplicateHKey(HKEY hKey) |
| { |
| HKEY newKey = 0; |
| |
| RegOpenKeyExA(hKey, 0, 0, MAXIMUM_ALLOWED, &newKey); |
| TRACE("new key is %p\n", newKey); |
| return newKey; |
| } |
| |
| |
| /************************************************************************* |
| * SHCopyKeyA [SHLWAPI.@] |
| * |
| * Copy a key and its values/sub keys to another location. |
| * |
| * PARAMS |
| * hKeySrc [I] Source key to copy from |
| * lpszSrcSubKey [I] Sub key under hKeySrc, or NULL to use hKeySrc directly |
| * hKeyDst [I] Destination key |
| * dwReserved [I] Reserved, must be 0 |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. The key is copied to the destination key. |
| * Failure: A standard windows error code. |
| * |
| * NOTES |
| * If hKeyDst is a key under hKeySrc, this function will misbehave |
| * (It will loop until out of stack, or the registry is full). This |
| * bug is present in Win32 also. |
| */ |
| DWORD WINAPI SHCopyKeyA(HKEY hKeySrc, LPCSTR lpszSrcSubKey, HKEY hKeyDst, DWORD dwReserved) |
| { |
| WCHAR szSubKeyW[MAX_PATH]; |
| |
| TRACE("(hkey=%p,%s,%p08x,%d)\n", hKeySrc, debugstr_a(lpszSrcSubKey), hKeyDst, dwReserved); |
| |
| if (lpszSrcSubKey) |
| MultiByteToWideChar(0, 0, lpszSrcSubKey, -1, szSubKeyW, MAX_PATH); |
| |
| return SHCopyKeyW(hKeySrc, lpszSrcSubKey ? szSubKeyW : NULL, hKeyDst, dwReserved); |
| } |
| |
| /************************************************************************* |
| * SHCopyKeyW [SHLWAPI.@] |
| * |
| * See SHCopyKeyA. |
| */ |
| DWORD WINAPI SHCopyKeyW(HKEY hKeySrc, LPCWSTR lpszSrcSubKey, HKEY hKeyDst, DWORD dwReserved) |
| { |
| DWORD dwKeyCount = 0, dwValueCount = 0, dwMaxKeyLen = 0; |
| DWORD dwMaxValueLen = 0, dwMaxDataLen = 0, i; |
| BYTE buff[1024]; |
| LPVOID lpBuff = (LPVOID)buff; |
| WCHAR szName[MAX_PATH], *lpszName = szName; |
| DWORD dwRet = S_OK; |
| |
| TRACE("hkey=%p,%s,%p08x,%d)\n", hKeySrc, debugstr_w(lpszSrcSubKey), hKeyDst, dwReserved); |
| |
| if(!hKeyDst || !hKeySrc) |
| dwRet = ERROR_INVALID_PARAMETER; |
| else |
| { |
| /* Open source key */ |
| if(lpszSrcSubKey) |
| dwRet = RegOpenKeyExW(hKeySrc, lpszSrcSubKey, 0, KEY_ALL_ACCESS, &hKeySrc); |
| |
| if(dwRet) |
| hKeyDst = NULL; /* Don't close this key since we didn't open it */ |
| else |
| { |
| /* Get details about sub keys and values */ |
| dwRet = RegQueryInfoKeyW(hKeySrc, NULL, NULL, NULL, &dwKeyCount, &dwMaxKeyLen, |
| NULL, &dwValueCount, &dwMaxValueLen, &dwMaxDataLen, |
| NULL, NULL); |
| if(!dwRet) |
| { |
| if (dwMaxValueLen > dwMaxKeyLen) |
| dwMaxKeyLen = dwMaxValueLen; /* Get max size for key/value names */ |
| |
| if (dwMaxKeyLen++ > MAX_PATH - 1) |
| lpszName = HeapAlloc(GetProcessHeap(), 0, dwMaxKeyLen * sizeof(WCHAR)); |
| |
| if (dwMaxDataLen > sizeof(buff)) |
| lpBuff = HeapAlloc(GetProcessHeap(), 0, dwMaxDataLen); |
| |
| if (!lpszName || !lpBuff) |
| dwRet = ERROR_NOT_ENOUGH_MEMORY; |
| } |
| } |
| } |
| |
| /* Copy all the sub keys */ |
| for(i = 0; i < dwKeyCount && !dwRet; i++) |
| { |
| HKEY hSubKeySrc, hSubKeyDst; |
| DWORD dwSize = dwMaxKeyLen; |
| |
| dwRet = RegEnumKeyExW(hKeySrc, i, lpszName, &dwSize, NULL, NULL, NULL, NULL); |
| |
| if(!dwRet) |
| { |
| /* Open source sub key */ |
| dwRet = RegOpenKeyExW(hKeySrc, lpszName, 0, KEY_READ, &hSubKeySrc); |
| |
| if(!dwRet) |
| { |
| /* Create destination sub key */ |
| dwRet = RegCreateKeyW(hKeyDst, lpszName, &hSubKeyDst); |
| |
| if(!dwRet) |
| { |
| /* Recursively copy keys and values from the sub key */ |
| dwRet = SHCopyKeyW(hSubKeySrc, NULL, hSubKeyDst, 0); |
| RegCloseKey(hSubKeyDst); |
| } |
| } |
| RegCloseKey(hSubKeySrc); |
| } |
| } |
| |
| /* Copy all the values in this key */ |
| for (i = 0; i < dwValueCount && !dwRet; i++) |
| { |
| DWORD dwNameSize = dwMaxKeyLen, dwType, dwLen = dwMaxDataLen; |
| |
| dwRet = RegEnumValueW(hKeySrc, i, lpszName, &dwNameSize, NULL, &dwType, lpBuff, &dwLen); |
| |
| if (!dwRet) |
| dwRet = SHSetValueW(hKeyDst, NULL, lpszName, dwType, lpBuff, dwLen); |
| } |
| |
| /* Free buffers if allocated */ |
| if (lpszName != szName) |
| HeapFree(GetProcessHeap(), 0, lpszName); |
| if (lpBuff != buff) |
| HeapFree(GetProcessHeap(), 0, lpBuff); |
| |
| if (lpszSrcSubKey && hKeyDst) |
| RegCloseKey(hKeyDst); |
| return dwRet; |
| } |
| |
| /* |
| * The following functions are ORDINAL ONLY: |
| */ |
| |
| /************************************************************************* |
| * @ [SHLWAPI.280] |
| * |
| * Read an integer value from the registry, falling back to a default. |
| * |
| * PARAMS |
| * hKey [I] Registry key to read from |
| * lpszValue [I] Value name to read |
| * iDefault [I] Default value to return |
| * |
| * RETURNS |
| * The value contained in the given registry value if present, otherwise |
| * iDefault. |
| */ |
| int WINAPI SHRegGetIntW(HKEY hKey, LPCWSTR lpszValue, int iDefault) |
| { |
| TRACE("(%p,%s,%d)\n", hKey, debugstr_w(lpszValue), iDefault); |
| |
| if (hKey) |
| { |
| WCHAR szBuff[32]; |
| DWORD dwSize = sizeof(szBuff); |
| szBuff[0] = '\0'; |
| SHQueryValueExW(hKey, lpszValue, 0, 0, szBuff, &dwSize); |
| |
| if(*szBuff >= '0' && *szBuff <= '9') |
| return StrToIntW(szBuff); |
| } |
| return iDefault; |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.343] |
| * |
| * Create or open an explorer ClassId Key. |
| * |
| * PARAMS |
| * guid [I] Explorer ClassId key to open |
| * lpszValue [I] Value name under the ClassId Key |
| * bUseHKCU [I] TRUE=Use HKEY_CURRENT_USER, FALSE=Use HKEY_CLASSES_ROOT |
| * bCreate [I] TRUE=Create the key if it doesn't exist, FALSE=Don't |
| * phKey [O] Destination for the resulting key handle |
| * |
| * RETURNS |
| * Success: S_OK. phKey contains the resulting registry handle. |
| * Failure: An HRESULT error code indicating the problem. |
| */ |
| HRESULT WINAPI SHRegGetCLSIDKeyA(REFGUID guid, LPCSTR lpszValue, BOOL bUseHKCU, BOOL bCreate, PHKEY phKey) |
| { |
| WCHAR szValue[MAX_PATH]; |
| |
| if (lpszValue) |
| MultiByteToWideChar(CP_ACP, 0, lpszValue, -1, szValue, sizeof(szValue)/sizeof(WCHAR)); |
| |
| return SHRegGetCLSIDKeyW(guid, lpszValue ? szValue : NULL, bUseHKCU, bCreate, phKey); |
| } |
| |
| /************************************************************************* |
| * @ [SHLWAPI.344] |
| * |
| * Unicode version of SHRegGetCLSIDKeyA. |
| */ |
| HRESULT WINAPI SHRegGetCLSIDKeyW(REFGUID guid, LPCWSTR lpszValue, BOOL bUseHKCU, |
| BOOL bCreate, PHKEY phKey) |
| { |
| static const WCHAR szClassIdKey[] = { 'S','o','f','t','w','a','r','e','\\', |
| 'M','i','c','r','o','s','o','f','t','\\','W','i','n','d','o','w','s','\\', |
| 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', |
| 'E','x','p','l','o','r','e','r','\\','C','L','S','I','D','\\' }; |
| #define szClassIdKeyLen (sizeof(szClassIdKey)/sizeof(WCHAR)) |
| WCHAR szKey[MAX_PATH]; |
| DWORD dwRet; |
| HKEY hkey; |
| |
| /* Create the key string */ |
| memcpy(szKey, szClassIdKey, sizeof(szClassIdKey)); |
| SHStringFromGUIDW(guid, szKey + szClassIdKeyLen, 39); /* Append guid */ |
| |
| if(lpszValue) |
| { |
| szKey[szClassIdKeyLen + 39] = '\\'; |
| strcpyW(szKey + szClassIdKeyLen + 40, lpszValue); /* Append value name */ |
| } |
| |
| hkey = bUseHKCU ? HKEY_CURRENT_USER : HKEY_CLASSES_ROOT; |
| |
| if(bCreate) |
| dwRet = RegCreateKeyW(hkey, szKey, phKey); |
| else |
| dwRet = RegOpenKeyExW(hkey, szKey, 0, KEY_READ, phKey); |
| |
| return dwRet ? HRESULT_FROM_WIN32(dwRet) : S_OK; |
| } |
| |
| /************************************************************************* |
| * SHRegisterValidateTemplate [SHLWAPI.@] |
| * |
| * observed from the ie 5.5 installer: |
| * - allocates a buffer with the size of the given file |
| * - read the file content into the buffer |
| * - creates the key szTemplateKey |
| * - sets "205523652929647911071668590831910975402"=dword:00002e37 at |
| * the key |
| * |
| * PARAMS |
| * filename [I] An existing file its content is read into an allocated |
| * buffer |
| * unknown [I] |
| * |
| * RETURNS |
| * Success: ERROR_SUCCESS. |
| */ |
| HRESULT WINAPI SHRegisterValidateTemplate(LPCWSTR filename, BOOL unknown) |
| { |
| /* static const WCHAR szTemplateKey[] = { 'S','o','f','t','w','a','r','e','\\', |
| * 'M','i','c','r','o','s','o','f','t','\\','W','i','n','d','o','w','s','\\', |
| * 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\', |
| * 'E','x','p','l','o','r','e','r','\\', |
| * 'T','e','m','p','l','a','t','e','R','e','g','i','s','t','r','y',0 }; |
| */ |
| FIXME("stub: %s, %08x\n", debugstr_w(filename), unknown); |
| |
| return S_OK; |
| } |