2008-08-06 18:36:30 +00:00
|
|
|
// This file is part of BOINC.
|
2006-01-23 08:47:05 +00:00
|
|
|
// http://boinc.berkeley.edu
|
2008-08-06 18:36:30 +00:00
|
|
|
// Copyright (C) 2008 University of California
|
2006-01-23 08:47:05 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC 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 3 of the License, or (at your option) any later version.
|
2006-01-23 08:47:05 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
2006-01-23 08:47:05 +00:00
|
|
|
// 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.
|
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
2006-01-23 08:47:05 +00:00
|
|
|
|
2010-05-11 19:10:29 +00:00
|
|
|
#if defined(_WIN32) && !defined(__STDWX_H__)
|
2006-12-14 18:15:22 +00:00
|
|
|
#include "boinc_win.h"
|
2010-05-11 19:10:29 +00:00
|
|
|
#elif defined(_WIN32) && defined(__STDWX_H__)
|
|
|
|
#include "stdwx.h"
|
2006-12-14 18:15:22 +00:00
|
|
|
#endif
|
2009-07-25 02:27:42 +00:00
|
|
|
|
2013-01-08 05:36:44 +00:00
|
|
|
#if defined(_MSC_VER) || defined(__MINGW32__)
|
|
|
|
#define snprintf _snprintf
|
|
|
|
#endif
|
2006-01-23 08:47:05 +00:00
|
|
|
|
2013-01-08 05:36:44 +00:00
|
|
|
#include "diagnostics.h"
|
2012-08-11 05:47:18 +00:00
|
|
|
#include "win_util.h"
|
|
|
|
|
2006-01-23 08:47:05 +00:00
|
|
|
/**
|
|
|
|
* Find out if we are on a Windows 2000 compatible system
|
|
|
|
**/
|
2006-12-14 16:59:28 +00:00
|
|
|
BOOL IsWindows2000Compatible() {
|
2011-01-07 20:23:22 +00:00
|
|
|
OSVERSIONINFO osvi;
|
|
|
|
ZeroMemory(&osvi, sizeof(OSVERSIONINFO));
|
|
|
|
osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
2006-01-23 08:47:05 +00:00
|
|
|
|
2011-01-07 20:23:22 +00:00
|
|
|
if (! GetVersionEx ( (OSVERSIONINFO *) &osvi) ) {
|
2006-01-23 08:47:05 +00:00
|
|
|
return FALSE;
|
2011-01-07 20:23:22 +00:00
|
|
|
}
|
2006-01-23 08:47:05 +00:00
|
|
|
|
|
|
|
return (osvi.dwMajorVersion >= 5);
|
|
|
|
}
|
|
|
|
|
2006-12-28 18:48:42 +00:00
|
|
|
/**
|
|
|
|
* Define these if they aren't defined. They are normally found in
|
|
|
|
* winnt.h, but some compilers don't have them.
|
|
|
|
**/
|
|
|
|
#ifndef VER_AND
|
|
|
|
#define VER_AND 6
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef VER_SUITENAME
|
|
|
|
#define VER_SUITENAME 0x0000040
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef VER_SUITE_SINGLEUSERTS
|
|
|
|
#define VER_SUITE_SINGLEUSERTS 0x00000100
|
|
|
|
#endif
|
|
|
|
|
2006-01-23 08:47:05 +00:00
|
|
|
/**
|
|
|
|
* This function performs the basic check to see if
|
|
|
|
* the platform on which it is running is Terminal
|
|
|
|
* services enabled. Note, this code is compatible on
|
|
|
|
* all Win32 platforms. For the Windows 2000 platform
|
|
|
|
* we perform a "lazy" bind to the new product suite
|
|
|
|
* APIs that were first introduced on that platform.
|
|
|
|
**/
|
2006-12-14 16:59:28 +00:00
|
|
|
BOOL IsTerminalServicesEnabled() {
|
2006-01-23 08:47:05 +00:00
|
|
|
BOOL bResult = FALSE; // assume Terminal Services is not enabled
|
|
|
|
|
|
|
|
DWORD dwVersion;
|
|
|
|
OSVERSIONINFOEXA osVersionInfo;
|
|
|
|
DWORDLONG dwlConditionMask = 0;
|
2009-07-25 03:51:18 +00:00
|
|
|
HMODULE hmodK32 = NULL;
|
|
|
|
HMODULE hmodNtDll = NULL;
|
|
|
|
typedef ULONGLONG (WINAPI *PFnVerSetConditionMask)(ULONGLONG,ULONG,UCHAR);
|
|
|
|
typedef BOOL (WINAPI *PFnVerifyVersionInfoA)(POSVERSIONINFOEXA, DWORD, DWORDLONG);
|
|
|
|
PFnVerSetConditionMask pfnVerSetConditionMask;
|
|
|
|
PFnVerifyVersionInfoA pfnVerifyVersionInfoA;
|
2006-01-23 08:47:05 +00:00
|
|
|
|
|
|
|
dwVersion = GetVersion();
|
|
|
|
|
|
|
|
// are we running NT ?
|
|
|
|
if (!(dwVersion & 0x80000000))
|
|
|
|
{
|
|
|
|
// Is it Windows 2000 (NT 5.0) or greater ?
|
|
|
|
if (LOBYTE(LOWORD(dwVersion)) > 4)
|
|
|
|
{
|
2009-07-25 03:51:18 +00:00
|
|
|
// In Windows 2000 we need to use the Product Suite APIs
|
|
|
|
// Don't static link because it won't load on non-Win2000 systems
|
|
|
|
hmodNtDll = GetModuleHandleA( "NTDLL.DLL" );
|
|
|
|
if (hmodNtDll != NULL)
|
|
|
|
{
|
|
|
|
pfnVerSetConditionMask = (PFnVerSetConditionMask )GetProcAddress( hmodNtDll, "VerSetConditionMask");
|
|
|
|
if (pfnVerSetConditionMask != NULL)
|
|
|
|
{
|
|
|
|
dwlConditionMask = (*pfnVerSetConditionMask)( dwlConditionMask, VER_SUITENAME, VER_AND );
|
|
|
|
hmodK32 = GetModuleHandleA( "KERNEL32.DLL" );
|
|
|
|
if (hmodK32 != NULL)
|
|
|
|
{
|
|
|
|
pfnVerifyVersionInfoA = (PFnVerifyVersionInfoA)GetProcAddress( hmodK32, "VerifyVersionInfoA") ;
|
|
|
|
if (pfnVerifyVersionInfoA != NULL)
|
|
|
|
{
|
|
|
|
ZeroMemory(&osVersionInfo, sizeof(osVersionInfo));
|
|
|
|
osVersionInfo.dwOSVersionInfoSize = sizeof(osVersionInfo);
|
|
|
|
osVersionInfo.wSuiteMask = VER_SUITE_TERMINAL | VER_SUITE_SINGLEUSERTS;
|
|
|
|
bResult = (*pfnVerifyVersionInfoA)(
|
|
|
|
&osVersionInfo,
|
|
|
|
VER_SUITENAME,
|
|
|
|
dwlConditionMask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// This is NT 4.0 or older
|
|
|
|
bResult = ValidateProductSuite( "Terminal Server" );
|
2006-01-23 08:47:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bResult;
|
|
|
|
}
|
|
|
|
|
2007-02-20 16:27:15 +00:00
|
|
|
|
2009-07-25 03:51:18 +00:00
|
|
|
/**
|
|
|
|
* This function compares the passed in "suite name" string
|
|
|
|
* to the product suite information stored in the registry.
|
|
|
|
* This only works on the Terminal Server 4.0 platform.
|
|
|
|
**/
|
|
|
|
BOOL ValidateProductSuite (LPSTR SuiteName) {
|
|
|
|
BOOL rVal = FALSE;
|
|
|
|
LONG Rslt;
|
|
|
|
HKEY hKey = NULL;
|
|
|
|
DWORD Type = 0;
|
|
|
|
DWORD Size = 0;
|
|
|
|
LPSTR ProductSuite = NULL;
|
|
|
|
LPSTR p;
|
|
|
|
|
|
|
|
Rslt = RegOpenKeyA(
|
|
|
|
HKEY_LOCAL_MACHINE,
|
|
|
|
"System\\CurrentControlSet\\Control\\ProductOptions",
|
|
|
|
&hKey
|
|
|
|
);
|
|
|
|
|
|
|
|
if (Rslt != ERROR_SUCCESS)
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
Rslt = RegQueryValueExA( hKey, "ProductSuite", NULL, &Type, NULL, &Size );
|
|
|
|
if (Rslt != ERROR_SUCCESS || !Size)
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
ProductSuite = (LPSTR) LocalAlloc( LPTR, Size );
|
|
|
|
if (!ProductSuite)
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
Rslt = RegQueryValueExA( hKey, "ProductSuite", NULL, &Type,
|
|
|
|
(LPBYTE) ProductSuite, &Size );
|
|
|
|
if (Rslt != ERROR_SUCCESS || Type != REG_MULTI_SZ)
|
|
|
|
goto exit;
|
|
|
|
|
|
|
|
p = ProductSuite;
|
|
|
|
while (*p)
|
|
|
|
{
|
|
|
|
if (lstrcmpA( p, SuiteName ) == 0)
|
|
|
|
{
|
|
|
|
rVal = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
p += (lstrlenA( p ) + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
exit:
|
|
|
|
if (ProductSuite)
|
|
|
|
LocalFree( ProductSuite );
|
|
|
|
|
|
|
|
if (hKey)
|
|
|
|
RegCloseKey( hKey );
|
|
|
|
|
|
|
|
return rVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-02-20 16:27:15 +00:00
|
|
|
/**
|
|
|
|
* This function terminates a process by process id instead of a handle.
|
|
|
|
**/
|
|
|
|
BOOL TerminateProcessById( DWORD dwProcessID ) {
|
|
|
|
HANDLE hProcess;
|
|
|
|
BOOL bRetVal = FALSE;
|
|
|
|
|
|
|
|
hProcess = OpenProcess( PROCESS_TERMINATE, FALSE, dwProcessID );
|
|
|
|
|
|
|
|
if (hProcess) {
|
|
|
|
bRetVal = TerminateProcess(hProcess, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
CloseHandle( hProcess );
|
|
|
|
|
|
|
|
return bRetVal;
|
|
|
|
}
|
|
|
|
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This function adjusts the specified WindowStation to include the specfied
|
|
|
|
* user.
|
|
|
|
*
|
|
|
|
* See: http://msdn2.microsoft.com/en-us/library/aa379608(VS.85).aspx
|
|
|
|
**/
|
|
|
|
BOOL AddAceToWindowStation(HWINSTA hwinsta, PSID psid)
|
|
|
|
{
|
|
|
|
ACCESS_ALLOWED_ACE *pace = NULL;
|
|
|
|
ACL_SIZE_INFORMATION aclSizeInfo;
|
|
|
|
BOOL bDaclExist;
|
|
|
|
BOOL bDaclPresent;
|
|
|
|
BOOL bSuccess = FALSE;
|
|
|
|
DWORD dwNewAclSize;
|
|
|
|
DWORD dwSidSize = 0;
|
|
|
|
DWORD dwSdSizeNeeded;
|
|
|
|
PACL pacl = NULL;
|
|
|
|
PACL pNewAcl = NULL;
|
|
|
|
PSECURITY_DESCRIPTOR psd = NULL;
|
|
|
|
PSECURITY_DESCRIPTOR psdNew = NULL;
|
|
|
|
PVOID pTempAce;
|
|
|
|
SECURITY_INFORMATION si = DACL_SECURITY_INFORMATION;
|
|
|
|
unsigned int i;
|
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
try
|
2008-02-15 15:52:16 +00:00
|
|
|
{
|
|
|
|
// Obtain the DACL for the window station.
|
|
|
|
|
|
|
|
if (!GetUserObjectSecurity(
|
|
|
|
hwinsta,
|
|
|
|
&si,
|
|
|
|
psd,
|
|
|
|
dwSidSize,
|
|
|
|
&dwSdSizeNeeded)
|
|
|
|
)
|
|
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
|
|
|
|
{
|
|
|
|
psd = (PSECURITY_DESCRIPTOR)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
dwSdSizeNeeded);
|
|
|
|
|
|
|
|
if (psd == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
psdNew = (PSECURITY_DESCRIPTOR)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
dwSdSizeNeeded);
|
|
|
|
|
|
|
|
if (psdNew == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
dwSidSize = dwSdSizeNeeded;
|
|
|
|
|
|
|
|
if (!GetUserObjectSecurity(
|
|
|
|
hwinsta,
|
|
|
|
&si,
|
|
|
|
psd,
|
|
|
|
dwSidSize,
|
|
|
|
&dwSdSizeNeeded)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
else
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Create a new DACL.
|
|
|
|
|
|
|
|
if (!InitializeSecurityDescriptor(
|
|
|
|
psdNew,
|
|
|
|
SECURITY_DESCRIPTOR_REVISION)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Get the DACL from the security descriptor.
|
|
|
|
|
|
|
|
if (!GetSecurityDescriptorDacl(
|
|
|
|
psd,
|
|
|
|
&bDaclPresent,
|
|
|
|
&pacl,
|
|
|
|
&bDaclExist)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Initialize the ACL.
|
|
|
|
|
|
|
|
ZeroMemory(&aclSizeInfo, sizeof(ACL_SIZE_INFORMATION));
|
|
|
|
aclSizeInfo.AclBytesInUse = sizeof(ACL);
|
|
|
|
|
|
|
|
// Call only if the DACL is not NULL.
|
|
|
|
|
|
|
|
if (pacl != NULL)
|
|
|
|
{
|
|
|
|
// get the file ACL size info
|
|
|
|
if (!GetAclInformation(
|
|
|
|
pacl,
|
|
|
|
(LPVOID)&aclSizeInfo,
|
|
|
|
sizeof(ACL_SIZE_INFORMATION),
|
|
|
|
AclSizeInformation)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compute the size of the new ACL.
|
|
|
|
|
|
|
|
dwNewAclSize = aclSizeInfo.AclBytesInUse +
|
|
|
|
(2*sizeof(ACCESS_ALLOWED_ACE)) + (2*GetLengthSid(psid)) -
|
|
|
|
(2*sizeof(DWORD));
|
|
|
|
|
|
|
|
// Allocate memory for the new ACL.
|
|
|
|
|
|
|
|
pNewAcl = (PACL)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
dwNewAclSize);
|
|
|
|
|
|
|
|
if (pNewAcl == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Initialize the new DACL.
|
|
|
|
|
|
|
|
if (!InitializeAcl(pNewAcl, dwNewAclSize, ACL_REVISION))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// If DACL is present, copy it to a new DACL.
|
|
|
|
|
|
|
|
if (bDaclPresent)
|
|
|
|
{
|
|
|
|
// Copy the ACEs to the new ACL.
|
|
|
|
if (aclSizeInfo.AceCount)
|
|
|
|
{
|
|
|
|
for (i=0; i < aclSizeInfo.AceCount; i++)
|
|
|
|
{
|
|
|
|
// Get an ACE.
|
|
|
|
if (!GetAce(pacl, i, &pTempAce))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Add the ACE to the new ACL.
|
|
|
|
if (!AddAce(
|
|
|
|
pNewAcl,
|
|
|
|
ACL_REVISION,
|
|
|
|
MAXDWORD,
|
|
|
|
pTempAce,
|
|
|
|
((PACE_HEADER)pTempAce)->AceSize)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the first ACE to the window station.
|
|
|
|
|
|
|
|
pace = (ACCESS_ALLOWED_ACE *)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
2009-12-16 19:55:09 +00:00
|
|
|
sizeof(ACCESS_ALLOWED_ACE) + GetLengthSid(psid) - sizeof(DWORD)
|
|
|
|
);
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
if (pace == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
pace->Header.AceType = ACCESS_ALLOWED_ACE_TYPE;
|
|
|
|
pace->Header.AceFlags = CONTAINER_INHERIT_ACE |
|
2009-12-16 19:55:09 +00:00
|
|
|
INHERIT_ONLY_ACE |
|
|
|
|
OBJECT_INHERIT_ACE;
|
|
|
|
pace->Header.AceSize = (WORD)sizeof(ACCESS_ALLOWED_ACE) +
|
|
|
|
(WORD)GetLengthSid(psid) -
|
|
|
|
(WORD)sizeof(DWORD);
|
2008-03-31 19:05:27 +00:00
|
|
|
pace->Mask = GENERIC_ALL;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
if (!CopySid(GetLengthSid(psid), &pace->SidStart, psid))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
if (!AddAce(
|
|
|
|
pNewAcl,
|
|
|
|
ACL_REVISION,
|
|
|
|
MAXDWORD,
|
|
|
|
(LPVOID)pace,
|
|
|
|
pace->Header.AceSize)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
2008-03-31 19:05:27 +00:00
|
|
|
// Add an ACE to the window station.
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
pace->Header.AceFlags = NO_PROPAGATE_INHERIT_ACE;
|
2008-03-31 19:05:27 +00:00
|
|
|
pace->Mask = GENERIC_ALL;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
if (!AddAce(
|
|
|
|
pNewAcl,
|
|
|
|
ACL_REVISION,
|
|
|
|
MAXDWORD,
|
|
|
|
(LPVOID)pace,
|
|
|
|
pace->Header.AceSize)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Set a new DACL for the security descriptor.
|
|
|
|
|
|
|
|
if (!SetSecurityDescriptorDacl(
|
|
|
|
psdNew,
|
|
|
|
TRUE,
|
|
|
|
pNewAcl,
|
|
|
|
FALSE)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Set the new security descriptor for the window station.
|
|
|
|
|
|
|
|
if (!SetUserObjectSecurity(hwinsta, &si, psdNew))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Indicate success.
|
|
|
|
|
|
|
|
bSuccess = TRUE;
|
|
|
|
}
|
2008-03-07 04:27:51 +00:00
|
|
|
catch(...)
|
2008-02-15 15:52:16 +00:00
|
|
|
{
|
|
|
|
// Free the allocated buffers.
|
|
|
|
|
|
|
|
if (pace != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)pace);
|
|
|
|
|
|
|
|
if (pNewAcl != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)pNewAcl);
|
|
|
|
|
|
|
|
if (psd != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)psd);
|
|
|
|
|
|
|
|
if (psdNew != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)psdNew);
|
|
|
|
}
|
|
|
|
|
|
|
|
return bSuccess;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function adjusts the specified Desktop to include the specfied
|
|
|
|
* user.
|
|
|
|
*
|
|
|
|
* See: http://msdn2.microsoft.com/en-us/library/aa379608(VS.85).aspx
|
|
|
|
**/
|
|
|
|
BOOL AddAceToDesktop(HDESK hdesk, PSID psid)
|
|
|
|
{
|
|
|
|
ACL_SIZE_INFORMATION aclSizeInfo;
|
|
|
|
BOOL bDaclExist;
|
|
|
|
BOOL bDaclPresent;
|
|
|
|
BOOL bSuccess = FALSE;
|
|
|
|
DWORD dwNewAclSize;
|
|
|
|
DWORD dwSidSize = 0;
|
|
|
|
DWORD dwSdSizeNeeded;
|
|
|
|
PACL pacl = NULL;
|
|
|
|
PACL pNewAcl = NULL;
|
|
|
|
PSECURITY_DESCRIPTOR psd = NULL;
|
|
|
|
PSECURITY_DESCRIPTOR psdNew = NULL;
|
|
|
|
PVOID pTempAce;
|
|
|
|
SECURITY_INFORMATION si = DACL_SECURITY_INFORMATION;
|
|
|
|
unsigned int i;
|
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
try
|
2008-02-15 15:52:16 +00:00
|
|
|
{
|
|
|
|
// Obtain the security descriptor for the desktop object.
|
|
|
|
|
|
|
|
if (!GetUserObjectSecurity(
|
|
|
|
hdesk,
|
|
|
|
&si,
|
|
|
|
psd,
|
|
|
|
dwSidSize,
|
|
|
|
&dwSdSizeNeeded))
|
|
|
|
{
|
|
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
|
|
|
|
{
|
|
|
|
psd = (PSECURITY_DESCRIPTOR)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
dwSdSizeNeeded );
|
|
|
|
|
|
|
|
if (psd == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
psdNew = (PSECURITY_DESCRIPTOR)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
dwSdSizeNeeded);
|
|
|
|
|
|
|
|
if (psdNew == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
dwSidSize = dwSdSizeNeeded;
|
|
|
|
|
|
|
|
if (!GetUserObjectSecurity(
|
|
|
|
hdesk,
|
|
|
|
&si,
|
|
|
|
psd,
|
|
|
|
dwSidSize,
|
|
|
|
&dwSdSizeNeeded)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
else
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new security descriptor.
|
|
|
|
|
|
|
|
if (!InitializeSecurityDescriptor(
|
|
|
|
psdNew,
|
|
|
|
SECURITY_DESCRIPTOR_REVISION)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Obtain the DACL from the security descriptor.
|
|
|
|
|
|
|
|
if (!GetSecurityDescriptorDacl(
|
|
|
|
psd,
|
|
|
|
&bDaclPresent,
|
|
|
|
&pacl,
|
|
|
|
&bDaclExist)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Initialize.
|
|
|
|
|
|
|
|
ZeroMemory(&aclSizeInfo, sizeof(ACL_SIZE_INFORMATION));
|
|
|
|
aclSizeInfo.AclBytesInUse = sizeof(ACL);
|
|
|
|
|
|
|
|
// Call only if NULL DACL.
|
|
|
|
|
|
|
|
if (pacl != NULL)
|
|
|
|
{
|
|
|
|
// Determine the size of the ACL information.
|
|
|
|
|
|
|
|
if (!GetAclInformation(
|
|
|
|
pacl,
|
|
|
|
(LPVOID)&aclSizeInfo,
|
|
|
|
sizeof(ACL_SIZE_INFORMATION),
|
|
|
|
AclSizeInformation)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compute the size of the new ACL.
|
|
|
|
|
|
|
|
dwNewAclSize = aclSizeInfo.AclBytesInUse +
|
|
|
|
sizeof(ACCESS_ALLOWED_ACE) +
|
|
|
|
GetLengthSid(psid) - sizeof(DWORD);
|
|
|
|
|
|
|
|
// Allocate buffer for the new ACL.
|
|
|
|
|
|
|
|
pNewAcl = (PACL)HeapAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
dwNewAclSize);
|
|
|
|
|
|
|
|
if (pNewAcl == NULL)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Initialize the new ACL.
|
|
|
|
|
|
|
|
if (!InitializeAcl(pNewAcl, dwNewAclSize, ACL_REVISION))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// If DACL is present, copy it to a new DACL.
|
|
|
|
|
|
|
|
if (bDaclPresent)
|
|
|
|
{
|
|
|
|
// Copy the ACEs to the new ACL.
|
|
|
|
if (aclSizeInfo.AceCount)
|
|
|
|
{
|
|
|
|
for (i=0; i < aclSizeInfo.AceCount; i++)
|
|
|
|
{
|
|
|
|
// Get an ACE.
|
|
|
|
if (!GetAce(pacl, i, &pTempAce))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Add the ACE to the new ACL.
|
|
|
|
if (!AddAce(
|
|
|
|
pNewAcl,
|
|
|
|
ACL_REVISION,
|
|
|
|
MAXDWORD,
|
|
|
|
pTempAce,
|
|
|
|
((PACE_HEADER)pTempAce)->AceSize)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add ACE to the DACL.
|
|
|
|
|
|
|
|
if (!AddAccessAllowedAce(
|
|
|
|
pNewAcl,
|
|
|
|
ACL_REVISION,
|
2008-03-31 19:05:27 +00:00
|
|
|
GENERIC_ALL,
|
2008-02-15 15:52:16 +00:00
|
|
|
psid)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Set new DACL to the new security descriptor.
|
|
|
|
|
|
|
|
if (!SetSecurityDescriptorDacl(
|
|
|
|
psdNew,
|
|
|
|
TRUE,
|
|
|
|
pNewAcl,
|
|
|
|
FALSE)
|
|
|
|
)
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Set the new security descriptor for the desktop object.
|
|
|
|
|
|
|
|
if (!SetUserObjectSecurity(hdesk, &si, psdNew))
|
2008-03-07 04:27:51 +00:00
|
|
|
throw;
|
2008-02-15 15:52:16 +00:00
|
|
|
|
|
|
|
// Indicate success.
|
|
|
|
|
|
|
|
bSuccess = TRUE;
|
|
|
|
}
|
2008-03-07 04:27:51 +00:00
|
|
|
catch(...)
|
2008-02-15 15:52:16 +00:00
|
|
|
{
|
|
|
|
// Free buffers.
|
|
|
|
|
|
|
|
if (pNewAcl != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)pNewAcl);
|
|
|
|
|
|
|
|
if (psd != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)psd);
|
|
|
|
|
|
|
|
if (psdNew != NULL)
|
|
|
|
HeapFree(GetProcessHeap(), 0, (LPVOID)psdNew);
|
|
|
|
}
|
|
|
|
|
|
|
|
return bSuccess;
|
|
|
|
}
|
|
|
|
|
2008-02-29 15:20:19 +00:00
|
|
|
|
|
|
|
/*++
|
|
|
|
This function attempts to obtain a SID representing the supplied
|
|
|
|
account on the supplied system.
|
|
|
|
|
|
|
|
If the function succeeds, the return value is TRUE. A buffer is
|
|
|
|
allocated which contains the SID representing the supplied account.
|
|
|
|
This buffer should be freed when it is no longer needed by calling
|
|
|
|
HeapFree(GetProcessHeap(), 0, buffer)
|
|
|
|
|
|
|
|
If the function fails, the return value is FALSE. Call GetLastError()
|
|
|
|
to obtain extended error information.
|
|
|
|
|
|
|
|
Scott Field (sfield) 12-Jul-95
|
|
|
|
--*/
|
|
|
|
|
|
|
|
BOOL
|
|
|
|
GetAccountSid(
|
2009-04-23 03:40:49 +00:00
|
|
|
LPCSTR SystemName,
|
|
|
|
LPCSTR AccountName,
|
2008-02-29 15:20:19 +00:00
|
|
|
PSID *Sid
|
|
|
|
)
|
|
|
|
{
|
2009-04-23 03:40:49 +00:00
|
|
|
LPSTR ReferencedDomain=NULL;
|
2008-02-29 15:20:19 +00:00
|
|
|
DWORD cbSid=128; // initial allocation attempt
|
|
|
|
DWORD cchReferencedDomain=16; // initial allocation size
|
|
|
|
SID_NAME_USE peUse;
|
|
|
|
BOOL bSuccess=FALSE; // assume this function will fail
|
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// initial memory allocations
|
|
|
|
//
|
|
|
|
*Sid = (PSID)HeapAlloc(GetProcessHeap(), 0, cbSid);
|
2008-02-29 15:20:19 +00:00
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
if(*Sid == NULL) throw;
|
|
|
|
|
2009-04-23 03:40:49 +00:00
|
|
|
ReferencedDomain = (LPSTR)HeapAlloc(
|
2008-02-29 15:20:19 +00:00
|
|
|
GetProcessHeap(),
|
|
|
|
0,
|
2009-04-23 03:40:49 +00:00
|
|
|
cchReferencedDomain * sizeof(CHAR)
|
2008-02-29 15:20:19 +00:00
|
|
|
);
|
2008-03-07 04:27:51 +00:00
|
|
|
|
|
|
|
if(ReferencedDomain == NULL) throw;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Obtain the SID of the specified account on the specified system.
|
|
|
|
//
|
2009-04-23 03:40:49 +00:00
|
|
|
while(!LookupAccountNameA(
|
2008-03-07 04:27:51 +00:00
|
|
|
SystemName, // machine to lookup account on
|
|
|
|
AccountName, // account to lookup
|
|
|
|
*Sid, // SID of interest
|
|
|
|
&cbSid, // size of SID
|
|
|
|
ReferencedDomain, // domain account was found on
|
|
|
|
&cchReferencedDomain,
|
|
|
|
&peUse
|
|
|
|
)) {
|
|
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
|
|
|
|
//
|
|
|
|
// reallocate memory
|
|
|
|
//
|
|
|
|
*Sid = (PSID)HeapReAlloc(
|
|
|
|
GetProcessHeap(),
|
|
|
|
0,
|
|
|
|
*Sid,
|
|
|
|
cbSid
|
|
|
|
);
|
|
|
|
if(*Sid == NULL) throw;
|
|
|
|
|
2009-04-23 03:40:49 +00:00
|
|
|
ReferencedDomain = (LPSTR)HeapReAlloc(
|
2008-03-07 04:27:51 +00:00
|
|
|
GetProcessHeap(),
|
|
|
|
0,
|
|
|
|
ReferencedDomain,
|
2009-04-23 03:40:49 +00:00
|
|
|
cchReferencedDomain * sizeof(CHAR)
|
2008-03-07 04:27:51 +00:00
|
|
|
);
|
|
|
|
if(ReferencedDomain == NULL) throw;
|
|
|
|
}
|
|
|
|
else throw;
|
2008-02-29 15:20:19 +00:00
|
|
|
}
|
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
//
|
|
|
|
// Indicate success.
|
|
|
|
//
|
|
|
|
bSuccess = TRUE;
|
2008-02-29 15:20:19 +00:00
|
|
|
|
|
|
|
} // try
|
2008-03-07 04:27:51 +00:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Cleanup and indicate failure, if appropriate.
|
|
|
|
//
|
2008-02-29 15:20:19 +00:00
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
HeapFree(GetProcessHeap(), 0, ReferencedDomain);
|
2008-02-29 15:20:19 +00:00
|
|
|
|
2008-03-07 04:27:51 +00:00
|
|
|
if(!bSuccess) {
|
|
|
|
if(*Sid != NULL) {
|
|
|
|
HeapFree(GetProcessHeap(), 0, *Sid);
|
|
|
|
*Sid = NULL;
|
|
|
|
}
|
2008-02-29 15:20:19 +00:00
|
|
|
}
|
|
|
|
} // finally
|
|
|
|
|
|
|
|
return bSuccess;
|
|
|
|
}
|
|
|
|
|
2008-05-22 03:09:11 +00:00
|
|
|
void chdir_to_data_dir() {
|
2011-01-07 20:23:22 +00:00
|
|
|
LONG lReturnValue;
|
|
|
|
HKEY hkSetupHive;
|
2008-05-22 03:09:11 +00:00
|
|
|
LPTSTR lpszRegistryValue = NULL;
|
2011-01-07 20:23:22 +00:00
|
|
|
DWORD dwSize = 0;
|
2008-05-22 03:09:11 +00:00
|
|
|
|
|
|
|
// change the current directory to the boinc data directory if it exists
|
2011-01-07 20:23:22 +00:00
|
|
|
lReturnValue = RegOpenKeyEx(
|
2008-05-22 03:09:11 +00:00
|
|
|
HKEY_LOCAL_MACHINE,
|
|
|
|
_T("SOFTWARE\\Space Sciences Laboratory, U.C. Berkeley\\BOINC Setup"),
|
2011-01-07 20:23:22 +00:00
|
|
|
0,
|
2008-05-22 03:09:11 +00:00
|
|
|
KEY_READ,
|
|
|
|
&hkSetupHive
|
|
|
|
);
|
|
|
|
if (lReturnValue == ERROR_SUCCESS) {
|
|
|
|
// How large does our buffer need to be?
|
|
|
|
lReturnValue = RegQueryValueEx(
|
|
|
|
hkSetupHive,
|
|
|
|
_T("DATADIR"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&dwSize
|
|
|
|
);
|
|
|
|
if (lReturnValue != ERROR_FILE_NOT_FOUND) {
|
|
|
|
// Allocate the buffer space.
|
|
|
|
lpszRegistryValue = (LPTSTR) malloc(dwSize);
|
|
|
|
(*lpszRegistryValue) = NULL;
|
|
|
|
|
|
|
|
// Now get the data
|
|
|
|
lReturnValue = RegQueryValueEx(
|
|
|
|
hkSetupHive,
|
|
|
|
_T("DATADIR"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
(LPBYTE)lpszRegistryValue,
|
|
|
|
&dwSize
|
|
|
|
);
|
|
|
|
|
|
|
|
SetCurrentDirectory(lpszRegistryValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-07 20:23:22 +00:00
|
|
|
if (hkSetupHive) RegCloseKey(hkSetupHive);
|
2008-05-22 03:09:11 +00:00
|
|
|
if (lpszRegistryValue) free(lpszRegistryValue);
|
|
|
|
}
|
2009-07-08 19:33:27 +00:00
|
|
|
|
2009-07-25 02:27:42 +00:00
|
|
|
|
2009-07-08 19:33:27 +00:00
|
|
|
// return true if running under remote desktop
|
2011-02-18 20:32:43 +00:00
|
|
|
// (in which case CUDA and Stream apps don't work)
|
2009-07-08 19:33:27 +00:00
|
|
|
//
|
2009-10-13 18:12:33 +00:00
|
|
|
typedef BOOL (__stdcall *tWTSQSI)( IN HANDLE, IN DWORD, IN DWORD, OUT LPTSTR*, OUT DWORD* );
|
|
|
|
typedef VOID (__stdcall *tWTSFM)( IN PVOID );
|
2009-07-25 02:27:42 +00:00
|
|
|
|
2009-10-13 18:12:33 +00:00
|
|
|
bool is_remote_desktop() {
|
|
|
|
static HMODULE wtsapi32lib = NULL;
|
|
|
|
static tWTSQSI pWTSQSI = NULL;
|
|
|
|
static tWTSFM pWTSFM = NULL;
|
|
|
|
LPTSTR pBuf = NULL;
|
|
|
|
DWORD dwLength;
|
2013-01-08 05:36:44 +00:00
|
|
|
USHORT usProtocol=0, usConnectionState=0;
|
2009-10-13 18:12:33 +00:00
|
|
|
|
|
|
|
if (!wtsapi32lib) {
|
2009-10-13 22:13:27 +00:00
|
|
|
wtsapi32lib = LoadLibrary(_T("wtsapi32.dll"));
|
2009-10-13 18:12:33 +00:00
|
|
|
if (wtsapi32lib) {
|
2009-10-20 14:51:20 +00:00
|
|
|
pWTSQSI = (tWTSQSI)GetProcAddress(wtsapi32lib, "WTSQuerySessionInformationA");
|
2009-10-13 18:12:33 +00:00
|
|
|
pWTSFM = (tWTSFM)GetProcAddress(wtsapi32lib, "WTSFreeMemory");
|
|
|
|
}
|
|
|
|
}
|
2009-07-25 02:27:42 +00:00
|
|
|
|
2009-10-13 18:12:33 +00:00
|
|
|
if (pWTSQSI) {
|
2011-02-18 20:32:43 +00:00
|
|
|
|
|
|
|
// WTSQuerySessionInformation(
|
|
|
|
// WTS_CURRENT_SERVER_HANDLE,
|
|
|
|
// WTS_CURRENT_SESSION,
|
|
|
|
// WTSClientProtocolType,
|
|
|
|
// &pBuf,
|
|
|
|
// &dwLength
|
|
|
|
// );
|
2009-10-13 18:12:33 +00:00
|
|
|
if (pWTSQSI(
|
|
|
|
(HANDLE)NULL,
|
|
|
|
(DWORD)-1,
|
|
|
|
(DWORD)16,
|
|
|
|
&pBuf,
|
|
|
|
&dwLength
|
|
|
|
)) {
|
2011-02-18 20:32:43 +00:00
|
|
|
usProtocol = *(USHORT*)pBuf;
|
2009-10-13 18:12:33 +00:00
|
|
|
pWTSFM(pBuf);
|
|
|
|
}
|
2011-02-18 20:32:43 +00:00
|
|
|
|
|
|
|
// WTSQuerySessionInformation(
|
|
|
|
// WTS_CURRENT_SERVER_HANDLE,
|
|
|
|
// WTS_CURRENT_SESSION,
|
|
|
|
// WTSConnectState,
|
|
|
|
// &pBuf,
|
|
|
|
// &dwLength
|
|
|
|
// );
|
|
|
|
if (pWTSQSI(
|
|
|
|
(HANDLE)NULL,
|
|
|
|
(DWORD)-1,
|
|
|
|
(DWORD)8,
|
|
|
|
&pBuf,
|
|
|
|
&dwLength
|
|
|
|
)) {
|
|
|
|
usConnectionState = *(USHORT*)pBuf;
|
|
|
|
pWTSFM(pBuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
// RDP Session implies Remote Desktop
|
|
|
|
if (usProtocol == 2) return true;
|
|
|
|
|
|
|
|
// Fast User Switching keeps the protocol set to the console but changes
|
|
|
|
// the connected state to disconnected.
|
|
|
|
if ((usProtocol == 0) && (usConnectionState == 4)) return true;
|
|
|
|
|
2009-07-08 19:33:27 +00:00
|
|
|
}
|
2009-07-25 02:27:42 +00:00
|
|
|
|
2009-07-08 19:33:27 +00:00
|
|
|
return false;
|
|
|
|
}
|
2009-07-25 02:27:42 +00:00
|
|
|
|
2013-01-08 05:36:44 +00:00
|
|
|
std::wstring A2W(const std::string& str) {
|
|
|
|
int length_wide = MultiByteToWideChar(CP_ACP, 0, str.data(), -1, NULL, 0);
|
|
|
|
wchar_t *string_wide = static_cast<wchar_t*>(_alloca((length_wide * sizeof(wchar_t)) + sizeof(wchar_t)));
|
|
|
|
MultiByteToWideChar(CP_ACP, 0, str.data(), -1, string_wide, length_wide);
|
|
|
|
std::wstring result(string_wide, length_wide);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string W2A(const std::wstring& str) {
|
|
|
|
int length_ansi = WideCharToMultiByte(CP_UTF8, 0, str.data(), -1, NULL, 0, NULL, NULL);
|
|
|
|
char* string_ansi = static_cast<char*>(_alloca(length_ansi + sizeof(char)));
|
|
|
|
WideCharToMultiByte(CP_UTF8, 0, str.data(), -1, string_ansi, length_ansi, NULL, NULL);
|
|
|
|
std::string result(string_ansi, length_ansi);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get message for given error
|
|
|
|
//
|
|
|
|
char* windows_format_error_string(
|
|
|
|
unsigned long dwError, char* pszBuf, int iSize
|
|
|
|
) {
|
|
|
|
DWORD dwRet;
|
2013-01-08 07:44:20 +00:00
|
|
|
LPWSTR lpszTemp = NULL;
|
2013-01-08 05:36:44 +00:00
|
|
|
|
2013-01-08 07:44:20 +00:00
|
|
|
dwRet = FormatMessageW(
|
2013-01-08 05:36:44 +00:00
|
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
|
|
FORMAT_MESSAGE_ARGUMENT_ARRAY,
|
|
|
|
NULL,
|
|
|
|
dwError,
|
|
|
|
LANG_NEUTRAL,
|
2013-01-08 07:44:20 +00:00
|
|
|
(LPWSTR)&lpszTemp,
|
2013-01-08 05:36:44 +00:00
|
|
|
0,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
|
2013-01-29 17:47:26 +00:00
|
|
|
if (dwRet != 0) {
|
|
|
|
// convert from current character encoding into UTF8
|
|
|
|
std::string encoded_message = W2A(std::wstring(lpszTemp));
|
2013-01-08 05:36:44 +00:00
|
|
|
|
2013-01-29 17:47:26 +00:00
|
|
|
// include the hex error code as well
|
|
|
|
snprintf(pszBuf, iSize, "%s (0x%x)", encoded_message.c_str(), dwError);
|
2013-01-08 05:36:44 +00:00
|
|
|
|
2013-01-29 17:47:26 +00:00
|
|
|
if (lpszTemp) {
|
|
|
|
LocalFree((HLOCAL) lpszTemp);
|
|
|
|
}
|
2013-04-11 08:22:43 +00:00
|
|
|
} else {
|
|
|
|
strcpy(pszBuf, "(unknown error)");
|
2013-01-08 05:36:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return pszBuf;
|
|
|
|
}
|
|
|
|
|