X-Git-Url: http://git.osdn.net/view?a=blobdiff_plain;f=winsup%2Fcygwin%2Fregistry.cc;fp=winsup%2Fcygwin%2Fregistry.cc;h=2923cbe86c82a9e04558d4292b42af34f353266f;hb=cbad6d5ba66a94967eaf14df64463a4abca700d6;hp=523218828d91d9ff0fd9f313db2ce91fbb647f4e;hpb=b110964dfb2d0779466e4e604983d3afa83993a4;p=pf3gnuchains%2Fpf3gnuchains4x.git diff --git a/winsup/cygwin/registry.cc b/winsup/cygwin/registry.cc index 523218828d..2923cbe86c 100644 --- a/winsup/cygwin/registry.cc +++ b/winsup/cygwin/registry.cc @@ -1,7 +1,7 @@ /* registry.cc: registry interface Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, - 2005, 2006, 2007, 2008, 2009 Red Hat, Inc. + 2005, 2006, 2007, 2008, 2009, 2010, 2011 Red Hat, Inc. This file is part of Cygwin. @@ -17,297 +17,268 @@ details. */ #include "dtable.h" #include "cygheap.h" #include "tls_pbuf.h" +#include "ntdll.h" #include - -reg_key::reg_key (HKEY top, REGSAM access, ...): _disposition (0) -{ - va_list av; - va_start (av, access); - build_reg (top, access, av); - va_end (av); -} +#include /* Opens a key under the appropriate Cygwin key. Do not use HKCU per MS KB 199190 */ - -reg_key::reg_key (bool isHKLM, REGSAM access, ...): _disposition (0) +static NTSTATUS +top_key (bool isHKLM, REGSAM access, PHANDLE top) { - va_list av; - HKEY top; + WCHAR rbuf[PATH_MAX], *p; + UNICODE_STRING rpath; + OBJECT_ATTRIBUTES attr; + NTSTATUS status; + InitializeObjectAttributes (&attr, &rpath, OBJ_CASE_INSENSITIVE, NULL, NULL); if (isHKLM) - top = HKEY_LOCAL_MACHINE; + { + wcpcpy (rbuf, L"\\Registry\\Machine"); + RtlInitUnicodeString (&rpath, rbuf); + status = NtOpenKey (top, access, &attr); + } else { - char name[128]; - const char *names[2] = {cygheap->user.get_windows_id (name), ".DEFAULT"}; + WCHAR name[128]; + PCWSTR names[2] = {cygheap->user.get_windows_id (name), + L".DEFAULT"}; + + p = wcpcpy (rbuf, L"\\Registry\\User\\"); for (int i = 0; i < 2; i++) { - key_is_invalid = RegOpenKeyEx (HKEY_USERS, names[i], 0, access, &top); - if (key_is_invalid == ERROR_SUCCESS) - goto OK; - debug_printf ("HKU\\%s failed, Win32 error %ld", names[i], key_is_invalid); + wcpcpy (p, names[i]); + RtlInitUnicodeString (&rpath, rbuf); + status = NtOpenKey (top, access, &attr); + if (NT_SUCCESS (status)) + break; } - return; } -OK: - new (this) reg_key (top, access, "SOFTWARE", - CYGWIN_INFO_CYGWIN_REGISTRY_NAME, NULL); - if (top != HKEY_LOCAL_MACHINE) - RegCloseKey (top); - if (key_is_invalid) - return; + return status; +} - top = key; +reg_key::reg_key (HKEY top, REGSAM access, ...): _disposition (0) +{ + va_list av; va_start (av, access); build_reg (top, access, av); va_end (av); - if (top != key) - RegCloseKey (top); +} + +reg_key::reg_key (bool isHKLM, REGSAM access, ...): _disposition (0) +{ + va_list av; + HANDLE top; + + key_is_invalid = top_key (isHKLM, access, &top); + if (NT_SUCCESS (key_is_invalid)) + { + new (this) reg_key ((HKEY) top, access, L"SOFTWARE", + _WIDE (CYGWIN_INFO_CYGWIN_REGISTRY_NAME), NULL); + NtClose (top); + if (key_is_invalid) + return; + top = key; + va_start (av, access); + build_reg ((HKEY) top, access, av); + va_end (av); + if (top != key) + NtClose (top); + } } void reg_key::build_reg (HKEY top, REGSAM access, va_list av) { - char *name; - HKEY r = top; + PWCHAR name; + HANDLE r; + UNICODE_STRING uname; + OBJECT_ATTRIBUTES attr; + NTSTATUS status; + + if (top != HKEY_LOCAL_MACHINE && top != HKEY_CURRENT_USER) + r = (HANDLE) top; + else if (!NT_SUCCESS (top_key (top == HKEY_LOCAL_MACHINE, access, &r))) + return; key_is_invalid = 0; - - /* FIXME: Most of the time a valid mount area should exist. Perhaps - we should just try an open of the correct key first and only resort - to this method in the unlikely situation that it's the first time - the current mount areas are being used. */ - - while ((name = va_arg (av, char *)) != NULL) + while ((name = va_arg (av, PWCHAR)) != NULL) { - int res = RegCreateKeyExA (r, - name, - 0, - NULL, - REG_OPTION_NON_VOLATILE, - access, - &sec_none_nih, - &key, - &_disposition); - if (r != top) - RegCloseKey (r); + RtlInitUnicodeString (&uname, name); + InitializeObjectAttributes (&attr, &uname, + OBJ_CASE_INSENSITIVE | OBJ_OPENIF, r, NULL); + + status = NtCreateKey (&key, access, &attr, 0, NULL, + REG_OPTION_NON_VOLATILE, &_disposition); + if (r != (HANDLE) top) + NtClose (r); r = key; - if (res != ERROR_SUCCESS) + if (!NT_SUCCESS (status)) { - key_is_invalid = res; - debug_printf ("failed to create key %s in the registry", name); + key_is_invalid = status; + debug_printf ("failed to create key %S in the registry", uname); break; } } } -/* Given the current registry key, return the specific int value +/* Given the current registry key, return the specific DWORD value requested. Return def on failure. */ -int -reg_key::get_int (const char *name, int def) -{ - DWORD type; - DWORD dst; - DWORD size = sizeof (dst); - - if (key_is_invalid) - return def; - - LONG res = RegQueryValueExA (key, name, 0, &type, (LPBYTE) &dst, &size); - - if (type != REG_DWORD || res != ERROR_SUCCESS) - return def; - - return dst; -} - -int -reg_key::get_int (const PWCHAR name, int def) +DWORD +reg_key::get_dword (PCWSTR name, DWORD def) { - DWORD type; - DWORD dst; - DWORD size = sizeof (dst); - if (key_is_invalid) return def; - LONG res = RegQueryValueExW (key, name, 0, &type, (LPBYTE) &dst, &size); - - if (type != REG_DWORD || res != ERROR_SUCCESS) + NTSTATUS status; + UNICODE_STRING uname; + ULONG size = sizeof (KEY_VALUE_PARTIAL_INFORMATION) + sizeof (DWORD); + ULONG rsize; + PKEY_VALUE_PARTIAL_INFORMATION vbuf = (PKEY_VALUE_PARTIAL_INFORMATION) + alloca (size); + + RtlInitUnicodeString (&uname, name); + status = NtQueryValueKey (key, &uname, KeyValuePartialInformation, vbuf, + size, &rsize); + if (status != STATUS_SUCCESS || vbuf->Type != REG_DWORD) return def; - - return dst; + DWORD *dst = (DWORD *) vbuf->Data; + return *dst; } -/* Given the current registry key, set a specific int value. */ +/* Given the current registry key, set a specific DWORD value. */ -int -reg_key::set_int (const char *name, int val) +NTSTATUS +reg_key::set_dword (PCWSTR name, DWORD val) { - DWORD value = val; if (key_is_invalid) return key_is_invalid; - return (int) RegSetValueExA (key, name, 0, REG_DWORD, - (const BYTE *) &value, sizeof (value)); -} - -int -reg_key::set_int (const PWCHAR name, int val) -{ - DWORD value = val; - if (key_is_invalid) - return key_is_invalid; - - return (int) RegSetValueExW (key, name, 0, REG_DWORD, - (const BYTE *) &value, sizeof (value)); + DWORD value = (DWORD) val; + UNICODE_STRING uname; + RtlInitUnicodeString (&uname, name); + return NtSetValueKey (key, &uname, 0, REG_DWORD, &value, sizeof (value)); } /* Given the current registry key, return the specific string value requested. Return zero on success, non-zero on failure. */ -int -reg_key::get_string (const char *name, char *dst, size_t max, const char *def) -{ - DWORD size = max; - DWORD type; - LONG res; - - if (key_is_invalid) - res = key_is_invalid; - else - res = RegQueryValueExA (key, name, 0, &type, (LPBYTE) dst, &size); - - if ((def != 0) && ((type != REG_SZ) || (res != ERROR_SUCCESS))) - strcpy (dst, def); - return (int) res; -} - -int -reg_key::get_string (const PWCHAR name, PWCHAR dst, size_t max, const PWCHAR def) +NTSTATUS +reg_key::get_string (PCWSTR name, PWCHAR dst, size_t max, PCWSTR def) { - DWORD size = max; - DWORD type; - LONG res; + NTSTATUS status; if (key_is_invalid) - res = key_is_invalid; + { + status = key_is_invalid; + if (def != NULL) + wcpncpy (dst, def, max); + } else - res = RegQueryValueExW (key, name, 0, &type, (LPBYTE) dst, &size); - - if ((def != 0) && ((type != REG_SZ) || (res != ERROR_SUCCESS))) - wcscpy (dst, def); - return (int) res; + { + UNICODE_STRING uname; + ULONG size = sizeof (KEY_VALUE_PARTIAL_INFORMATION) + max * sizeof (WCHAR); + ULONG rsize; + PKEY_VALUE_PARTIAL_INFORMATION vbuf = (PKEY_VALUE_PARTIAL_INFORMATION) + alloca (size); + + RtlInitUnicodeString (&uname, name); + status = NtQueryValueKey (key, &uname, KeyValuePartialInformation, vbuf, + size, &rsize); + if (status != STATUS_SUCCESS || vbuf->Type != REG_SZ) + wcpncpy (dst, def, max); + else + wcpncpy (dst, (PWCHAR) vbuf->Data, max); + } + return status; } /* Given the current registry key, set a specific string value. */ -int -reg_key::set_string (const char *name, const char *src) -{ - if (key_is_invalid) - return key_is_invalid; - return (int) RegSetValueExA (key, name, 0, REG_SZ, (const BYTE*) src, - strlen (src) + 1); -} - -int -reg_key::set_string (const PWCHAR name, const PWCHAR src) -{ - if (key_is_invalid) - return key_is_invalid; - return (int) RegSetValueExW (key, name, 0, REG_SZ, (const BYTE*) src, - (wcslen (src) + 1) * sizeof (WCHAR)); -} - -/* Return the handle to key. */ - -HKEY -reg_key::get_key () -{ - return key; -} - -/* Delete subkey of current key. Returns the error code from the - RegDeleteKeyA invocation. */ - -int -reg_key::kill (const char *name) +NTSTATUS +reg_key::set_string (PCWSTR name, PCWSTR src) { if (key_is_invalid) return key_is_invalid; - return RegDeleteKeyA (key, name); -} - -/* Delete the value specified by name of current key. Returns the error code - from the RegDeleteValueA invocation. */ -int -reg_key::killvalue (const char *name) -{ - if (key_is_invalid) - return key_is_invalid; - return RegDeleteValueA (key, name); + UNICODE_STRING uname; + RtlInitUnicodeString (&uname, name); + return NtSetValueKey (key, &uname, 0, REG_SZ, (PVOID) src, + (wcslen (src) + 1) * sizeof (WCHAR)); } reg_key::~reg_key () { if (!key_is_invalid) - RegCloseKey (key); + NtClose (key); key_is_invalid = 1; } PWCHAR -get_registry_hive_path (const PWCHAR name, PWCHAR path) +get_registry_hive_path (PCWSTR name, PWCHAR path) { - WCHAR key[256], *kend; - HKEY hkey; - if (!name || !path) return NULL; - kend = wcpcpy (key, L"SOFTWARE\\Microsoft\\WindowsNT\\CurrentVersion\\ProfileList\\"); - wcpcpy (kend, name); - if (!RegOpenKeyExW (HKEY_LOCAL_MACHINE, key, 0, KEY_READ, &hkey)) + + WCHAR key[256]; + UNICODE_STRING buf; + tmp_pathbuf tp; + tp.u_get (&buf); + NTSTATUS status; + + RTL_QUERY_REGISTRY_TABLE tab[2] = { + { NULL, RTL_QUERY_REGISTRY_NOEXPAND | RTL_QUERY_REGISTRY_DIRECT + | RTL_QUERY_REGISTRY_REQUIRED, + L"ProfileImagePath", &buf, REG_NONE, NULL, 0 }, + { NULL, 0, NULL, NULL, 0, NULL, 0 } + }; + wcpcpy (wcpcpy (key, L"ProfileList\\"), name); + status = RtlQueryRegistryValues (RTL_REGISTRY_WINDOWS_NT, key, tab, + NULL, NULL); + if (!NT_SUCCESS (status) || buf.Length == 0) { - tmp_pathbuf tp; - PWCHAR buf = tp.w_get (); - DWORD type, siz; - - path[0] = L'\0'; - if (!RegQueryValueExW (hkey, L"ProfileImagePath", 0, &type, - (BYTE *)buf, (siz = NT_MAX_PATH, &siz))) - ExpandEnvironmentStringsW (buf, path, NT_MAX_PATH); - RegCloseKey (hkey); - if (path[0]) - return path; + debug_printf ("ProfileImagePath for %W not found, status %p", name, + status); + return NULL; } - debug_printf ("HKLM\\%W not found", key); - return NULL; + wcpcpy (path, L"\\??\\"); + ExpandEnvironmentStringsW (buf.Buffer, path + 4, NT_MAX_PATH - 4); + debug_printf ("ProfileImagePath for %W: %W", name, path); + return path; } void -load_registry_hive (const PWCHAR name) +load_registry_hive (PCWSTR name) { + if (!name) + return; + + /* Fetch the path. */ tmp_pathbuf tp; PWCHAR path = tp.w_get (); - HKEY hkey; - LONG ret; - - if (!name) + if (!get_registry_hive_path (name, path)) return; - /* Check if user hive is already loaded. */ - if (!RegOpenKeyExW (HKEY_USERS, name, 0, KEY_READ, &hkey)) - { - debug_printf ("User registry hive for %W already exists", name); - RegCloseKey (hkey); - return; - } - if (get_registry_hive_path (name, path)) - { - wcscat (path, L"\\NTUSER.DAT"); - if ((ret = RegLoadKeyW (HKEY_USERS, name, path)) != ERROR_SUCCESS) - debug_printf ("Loading user registry hive for %W failed: %d", name, ret); - } -} + WCHAR key[256]; + UNICODE_STRING ukey, upath; + OBJECT_ATTRIBUTES key_attr, path_attr; + NTSTATUS status; + + /* Create the object attributes for key and path. */ + wcpcpy (wcpcpy (key, L"\\Registry\\User\\"), name); + RtlInitUnicodeString (&ukey, key); + InitializeObjectAttributes (&key_attr, &ukey, OBJ_CASE_INSENSITIVE, + NULL, NULL); + wcscat (path, L"\\NTUSER.DAT"); + RtlInitUnicodeString (&upath, path); + InitializeObjectAttributes (&path_attr, &upath, OBJ_CASE_INSENSITIVE, + NULL, NULL); + /* Load file into key. */ + status = NtLoadKey (&key_attr, &path_attr); + if (!NT_SUCCESS (status)) + debug_printf ("Loading user registry hive %S into %S failed: %p", + &upath, &ukey, status); + else + debug_printf ("Loading user registry hive %S into %S SUCCEEDED: %p", + &upath, &ukey, status); +}