2008-03-15 10:38:45 +01:00
|
|
|
/*
|
|
|
|
* 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>
|
2012-01-23 12:16:36 +01:00
|
|
|
#include <winsvc.h>
|
2008-03-15 10:38:45 +01:00
|
|
|
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
#include "services.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(service);
|
|
|
|
|
|
|
|
LPWSTR strdupW(LPCWSTR str)
|
|
|
|
{
|
2008-03-15 16:42:38 +01:00
|
|
|
int len;
|
|
|
|
WCHAR *buf;
|
|
|
|
|
|
|
|
if (str == NULL)
|
|
|
|
return NULL;
|
|
|
|
len = strlenW(str);
|
|
|
|
buf = HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR)*(len+1));
|
2008-03-15 10:38:45 +01:00
|
|
|
if (buf == NULL)
|
|
|
|
return NULL;
|
|
|
|
strcpyW(buf, str);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2008-03-15 16:42:38 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2008-03-15 10:38:45 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2008-09-17 00:35:01 +02:00
|
|
|
DWORD load_reg_multisz(HKEY hKey, LPCWSTR szValue, BOOL bAllowSingle, LPWSTR *output)
|
2008-03-15 10:38:45 +01:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
2008-09-17 00:35:01 +02:00
|
|
|
if (!((type == REG_MULTI_SZ) || ((type == REG_SZ) && bAllowSingle)))
|
2008-03-15 10:38:45 +01:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|