mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
493 lines
11 KiB
493 lines
11 KiB
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation 1996-2001.
|
|
//
|
|
// File: wrapper.cpp
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
#include "stdafx.h"
|
|
#include "util.h"
|
|
#include "resource.h"
|
|
#include "winreg.h"
|
|
#include "areaprog.h"
|
|
#include "wrapper.h"
|
|
|
|
#include "dsgetdc.h"
|
|
|
|
typedef DWORD (WINAPI *PFNDSGETDCNAME)(LPCWSTR, LPCWSTR, GUID *, LPCWSTR, ULONG, PDOMAIN_CONTROLLER_INFOW *);
|
|
|
|
typedef struct {
|
|
LPCWSTR szProfile;
|
|
LPCWSTR szDatabase;
|
|
LPCWSTR szLog;
|
|
AREA_INFORMATION Area;
|
|
LPVOID *pHandle;
|
|
PSCE_AREA_CALLBACK_ROUTINE pCallback;
|
|
HANDLE hWndCallback;
|
|
DWORD dwFlags;
|
|
} ENGINEARGS;
|
|
|
|
BOOL
|
|
PostProgressArea(
|
|
IN HANDLE CallbackHandle,
|
|
IN AREA_INFORMATION Area,
|
|
IN DWORD TotalTicks,
|
|
IN DWORD CurrentTicks
|
|
);
|
|
|
|
static BOOL bRangeSet=FALSE;
|
|
|
|
CRITICAL_SECTION csOpenDatabase;
|
|
#define OPEN_DATABASE_TIMEOUT INFINITE
|
|
|
|
//
|
|
// Helper functions to call the engine from a secondary thread:
|
|
//
|
|
DWORD WINAPI
|
|
InspectSystemEx(LPVOID lpv) {
|
|
|
|
if ( lpv == NULL ) return ERROR_INVALID_PARAMETER;
|
|
|
|
ENGINEARGS *ea;
|
|
SCESTATUS rc;
|
|
ea = (ENGINEARGS *)lpv;
|
|
|
|
DWORD dWarning=0;
|
|
rc = SceAnalyzeSystem(NULL,
|
|
ea->szProfile,
|
|
ea->szDatabase,
|
|
ea->szLog,
|
|
SCE_UPDATE_DB|SCE_VERBOSE_LOG,
|
|
ea->Area,
|
|
ea->pCallback,
|
|
ea->hWndCallback,
|
|
&dWarning // this is required (by RPC)
|
|
);
|
|
return rc;
|
|
}
|
|
|
|
//
|
|
// call to SCE engine to apply the template
|
|
//
|
|
|
|
DWORD WINAPI
|
|
ApplyTemplateEx(LPVOID lpv) {
|
|
|
|
if ( lpv == NULL ) return ERROR_INVALID_PARAMETER;
|
|
|
|
ENGINEARGS *ea;
|
|
SCESTATUS rc;
|
|
ea = (ENGINEARGS *)lpv;
|
|
|
|
rc = SceConfigureSystem(NULL,
|
|
ea->szProfile,
|
|
ea->szDatabase,
|
|
ea->szLog,
|
|
SCE_OVERWRITE_DB|SCE_VERBOSE_LOG,
|
|
ea->Area,
|
|
ea->pCallback,
|
|
ea->hWndCallback,
|
|
NULL
|
|
);
|
|
|
|
return rc;
|
|
}
|
|
|
|
|
|
WINBASEAPI
|
|
BOOL
|
|
WINAPI
|
|
TryEnterCriticalSection(
|
|
LPCRITICAL_SECTION lpCriticalSection
|
|
);
|
|
|
|
/*------------------------------------------------------------------------------
|
|
Method: OpenDatabaseEx
|
|
|
|
Synopsis: open database on a separate thread
|
|
|
|
Arugments:
|
|
|
|
Returns:
|
|
|
|
History: a-mthoge 06-09-1998 - Added the _NT4BACK_PORT compile condition.
|
|
------------------------------------------------------------------------------*/
|
|
DWORD
|
|
WINAPI
|
|
OpenDatabaseEx(LPVOID lpv) {
|
|
|
|
if ( lpv == NULL ) return ERROR_INVALID_PARAMETER;
|
|
|
|
ENGINEARGS *ea;
|
|
SCESTATUS rc=0;
|
|
|
|
|
|
if (TryEnterCriticalSection(&csOpenDatabase)) {
|
|
ea = (ENGINEARGS *)lpv;
|
|
|
|
rc = SceOpenProfile(ea->szProfile,
|
|
(SCE_FORMAT_TYPE) ea->dwFlags, // SCE_JET_FORMAT || SCE_JET_ANALYSIS_REQUIRED
|
|
ea->pHandle
|
|
);
|
|
|
|
LeaveCriticalSection(&csOpenDatabase);
|
|
} else {
|
|
rc = SCESTATUS_OTHER_ERROR;
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
//
|
|
// Assign a template to the system without configuring it
|
|
//
|
|
SCESTATUS
|
|
AssignTemplate(LPCWSTR szTemplate,
|
|
LPCWSTR szDatabase,
|
|
BOOL bIncremental) {
|
|
SCESTATUS rc;
|
|
|
|
rc = SceConfigureSystem(NULL,
|
|
szTemplate,
|
|
szDatabase,
|
|
NULL,
|
|
(bIncremental ? SCE_UPDATE_DB : SCE_OVERWRITE_DB) | SCE_NO_CONFIG | SCE_VERBOSE_LOG,
|
|
AREA_ALL,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
);
|
|
|
|
return rc;
|
|
}
|
|
|
|
//
|
|
// apply a template to the system
|
|
//
|
|
DWORD
|
|
ApplyTemplate(
|
|
LPCWSTR szProfile,
|
|
LPCWSTR szDatabase,
|
|
LPCWSTR szLogFile,
|
|
AREA_INFORMATION Area
|
|
)
|
|
{
|
|
// Spawn a thread to call the engine & apply the profile, since this can
|
|
// take a while and we want to stay responsive & have a change to provide
|
|
// feedback.
|
|
|
|
ENGINEARGS ea;
|
|
HANDLE hThread=NULL;
|
|
|
|
ea.szProfile = szProfile;
|
|
ea.szDatabase = szDatabase;
|
|
ea.szLog = szLogFile;
|
|
ea.Area = Area;
|
|
|
|
//
|
|
// this is the progress call back dialog which
|
|
// will be passed to SCE client stub for progress
|
|
// callback
|
|
//
|
|
AreaProgress *ap = new AreaProgress;
|
|
if ( ap ) {
|
|
|
|
CString strTitle;
|
|
CString strVerb;
|
|
strTitle.LoadString(IDS_CONFIGURE_PROGRESS_TITLE);
|
|
strVerb.LoadString(IDS_CONFIGURE_PROGRESS_VERB);
|
|
|
|
ap->Create(IDD_ANALYZE_PROGRESS);
|
|
ap->SetWindowText(strTitle);
|
|
ap->SetDlgItemText(IDC_VERB,strVerb);
|
|
ap->ShowWindow(SW_SHOW);
|
|
bRangeSet = FALSE;
|
|
}
|
|
|
|
ea.pCallback = (PSCE_AREA_CALLBACK_ROUTINE)PostProgressArea;
|
|
ea.hWndCallback = (HANDLE)ap;
|
|
|
|
hThread = CreateThread(NULL,0,ApplyTemplateEx,&ea,0,NULL);
|
|
|
|
DWORD dw=0;
|
|
if ( hThread ) {
|
|
|
|
MSG msg;
|
|
|
|
DWORD dwTotalTicks=100;
|
|
do {
|
|
|
|
dw = MsgWaitForMultipleObjects(1,&hThread,0,INFINITE,QS_ALLINPUT);
|
|
while (PeekMessage(&msg,NULL,0,0,PM_REMOVE)) {
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
} while (WAIT_OBJECT_0 != dw);
|
|
|
|
GetExitCodeThread(hThread,&dw);
|
|
|
|
CloseHandle(hThread);
|
|
|
|
} else {
|
|
|
|
dw = GetLastError();
|
|
|
|
CString str;
|
|
str.LoadString(IDS_CANT_CREATE_THREAD);
|
|
AfxMessageBox(str);
|
|
}
|
|
|
|
//
|
|
// free the dialog if it's created
|
|
//
|
|
if ( ap ) {
|
|
if ( ap->GetSafeHwnd() )
|
|
ap->DestroyWindow();
|
|
delete ap;
|
|
}
|
|
|
|
return dw;
|
|
}
|
|
|
|
//
|
|
// post progress
|
|
//
|
|
BOOL
|
|
PostProgressArea(
|
|
IN HANDLE CallbackHandle,
|
|
IN AREA_INFORMATION Area,
|
|
IN DWORD TotalTicks,
|
|
IN DWORD CurrentTicks
|
|
)
|
|
{
|
|
if ( CallbackHandle ) {
|
|
|
|
AreaProgress *ap = (AreaProgress *)CallbackHandle;
|
|
|
|
ap->ShowWindow(SW_SHOW);
|
|
|
|
if ( !bRangeSet ) {
|
|
ap->SetMaxTicks(TotalTicks);
|
|
bRangeSet = TRUE;
|
|
}
|
|
ap->SetCurTicks(CurrentTicks);
|
|
ap->SetArea(Area);
|
|
|
|
return TRUE;
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// inspect a system
|
|
//
|
|
DWORD
|
|
InspectSystem(
|
|
LPCWSTR szProfile,
|
|
LPCWSTR szDatabase,
|
|
LPCWSTR szLogFile,
|
|
AREA_INFORMATION Area
|
|
)
|
|
{
|
|
// Spawn a thread to call the engine & inspect the system, since this can
|
|
// take a while and we want to stay responsive & have a change to provide
|
|
// feedback.
|
|
|
|
ENGINEARGS ea;
|
|
HANDLE hThread=NULL;
|
|
|
|
ea.szProfile = szProfile;
|
|
ea.szDatabase = szDatabase;
|
|
ea.szLog = szLogFile;
|
|
ea.Area = Area;
|
|
|
|
AreaProgress *ap = new AreaProgress;
|
|
if ( ap ) {
|
|
|
|
ap->Create(IDD_ANALYZE_PROGRESS);
|
|
ap->ShowWindow(SW_SHOW);
|
|
bRangeSet = FALSE;
|
|
}
|
|
|
|
ea.pCallback = (PSCE_AREA_CALLBACK_ROUTINE)PostProgressArea;
|
|
ea.hWndCallback = (HANDLE)ap;
|
|
|
|
|
|
// return InspectSystemEx(&ea);
|
|
|
|
|
|
hThread = CreateThread(NULL,0,InspectSystemEx,&ea,0,NULL);
|
|
if (!hThread) {
|
|
DWORD rc = GetLastError();
|
|
|
|
CString str;
|
|
str.LoadString(IDS_CANT_CREATE_THREAD);
|
|
AfxMessageBox(str);
|
|
// Display an error
|
|
|
|
if ( ap ) {
|
|
if ( ap->GetSafeHwnd() )
|
|
ap->DestroyWindow();
|
|
delete ap;
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
MSG msg;
|
|
DWORD dw=0;
|
|
|
|
DWORD dwTotalTicks=100;
|
|
int n = 0;
|
|
do {
|
|
dw = MsgWaitForMultipleObjects(1,&hThread,0,100,QS_ALLINPUT);
|
|
while (PeekMessage(&msg,NULL,0,0,PM_REMOVE)) {
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
|
|
} while (WAIT_OBJECT_0 != dw);
|
|
|
|
GetExitCodeThread(hThread,&dw);
|
|
|
|
CloseHandle(hThread);
|
|
|
|
if ( ap ) {
|
|
if ( ap->GetSafeHwnd() )
|
|
ap->DestroyWindow();
|
|
delete ap;
|
|
}
|
|
|
|
return dw;
|
|
|
|
}
|
|
|
|
BOOL GetProfileDescription(LPCTSTR ProfileName, LPWSTR* Description)
|
|
// Description must be freed by LocalFree
|
|
// This should only be called for INF format profiles
|
|
{
|
|
PVOID hProfile=NULL;
|
|
SCESTATUS rc;
|
|
|
|
if (EngineOpenProfile(ProfileName,OPEN_PROFILE_CONFIGURE,&hProfile) == SCESTATUS_SUCCESS) {
|
|
rc = SceGetScpProfileDescription(
|
|
hProfile,
|
|
Description);
|
|
|
|
SceCloseProfile(&hProfile);
|
|
|
|
if ( rc == SCESTATUS_SUCCESS ) {
|
|
return(TRUE);
|
|
} else {
|
|
return(FALSE);
|
|
}
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
SCESTATUS
|
|
EngineOpenProfile(
|
|
LPCWSTR FileName OPTIONAL,
|
|
int format,
|
|
PVOID* hProfile
|
|
)
|
|
{
|
|
SCESTATUS status;
|
|
ENGINEARGS ea;
|
|
DWORD dw;
|
|
HANDLE hThread=NULL;
|
|
CString str;
|
|
|
|
if ( !hProfile ) { // do not check !FileName because it's optional now
|
|
return SCESTATUS_PROFILE_NOT_FOUND;
|
|
}
|
|
|
|
if ( (OPEN_PROFILE_LOCALPOL != format) &&
|
|
!FileName ) {
|
|
return SCESTATUS_PROFILE_NOT_FOUND;
|
|
}
|
|
|
|
ZeroMemory(&ea, sizeof( ENGINEARGS ) );
|
|
|
|
// This is multithreaded for responsiveness, since a
|
|
// crashed jet database can take forever and a day to open.
|
|
|
|
// If we can open it quickly (where quickly is defined as within
|
|
// OPEN_DATABASE_TIMEOUT milliseconds then
|
|
if ( (OPEN_PROFILE_ANALYSIS == format) ||
|
|
(OPEN_PROFILE_LOCALPOL == format)) {// JET {
|
|
ea.szProfile = FileName;
|
|
ea.pHandle = hProfile;
|
|
if (OPEN_PROFILE_LOCALPOL == format) {
|
|
ea.dwFlags = SCE_JET_FORMAT;
|
|
} else {
|
|
ea.dwFlags = SCE_JET_ANALYSIS_REQUIRED;
|
|
}
|
|
|
|
#if SPAWN_OPEN_DATABASE_THREAD
|
|
hThread = CreateThread(NULL,0,OpenDatabaseEx,&ea,0,NULL);
|
|
|
|
if ( hThread ) {
|
|
|
|
dw = MsgWaitForMultipleObjects(1,&hThread,0,OPEN_DATABASE_TIMEOUT,0);
|
|
if (WAIT_TIMEOUT == dw) {
|
|
status = SCESTATUS_OTHER_ERROR;
|
|
} else {
|
|
GetExitCodeThread(hThread,&status);
|
|
}
|
|
|
|
CloseHandle(hThread);
|
|
|
|
} else {
|
|
status = GetLastError();
|
|
}
|
|
#else
|
|
status = OpenDatabaseEx(&ea);
|
|
#endif
|
|
if( status != SCESTATUS_SUCCESS && *hProfile ){
|
|
status = SCESTATUS_INVALID_DATA;
|
|
}
|
|
|
|
} else { // INF
|
|
status = SceOpenProfile( FileName, SCE_INF_FORMAT, hProfile );
|
|
}
|
|
|
|
if ( status != SCESTATUS_SUCCESS ){
|
|
*hProfile = NULL;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
void EngineCloseProfile(PVOID* hProfile)
|
|
{
|
|
if ( hProfile ) {
|
|
SceCloseProfile(hProfile);
|
|
}
|
|
*hProfile = NULL;
|
|
}
|
|
|
|
BOOL EngineGetDescription(PVOID hProfile, LPWSTR* Desc)
|
|
{
|
|
if ( SceGetScpProfileDescription( hProfile, Desc) != SCESTATUS_SUCCESS ) {
|
|
return FALSE;
|
|
} else {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
BOOL IsDomainController( LPCTSTR pszComputer )
|
|
{
|
|
//
|
|
// for remote computers, connect to the remote registry
|
|
// currently this api only works for local computer
|
|
//
|
|
SCE_SERVER_TYPE ServerType = SCESVR_UNKNOWN;
|
|
SCESTATUS rc = SceGetServerProductType((LPTSTR)pszComputer, &ServerType);
|
|
return ( (SCESTATUS_SUCCESS == rc) && (SCESVR_DC_WITH_DS == ServerType) );
|
|
}
|
|
|
|
|