Windows2003-3790/admin/cmdline/scheduledtasks/scheduledtasks.cpp
2020-09-30 16:53:55 +02:00

1125 lines
31 KiB
C++

/******************************************************************************
Copyright(c) Microsoft Corporation
Module Name:
ScheduledTasks.cpp
Abstract:
This module initialises the OLE library,Interfaces, & reads the input data
from the command line.This module calls the appropriate functions for acheiving
the functionality of different options.
Author:
Raghu B 10-Sep-2000
Revision History:
Raghu B 10-Sep-2000 : Created it
G.Surender Reddy 25-sep-2000 : Modified it
[ Added error checking ]
G.Surender Reddy 10-oct-2000 : Modified it
[ Moved the strings to Resource table ]
Venu Gopal Choudary 01-Mar-2001 : Modified it
[ Added -change option]
Venu Gopal Choudary 12-Mar-2001 : Modified it
[ Added -run and -end options]
******************************************************************************/
//common header files needed for this file
#include "pch.h"
#include "CommonHeaderFiles.h"
/******************************************************************************
Routine Description:
This function process the options specified in the command line & routes to
different appropriate options [-create,-query,-delete,-change,-run,-end]
handling functions.This is the MAIN entry point for this utility.
Arguments:
[ in ] argc : The count of arguments specified in the command line
[ in ] argv : Array of command line arguments
Return Value :
A DWORD value indicating EXIT_SUCCESS on success else
EXIT_FAILURE on failure
******************************************************************************/
DWORD _cdecl
wmain(
IN DWORD argc,
IN LPCTSTR argv[]
)
{
// Declaring the main option switches as boolean values
BOOL bUsage = FALSE;
BOOL bCreate = FALSE;
BOOL bQuery = FALSE;
BOOL bDelete = FALSE;
BOOL bChange = FALSE;
BOOL bRun = FALSE;
BOOL bEnd = FALSE;
BOOL bDefVal = FALSE;
DWORD dwRetStatus = EXIT_SUCCESS;
HRESULT hr = S_OK;
// Call the preProcessOptions function to find out the option selected by the user
BOOL bValue = PreProcessOptions( argc , argv , &bUsage , &bCreate , &bQuery , &bDelete ,
&bChange , &bRun , &bEnd , &bDefVal );
if(bValue == FALSE)
{
ReleaseGlobals();
return EXIT_FAILURE;
}
// If ScheduledTasks.exe /?
if( bUsage && ( bCreate + bQuery + bDelete + bChange + bRun + bEnd ) == 0 )
{
displayMainUsage();
ReleaseGlobals();
return EXIT_SUCCESS;
}
// If ScheduledTasks.exe -create option is selected
if( bCreate == TRUE)
{
hr = CreateScheduledTask( argc, argv );
ReleaseGlobals();
if ( FAILED(hr) )
{
return EXIT_FAILURE;
}
else
{
return EXIT_SUCCESS;
}
}
// If ScheduledTasks.exe -Query option is selected
if( bQuery == TRUE )
{
dwRetStatus = QueryScheduledTasks( argc, argv );
ReleaseGlobals();
return dwRetStatus;
}
// If ScheduledTasks.exe -delete option is selected
if( bDelete == TRUE)
{
dwRetStatus = DeleteScheduledTask( argc, argv );
ReleaseGlobals();
return dwRetStatus;
}
// If ScheduledTasks.exe -change option is selected
if( bChange == TRUE)
{
dwRetStatus = ChangeScheduledTaskParams( argc, argv );
ReleaseGlobals();
return dwRetStatus;
}
// If ScheduledTasks.exe -run option is selected
if( bRun == TRUE)
{
dwRetStatus = RunScheduledTask( argc, argv );
ReleaseGlobals();
return dwRetStatus;
}
// If ScheduledTasks.exe -end option is selected
if( bEnd == TRUE)
{
dwRetStatus = TerminateScheduledTask( argc, argv );
ReleaseGlobals();
return dwRetStatus;
}
// If ScheduledTasks.exe option is selected
if( bDefVal == TRUE )
{
dwRetStatus = QueryScheduledTasks( argc, argv );
ReleaseGlobals();
return dwRetStatus;
}
ReleaseGlobals();
return dwRetStatus;
}
/******************************************************************************
Routine Description:
This function process the options specified in the command line & routes to
different appropriate functions.
Arguments:
[ in ] argc : The count of arguments specified in the command line
[ in ] argv : Array of command line arguments
[ out ] pbUsage : pointer to flag for determining [usage] -? option
[ out ] pbCreate : pointer to flag for determining -create option
[ out ] pbQuery : pointer to flag for determining -query option
[ out ] pbDelete : pointer to flag for determining -delete option
[ out ] pbChange : pointer to flag for determining -change option
[ out ] pbRun : pointer to flag for determining -run option
[ out ] pbEnd : pointer to flag for determining -end option
[ out ] pbDefVal : pointer to flag for determining default value
Return Value :
A BOOL value indicating TRUE on success else FALSE
******************************************************************************/
BOOL
PreProcessOptions(
IN DWORD argc,
IN LPCTSTR argv[] ,
OUT PBOOL pbUsage,
OUT PBOOL pbCreate,
OUT PBOOL pbQuery,
OUT PBOOL pbDelete ,
OUT PBOOL pbChange ,
OUT PBOOL pbRun ,
OUT PBOOL pbEnd ,
OUT PBOOL pbDefVal
)
{
// sub-local variables
TCMDPARSER2 cmdOptions[MAX_MAIN_COMMANDLINE_OPTIONS];
BOOL bReturn = FALSE;
//BOOL bOthers = FALSE;
// command line options
const WCHAR szCreateOpt[] = L"create";
const WCHAR szDeleteOpt[] = L"delete";
const WCHAR szQueryOpt[] = L"query";
const WCHAR szChangeOpt[] = L"change";
const WCHAR szRunOpt[] = L"run";
const WCHAR szEndOpt[] = L"end";
const WCHAR szHelpOpt[] = L"?";
TARRAY arrTemp = NULL;
arrTemp = CreateDynamicArray();
if( NULL == arrTemp)
{
SetLastError((DWORD)E_OUTOFMEMORY);
SaveLastError();
ShowLastErrorEx(stderr, SLE_ERROR| SLE_INTERNAL);
return FALSE;
}
SecureZeroMemory(cmdOptions,sizeof(TCMDPARSER2) * MAX_MAIN_COMMANDLINE_OPTIONS);
//
// fill the commandline parser
//
// /? option
StringCopyA( cmdOptions[ OI_USAGE ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_USAGE ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_USAGE ].pwszOptions = szHelpOpt;
cmdOptions[ OI_USAGE ].dwCount = 1;
cmdOptions[ OI_USAGE ].dwFlags = CP2_USAGE ;
cmdOptions[ OI_USAGE ].pValue = pbUsage;
// /create option
StringCopyA( cmdOptions[ OI_CREATE ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_CREATE ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_CREATE ].pwszOptions = szCreateOpt;
cmdOptions[ OI_CREATE ].dwCount = 1;
cmdOptions[ OI_CREATE ].pValue = pbCreate;
// /delete option
StringCopyA( cmdOptions[ OI_DELETE ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_DELETE ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_DELETE ].pwszOptions = szDeleteOpt;
cmdOptions[ OI_DELETE ].dwCount = 1;
cmdOptions[ OI_DELETE ].dwActuals = 0;
cmdOptions[ OI_DELETE ].pValue = pbDelete;
// /query option
StringCopyA( cmdOptions[ OI_QUERY ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_QUERY ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_QUERY ].pwszOptions = szQueryOpt;
cmdOptions[ OI_QUERY ].dwCount = 1;
cmdOptions[ OI_QUERY ].pValue = pbQuery;
// /change option
StringCopyA( cmdOptions[ OI_CHANGE ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_CHANGE ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_CHANGE ].pwszOptions = szChangeOpt;
cmdOptions[ OI_CHANGE ].dwCount = 1;
cmdOptions[ OI_CHANGE ].pValue = pbChange;
// /run option
StringCopyA( cmdOptions[ OI_RUN ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_RUN ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_RUN ].pwszOptions = szRunOpt;
cmdOptions[ OI_RUN ].dwCount = 1;
cmdOptions[ OI_RUN ].pValue = pbRun;
// /end option
StringCopyA( cmdOptions[ OI_END ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_END ].dwType = CP_TYPE_BOOLEAN;
cmdOptions[ OI_END ].pwszOptions = szEndOpt;
cmdOptions[ OI_END ].dwCount = 1;
cmdOptions[ OI_END ].pValue = pbEnd;
// default/sub options
StringCopyA( cmdOptions[ OI_OTHERS ].szSignature, "PARSER2\0", 8 );
cmdOptions[ OI_OTHERS ].dwType = CP_TYPE_TEXT;
cmdOptions[ OI_OTHERS ].dwFlags = CP2_MODE_ARRAY|CP2_DEFAULT;
cmdOptions[ OI_OTHERS ].pValue = &arrTemp;
//parse command line arguments
bReturn = DoParseParam2( argc, argv, -1, MAX_MAIN_COMMANDLINE_OPTIONS, cmdOptions, 0);
if( FALSE == bReturn) // Invalid commandline
{
//display an error message
ShowLastErrorEx ( stderr, SLE_TYPE_ERROR | SLE_INTERNAL );
// destroy dynamic array
if(arrTemp != NULL)
{
DestroyDynamicArray(&arrTemp);
arrTemp = NULL;
}
ReleaseGlobals();
return FALSE;
}
// destroy dynamic array
if(arrTemp != NULL)
{
DestroyDynamicArray(&arrTemp);
arrTemp = NULL;
}
//
// check for invalid syntax
//
if ( (( *pbCreate + *pbQuery + *pbDelete + *pbChange + *pbRun + *pbEnd ) == 0) &&
(TRUE == *pbUsage) && (argc > 2) )
{
ShowMessage( stderr, GetResString(IDS_RES_ERROR ));
return FALSE;
}
if(((*pbCreate + *pbQuery + *pbDelete + *pbChange + *pbRun + *pbEnd)> 1 ) ||
(( *pbCreate + *pbQuery + *pbDelete + *pbChange + *pbRun + *pbEnd + *pbUsage ) == 0 ) )
{
if ( ( *pbCreate + *pbQuery + *pbDelete + *pbChange + *pbRun + *pbEnd + *pbUsage ) > 1 )
{
ShowMessage( stderr, GetResString(IDS_RES_ERROR ));
return FALSE;
}
else if( *pbCreate == TRUE )
{
ShowMessage(stderr, GetResString(IDS_CREATE_USAGE));
return FALSE;
}
else if( *pbQuery == TRUE )
{
ShowMessage(stderr, GetResString(IDS_QUERY_USAGE));
return FALSE;
}
else if( *pbDelete == TRUE )
{
ShowMessage(stderr, GetResString(IDS_DELETE_SYNERROR));
return FALSE;
}
else if( *pbChange == TRUE )
{
ShowMessage(stderr, GetResString(IDS_CHANGE_SYNERROR));
return FALSE;
}
else if( *pbRun == TRUE )
{
ShowMessage(stderr, GetResString(IDS_RUN_SYNERROR));
return FALSE;
}
else if( *pbEnd == TRUE )
{
ShowMessage(stderr, GetResString(IDS_END_SYNERROR));
return FALSE;
}
else if( (!( *pbQuery )) && ( argc > 2 ) )
{
ShowMessage( stderr, GetResString(IDS_RES_ERROR ));
return FALSE;
}
else
{
*pbDefVal = TRUE;
}
}
return TRUE;
}
/******************************************************************************
Routine Description:
This function fetches the ITaskScheduler Interface.It also connects to
the remote machine if specified & helps to operate
ITaskScheduler on the specified target m/c.
Arguments:
[ in ] szServer : server's name
Return Value :
ITaskScheduler interface pointer on success else NULL
******************************************************************************/
ITaskScheduler*
GetTaskScheduler(
IN LPCTSTR szServer
)
{
HRESULT hr = S_OK;
ITaskScheduler *pITaskScheduler = NULL;
LPWSTR wszComputerName = NULL;
WCHAR wszActualComputerName[ 2 * MAX_STRING_LENGTH ] = DOMAIN_U_STRING;
wchar_t* pwsz = L"";
WORD wSlashCount = 0 ;
hr = Init( &pITaskScheduler );
if( FAILED(hr))
{
SetLastError ((DWORD) hr);
ShowLastErrorEx ( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
return NULL;
}
//If the operation is on remote machine
if( IsLocalSystem(szServer) == FALSE )
{
wszComputerName = (LPWSTR)szServer;
//check whether the server name prefixed with \\ or not.
if( wszComputerName != NULL )
{
pwsz = wszComputerName;
while ( ( *pwsz != NULL_U_CHAR ) && ( *pwsz == BACK_SLASH_U ) )
{
// server name prefixed with '\'..
// so..increment the pointer and count number of black slashes..
pwsz = _wcsinc(pwsz);
wSlashCount++;
}
if( (wSlashCount == 2 ) ) // two back slashes are present
{
StringCopy( wszActualComputerName, wszComputerName, SIZE_OF_ARRAY(wszActualComputerName) );
}
else if ( wSlashCount == 0 )
{
//Append "\\" to computer name
StringConcat(wszActualComputerName, wszComputerName, 2 * MAX_RES_STRING);
}
else
{
// display an error message as invalid address specified.
ShowMessage (stderr, GetResString ( IDS_INVALID_NET_ADDRESS ));
return NULL;
}
}
hr = pITaskScheduler->SetTargetComputer( wszActualComputerName );
}
else
{
//Local Machine
hr = pITaskScheduler->SetTargetComputer( NULL );
}
if( FAILED( hr ) )
{
SetLastError ((DWORD) hr);
ShowLastErrorEx ( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
return NULL;
}
return pITaskScheduler;
}
/******************************************************************************
Routine Description:
This function initialises the COM library & fetches the ITaskScheduler interface.
Arguments:
[ in ] pITaskScheduler : double pointer to taskscheduler interface
Return Value:
A HRESULT value indicating success code else failure code
******************************************************************************/
HRESULT
Init(
IN OUT ITaskScheduler **pITaskScheduler
)
{
// Initalize the HRESULT value.
HRESULT hr = S_OK;
// Bring in the library
hr = CoInitializeEx( NULL , COINIT_APARTMENTTHREADED );
if (FAILED(hr))
{
return hr;
}
hr = CoInitializeSecurity(NULL, -1, NULL, NULL,
RPC_C_AUTHN_LEVEL_NONE,
RPC_C_IMP_LEVEL_IMPERSONATE,
NULL, EOAC_NONE, 0 );
if (FAILED(hr))
{
CoUninitialize();
return hr;
}
// Create the pointer to Task Scheduler object
// CLSID from the header file mstask.h
// Fill the task schdeuler object.
hr = CoCreateInstance( CLSID_CTaskScheduler, NULL, CLSCTX_ALL,
IID_ITaskScheduler,(LPVOID*) pITaskScheduler );
// Should we fail, unload the library
if (FAILED(hr))
{
CoUninitialize();
}
return hr;
}
/******************************************************************************
Routine Description:
This function releases the ITaskScheduler & unloads the COM library
Arguments:
[ in ] pITaskScheduler : pointer to the ITaskScheduler
Return Value :
VOID
******************************************************************************/
VOID
Cleanup(
IN ITaskScheduler *pITaskScheduler
)
{
if (pITaskScheduler)
{
pITaskScheduler->Release();
}
// Unload the library, now that our pointer is freed.
CoUninitialize();
return;
}
/******************************************************************************
Routine Description:
This function displays the main usage help of this utility
Arguments:
None
Return Value :
VOID
******************************************************************************/
VOID
displayMainUsage()
{
DisplayUsage( IDS_MAINHLP1, IDS_MAINHLP21);
return;
}
/******************************************************************************
Routine Description:
This function deletes the .job extension from the task name
Arguments:
[ in ] lpszTaskName : Task name
Return Value :
None
******************************************************************************/
DWORD
ParseTaskName(
IN LPWSTR lpszTaskName
)
{
if(lpszTaskName == NULL)
{
return ERROR_INVALID_PARAMETER;
}
// Remove the .Job extension from the task name
lpszTaskName[StringLength(lpszTaskName, 0 ) - StringLength(JOB, 0) ] = L'\0';
return EXIT_SUCCESS;
}
/******************************************************************************
Routine Description:
This function displays the messages for usage of different option
Arguments:
[ in ] StartingMessage : First string to display
[ in ] EndingMessage : Last string to display
Return Value :
DWORD
******************************************************************************/
DWORD
DisplayUsage(
IN ULONG StartingMessage,
IN ULONG EndingMessage
)
{
ULONG ulCounter = 0;
LPCTSTR lpszCurrentString = NULL;
for( ulCounter = StartingMessage; ulCounter <= EndingMessage; ulCounter++ )
{
lpszCurrentString = GetResString( ulCounter );
if( lpszCurrentString != NULL )
{
ShowMessage( stdout, _X(lpszCurrentString) );
}
else
{
return ERROR_INVALID_PARAMETER;
}
}
return ERROR_SUCCESS;
}
BOOL
GetGroupPolicy(
IN LPWSTR szServer,
IN LPWSTR szUserName,
IN LPWSTR szPolicyType,
OUT LPDWORD lpdwPolicy
)
/*++
Routine Description:
This function gets the value of a group policy in the registry
for a specified policy type.
Arguments:
[ in ] szServer : Server name
[ in ] szPolicyType : Policy Type
[ out ] lpdwPolicy : Value of the policy
Return Value :
DWORD
--*/
{
// sub-variables
LONG lResult = 0;
HKEY hKey = 0;
HKEY hLMKey = 0;
HKEY hUKey = 0;
HKEY hPolicyKey = 0;
PBYTE pByteData = NULL;
LPWSTR wszComputerName = NULL;
LPWSTR pwsz = NULL;
LPWSTR pszStopStr = NULL;
WCHAR wszActualComputerName[ 2 * MAX_STRING_LENGTH ];
WCHAR wszBuffer[ MAX_STRING_LENGTH ];
WCHAR wszSid[ MAX_STRING_LENGTH ];
DWORD dwType = 0;
WORD wSlashCount = 0;
DWORD dwPolicy = 0;
SecureZeroMemory ( wszActualComputerName, SIZE_OF_ARRAY(wszActualComputerName) );
SecureZeroMemory ( wszBuffer, SIZE_OF_ARRAY(wszBuffer) );
StringCopy ( wszActualComputerName, DOMAIN_U_STRING, SIZE_OF_ARRAY(wszActualComputerName) );
// check whether server name prefixed with "\\" or not..If not, append the same
// to the server name
if ( (StringLength (szServer, 0 ) != 0) && (IsLocalSystem (szServer) == FALSE ))
{
wszComputerName = (LPWSTR)szServer;
//check whether the server name prefixed with \\ or not.
if( wszComputerName != NULL )
{
pwsz = wszComputerName;
while ( ( *pwsz != NULL_U_CHAR ) && ( *pwsz == BACK_SLASH_U ) )
{
// server name prefixed with '\'..
// so..increment the pointer and count number of black slashes..
pwsz = _wcsinc(pwsz);
wSlashCount++;
}
if( (wSlashCount == 2 ) ) // two back slashes are present
{
StringCopy( wszActualComputerName, wszComputerName, SIZE_OF_ARRAY(wszActualComputerName) );
}
else if ( wSlashCount == 0 )
{
//Append "\\" to computer name
StringConcat(wszActualComputerName, wszComputerName, 2 * MAX_RES_STRING);
}
}
DWORD cbSid = 0;
DWORD cbDomainName = 0;
LPWSTR szDomain = NULL;
WCHAR szUser[MAX_RES_STRING+5];
SID_NAME_USE peUse;
PSID pSid = NULL;
DWORD dwUserLength = 0;
BOOL bResult = FALSE;
dwUserLength = MAX_RES_STRING + 5;
SecureZeroMemory (szUser, SIZE_OF_ARRAY(szUser));
if ( StringLength (szUserName, 0) == 0 )
{
if(FALSE == GetUserName ( szUser, &dwUserLength ))
{
SaveLastError();
return FALSE;
}
szUserName = szUser;
}
#ifdef _WIN64
INT64 dwPos ;
#else
DWORD dwPos ;
#endif
pszStopStr = StrRChrI( (LPCWSTR)szUserName, NULL, L'\\' );
if ( NULL != pszStopStr )
{
pszStopStr++;
szUserName = pszStopStr;
}
//
// Get the actual size of domain name and SID
//
bResult = LookupAccountName( szServer, szUserName, pSid, &cbSid, szDomain, &cbDomainName, &peUse );
// API should have failed with insufficient buffer.
// allocate the buffer with the actual size
pSid = (PSID) AllocateMemory( cbSid );
if ( pSid == NULL )
{
SetLastError( ERROR_OUTOFMEMORY );
return FALSE;
}
// allocate the buffer with the actual size
szDomain = (LPWSTR) AllocateMemory(cbDomainName*sizeof(WCHAR));
if(NULL == szDomain)
{
SaveLastError();
FreeMemory((LPVOID*) &pSid);
return FALSE;
}
//Retrieve SID and Domain name for a specified computer and account names
if ( FALSE == LookupAccountName( szServer, szUserName, pSid, &cbSid, szDomain, &cbDomainName, &peUse ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
FreeMemory((LPVOID*) &pSid);
FreeMemory((LPVOID*) &szDomain);
return FALSE;
}
// Get SID string for a specified username
if ( FALSE == GetSidString ( pSid, wszSid ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
FreeMemory((LPVOID*) &pSid);
FreeMemory((LPVOID*) &szDomain);
return FALSE;
}
//release memory
FreeMemory((LPVOID*) &pSid);
FreeMemory((LPVOID*) &szDomain);
// form the registry path to get the value for policy
StringCopy ( wszBuffer, wszSid, SIZE_OF_ARRAY(wszBuffer));
StringConcat ( wszBuffer, L"\\", SIZE_OF_ARRAY(wszBuffer));
StringConcat ( wszBuffer, TS_KEYPOLICY_BASE, SIZE_OF_ARRAY(wszBuffer));
//
// Connect to the remote machine
//
// connect to HKEY_LOCAL_MACHINE on remote machine
lResult = RegConnectRegistry( wszActualComputerName, HKEY_LOCAL_MACHINE, &hLMKey );
if ( ERROR_SUCCESS != lResult )
{
SaveLastError();
return FALSE;
}
// connect to HKEY_USERS on remote machine
lResult = RegConnectRegistry( wszActualComputerName, HKEY_USERS, &hUKey );
if ( ERROR_SUCCESS != lResult )
{
SaveLastError();
return FALSE;
}
// check for NULL
if (NULL != hLMKey )
{
//
// Open the registry key
//
lResult = RegOpenKeyEx( hLMKey,
TS_KEYPOLICY_BASE, 0, KEY_READ, &hPolicyKey );
if ( NULL == hPolicyKey && NULL != hUKey)
{
lResult = RegOpenKeyEx( hUKey,
wszBuffer, 0, KEY_READ, &hPolicyKey );
}
}
// Get the value of a policy in the registry
if ( ( NULL != hPolicyKey ) && (FALSE == GetPolicyValue (hPolicyKey, szPolicyType, &dwPolicy) ) )
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
if ( NULL != hPolicyKey )
{
RegCloseKey (hPolicyKey);
hPolicyKey = NULL;
}
if ( NULL != hLMKey )
{
RegCloseKey (hLMKey);
hLMKey = NULL;
}
if ( NULL != hUKey )
{
RegCloseKey (hUKey);
hUKey = NULL;
}
return FALSE;
}
// release all the keys
if ( NULL != hPolicyKey )
{
RegCloseKey (hPolicyKey);
hPolicyKey = NULL;
}
if ( NULL != hLMKey )
{
RegCloseKey (hLMKey);
hLMKey = NULL;
}
if ( NULL != hUKey )
{
RegCloseKey (hUKey);
hUKey = NULL;
}
}
else
{
//
// Open the registry key for HKEY_LOCAL_MACHINE
//
lResult = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
TS_KEYPOLICY_BASE, 0, KEY_READ, &hKey );
if( lResult != ERROR_SUCCESS)
{
// check the keyvalue
if ( NULL == hKey )
{
//
// Open the registry key for HKEY_CURRENT_USER
//
lResult = RegOpenKeyEx( HKEY_CURRENT_USER,
TS_KEYPOLICY_BASE, 0, KEY_READ, &hKey );
}
}
// Get the value of a policy in the registry
if ( ( NULL != hKey ) && (FALSE == GetPolicyValue (hKey, szPolicyType, &dwPolicy) ))
{
ShowLastErrorEx( stderr, SLE_TYPE_ERROR | SLE_SYSTEM );
// release the resource
if ( NULL != hKey )
{
RegCloseKey (hKey);
}
return FALSE;
}
// check for NULL
if ( NULL != hKey )
{
RegCloseKey (hKey);
}
}
// assign the value
*lpdwPolicy = dwPolicy;
// return success
return TRUE;
}
BOOL
GetPolicyValue(
IN HKEY hKey,
IN LPWSTR szPolicyType,
OUT LPDWORD lpdwPolicy
)
/*++
Routine Description:
This function gets the value of a group policy in the registry
for a given Register Key
Arguments:
[ in ] hKey : Register Key
[ in ] szPolicyType : Policy Type
[ out ] lpdwPolicy : Value of the policy
Return Value :
BOOL
--*/
{
// sub-variables
LONG lResult = 0;
DWORD dwLength = 0;
LPBYTE pByteData = NULL;
DWORD dwType = 0;
// get the size of the buffer to hold the value associated with the value name
lResult = RegQueryValueEx( hKey, szPolicyType, NULL, NULL, NULL, &dwLength );
// allocate the buffer
pByteData = (LPBYTE) AllocateMemory( (dwLength + 10) * sizeof( BYTE ) );
if ( pByteData == NULL )
{
SetLastError( ERROR_OUTOFMEMORY );
return FALSE;
}
// now get the data
lResult = RegQueryValueEx( hKey, szPolicyType, NULL, &dwType, pByteData, &dwLength );
*lpdwPolicy = *((DWORD*) pByteData);
FreeMemory( (LPVOID*) &pByteData );
return TRUE;
}
BOOL
GetSidString (
IN PSID pSid,
OUT LPWSTR wszSid
)
/*++
Routine Description:
This function gets the SID string.
Arguments:
[IN] PSID pSid : SID structure
[OUT] LPWSTR wszSid : Stores SID string
Return Value:
TRUE On success
FALSE On failure
--*/
{
// sub-local variables
PSID_IDENTIFIER_AUTHORITY Auth ;
PUCHAR lpNbSubAuth ;
LPDWORD lpSubAuth = 0 ;
UCHAR uloop ;
WCHAR wszTmp[MAX_RES_STRING] ;
WCHAR wszStr[ MAX_RES_STRING ] ;
// initialize the variables
SecureZeroMemory ( wszTmp, SIZE_OF_ARRAY(wszTmp) );
SecureZeroMemory ( wszStr, SIZE_OF_ARRAY(wszStr) );
//Add the revision
StringCopy ( wszStr, SID_STRING, MAX_RES_STRING );
//Get identifier authority
Auth = GetSidIdentifierAuthority ( pSid ) ;
if ( NULL == Auth )
{
SaveLastError();
return FALSE ;
}
// format authority value
if ( (Auth->Value[0] != 0) || (Auth->Value[1] != 0) ) {
StringCchPrintf ( wszTmp, SIZE_OF_ARRAY(wszTmp), AUTH_FORMAT_STR1 ,
(ULONG)Auth->Value[0],
(ULONG)Auth->Value[1],
(ULONG)Auth->Value[2],
(ULONG)Auth->Value[3],
(ULONG)Auth->Value[4],
(ULONG)Auth->Value[5] );
}
else {
StringCchPrintf ( wszTmp, SIZE_OF_ARRAY(wszTmp), AUTH_FORMAT_STR2 ,
(ULONG)(Auth->Value[5] ) +
(ULONG)(Auth->Value[4] << 8) +
(ULONG)(Auth->Value[3] << 16) +
(ULONG)(Auth->Value[2] << 24) );
}
StringConcat (wszStr, DASH , SIZE_OF_ARRAY(wszStr));
StringConcat (wszStr, wszTmp, SIZE_OF_ARRAY(wszStr));
//Get sub authorities
lpNbSubAuth = GetSidSubAuthorityCount ( pSid ) ;
if ( NULL == lpNbSubAuth )
{
SaveLastError();
return FALSE ;
}
// loop through and get sub authority
for ( uloop = 0 ; uloop < *lpNbSubAuth ; uloop++ ) {
lpSubAuth = GetSidSubAuthority ( pSid,(DWORD)uloop ) ;
if ( NULL == lpSubAuth )
{
SaveLastError();
return FALSE;
}
// convert long integer to a string
_ultot (*lpSubAuth, wszTmp, BASE_TEN) ;
StringConcat ( wszStr, DASH, SIZE_OF_ARRAY(wszStr) ) ;
StringConcat (wszStr, wszTmp, SIZE_OF_ARRAY(wszStr) ) ;
}
StringCopy ( wszSid, wszStr, MAX_RES_STRING );
// retunr success
return TRUE ;
}