| /* |
| * Services.exe - some utility functions |
| * |
| * Copyright 2007 Google (Mikolaj Zalewski) |
| * |
| * 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 |
| */ |
| |
| #define WIN32_LEAN_AND_MEAN |
| |
| #include <stdarg.h> |
| #include <windows.h> |
| |
| #include "wine/unicode.h" |
| #include "wine/debug.h" |
| #include "services.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(service); |
| |
| LPWSTR strdupW(LPCWSTR str) |
| { |
| int len; |
| WCHAR *buf; |
| |
| if (str == NULL) |
| return NULL; |
| len = strlenW(str); |
| buf = HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR)*(len+1)); |
| if (buf == NULL) |
| return NULL; |
| strcpyW(buf, str); |
| return buf; |
| } |
| |
| BOOL check_multisz(LPCWSTR lpMultiSz, DWORD cbSize) |
| { |
| if (cbSize == 0 || (cbSize == sizeof(WCHAR) && lpMultiSz[0] == 0)) |
| return TRUE; |
| if ((cbSize % sizeof(WCHAR)) != 0 || cbSize < 2*sizeof(WCHAR)) |
| return FALSE; |
| if (lpMultiSz[cbSize/2 - 1] || lpMultiSz[cbSize/2 - 2]) |
| return FALSE; |
| return TRUE; |
| } |
| |
| DWORD load_reg_string(HKEY hKey, LPCWSTR szValue, BOOL bExpand, LPWSTR *output) |
| { |
| DWORD size, type; |
| LPWSTR buf = NULL; |
| DWORD err; |
| |
| *output = NULL; |
| if ((err = RegQueryValueExW(hKey, szValue, 0, &type, NULL, &size)) != 0) |
| { |
| if (err == ERROR_FILE_NOT_FOUND) |
| return ERROR_SUCCESS; |
| goto failed; |
| } |
| if (!(type == REG_SZ || (type == REG_EXPAND_SZ && bExpand))) |
| { |
| err = ERROR_INVALID_DATATYPE; |
| goto failed; |
| } |
| buf = HeapAlloc(GetProcessHeap(), 0, size + sizeof(WCHAR)); |
| if ((err = RegQueryValueExW(hKey, szValue, 0, &type, (LPBYTE)buf, &size)) != 0) |
| goto failed; |
| buf[size/sizeof(WCHAR)] = 0; |
| |
| if (type == REG_EXPAND_SZ) |
| { |
| LPWSTR str; |
| DWORD size = ExpandEnvironmentStringsW(buf, NULL, 0); |
| if (size == 0) |
| { |
| err = GetLastError(); |
| goto failed; |
| } |
| str = HeapAlloc(GetProcessHeap(), 0, size * sizeof(WCHAR)); |
| ExpandEnvironmentStringsW(buf, str, size); |
| HeapFree(GetProcessHeap(), 0, buf); |
| *output = str; |
| } |
| else |
| *output = buf; |
| return ERROR_SUCCESS; |
| |
| failed: |
| WINE_ERR("Error %d while reading value %s\n", err, wine_dbgstr_w(szValue)); |
| HeapFree(GetProcessHeap(), 0, buf); |
| return err; |
| } |
| |
| DWORD load_reg_multisz(HKEY hKey, LPCWSTR szValue, LPWSTR *output) |
| { |
| DWORD size, type; |
| LPWSTR buf = NULL; |
| DWORD err; |
| |
| *output = NULL; |
| if ((err = RegQueryValueExW(hKey, szValue, 0, &type, NULL, &size)) != 0) |
| { |
| if (err == ERROR_FILE_NOT_FOUND) |
| { |
| *output = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WCHAR)); |
| return ERROR_SUCCESS; |
| } |
| goto failed; |
| } |
| if (type != REG_MULTI_SZ) |
| { |
| err = ERROR_INVALID_DATATYPE; |
| goto failed; |
| } |
| buf = HeapAlloc(GetProcessHeap(), 0, size + 2*sizeof(WCHAR)); |
| if ((err = RegQueryValueExW(hKey, szValue, 0, &type, (LPBYTE)buf, &size)) != 0) |
| goto failed; |
| buf[size/sizeof(WCHAR)] = 0; |
| buf[size/sizeof(WCHAR) + 1] = 0; |
| *output = buf; |
| return ERROR_SUCCESS; |
| |
| failed: |
| WINE_ERR("Error %d while reading value %s\n", err, wine_dbgstr_w(szValue)); |
| HeapFree(GetProcessHeap(), 0, buf); |
| return err; |
| } |
| |
| DWORD load_reg_dword(HKEY hKey, LPCWSTR szValue, DWORD *output) |
| { |
| DWORD size, type; |
| DWORD err; |
| |
| *output = 0; |
| size = sizeof(DWORD); |
| if ((err = RegQueryValueExW(hKey, szValue, 0, &type, (LPBYTE)output, &size)) != 0) |
| { |
| if (err == ERROR_FILE_NOT_FOUND) |
| return ERROR_SUCCESS; |
| goto failed; |
| } |
| if (type != REG_DWORD || size != sizeof(DWORD)) |
| { |
| err = ERROR_INVALID_DATATYPE; |
| goto failed; |
| } |
| return ERROR_SUCCESS; |
| |
| failed: |
| WINE_ERR("Error %d while reading value %s\n", err, wine_dbgstr_w(szValue)); |
| return err; |
| } |