2020-09-30 16:53:55 +02:00

634 lines
18 KiB
C++

//***************************************************************************
//
// MAINDLL.CPP
//
// Module: WMI Framework Instance provider
//
// Purpose: Contains DLL entry points. Also has code that controls
// when the DLL can be unloaded by tracking the number of
// objects and locks as well as routines that support
// self registration.
//
// Copyright (c)1999 Microsoft Corporation, All Rights Reserved
//
//***************************************************************************
//
// need this to access older WMI methods
// BUGBUG - better alternative is to change the params
// to use the new methods
//
#define FRAMEWORK_ALLOW_DEPRECATED 0
#include "stdafx.h"
#include <FWcommon.h>
#include <objbase.h>
#include <initguid.h>
#include <wbemidl.h>
#include <setupbat.h>
#include "diskcleanup.h"
#include "..\rstrcore\resource.h"
#ifdef THIS_FILE
#undef THIS_FILE
#endif
static char __szTraceSourceFile[] = __FILE__;
#define THIS_FILE __szTraceSourceFile
void UnattendedFileParameters ();
HMODULE ghModule;
//============
WCHAR *GUIDSTRING = L"{a47401f6-a8a6-40ea-9c29-b8f6026c98b8}";
CLSID CLSID_SYSTEMRESTORE = {0xa47401f6, 0xa8a6, 0x40ea,
{0x9c, 0x29, 0xb8, 0xf6, 0x02, 0x6c, 0x98, 0xb8}} ;
WCHAR *GUID_SRDiskCleanup = L"{7325c922-bb81-47b0-8b2f-a5f8605e242f}";
CLSID Clsid_SRDiskCleanup = {/*7325c922-bb81-47b0-8b2f-a5f8605e242f*/
0x7325c922,
0xbb81,
0x47b0,
{0x8b, 0x2f, 0xa5, 0xf8, 0x60, 0x5e, 0x24, 0x2f}
};
//Count number of objects and number of locks.
long g_cLock=0;
//***************************************************************************
//
// DllGetClassObject
//
// Purpose: Called by Ole when some client wants a class factory. Return
// one only if it is the sort of class this DLL supports.
//
//***************************************************************************
STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, PPVOID ppv)
{
HRESULT hr;
if (CLSID_SYSTEMRESTORE == rclsid)
{
CWbemGlueFactory *pObj;
pObj=new CWbemGlueFactory();
if (NULL==pObj)
return E_OUTOFMEMORY;
hr=pObj->QueryInterface(riid, ppv);
if (FAILED(hr))
delete pObj;
}
else if (rclsid == Clsid_SRDiskCleanup)
{
CSRClassFactory *pcf = new CSRClassFactory ();
if (pcf == NULL)
return E_OUTOFMEMORY;
hr = pcf->QueryInterface (riid, ppv);
pcf->Release(); // release constructor refcount
}
else hr = E_FAIL;
return hr;
}
//***************************************************************************
//
// DllCanUnloadNow
//
// Purpose: Called periodically by Ole in order to determine if the
// DLL can be freed.
//
// Return: S_OK if there are no objects in use and the class factory
// isn't locked.
//
//***************************************************************************
STDAPI DllCanUnloadNow(void)
{
SCODE sc;
// It is OK to unload if there are no objects or locks on the
// class factory and the framework is done with you.
if ((0L==g_cLock) && CWbemProviderGlue::FrameworkLogoffDLL(L"SYSTEMRESTORE"))
{
sc = S_OK;
}
else
{
sc = S_FALSE;
}
return sc;
}
//***************************************************************************
//
// Is4OrMore
//
// Returns true if win95 or any version of NT > 3.51
//
//***************************************************************************
BOOL Is4OrMore(void)
{
OSVERSIONINFO os;
os.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
if(!GetVersionEx(&os))
return FALSE; // should never happen
return os.dwMajorVersion >= 4;
}
DWORD
CopyMofFile()
{
TENTER("CopyMofFile");
WCHAR szSrc[MAX_PATH];
IMofCompiler *pimof = NULL;
HRESULT hr;
WBEM_COMPILE_STATUS_INFO Info;
BOOL fInitialized = FALSE;
hr = CoInitialize(NULL);
if (FAILED(hr))
{
trace(0, "! CoInitialize : %ld", hr);
goto done;
}
fInitialized = TRUE;
//
// create an IMofCompiler instance
//
hr = CoCreateInstance( CLSID_MofCompiler,
0,
CLSCTX_INPROC_SERVER,
IID_IMofCompiler,
(LPVOID*) &pimof );
if (FAILED(hr) || NULL == pimof)
{
trace(0, "! CoCreateInstance : %ld", hr);
goto done;
}
//
// get the path of the source mof file
// %windir%\system32\restore\sr.mof
//
if (0 == ExpandEnvironmentStrings(s_cszWinRestDir, szSrc, MAX_PATH))
{
hr = (HRESULT) GetLastError();
trace(0, "! ExpandEnvironmentStrings : %ld", hr);
goto done;
}
lstrcat(szSrc, s_cszMofFile);
//
// compile the mof file
//
hr = pimof->CompileFile(szSrc,
0, // no server & namespace
0, // no user
0, // no authority
0, // no password
0, // no options
0, // no class flags
0, // no instance flags
&Info );
if (hr != S_OK)
{
trace(0, "! CompileFile : %ld", hr);
goto done;
}
done:
if (pimof)
{
pimof->Release();
}
if (fInitialized)
CoUninitialize();
TLEAVE();
return hr;
}
//***************************************************************************
//
// DllRegisterServer
//
// Purpose: Called during setup or by regsvr32.
//
// Return: NOERROR if registration successful, error otherwise.
//***************************************************************************
STDAPI DllRegisterServer(void)
{
WCHAR wcID[128];
WCHAR wcCLSID[128];
WCHAR wcModule[MAX_PATH];
WCHAR * pName = L"";
WCHAR * pModel = L"Both";
HKEY hKey1 = NULL, hKey2 = NULL, hKeySR = NULL;
DWORD dwRc;
// Create the path.
lstrcpy(wcCLSID, L"SOFTWARE\\CLASSES\\CLSID\\");
lstrcat(wcCLSID, GUIDSTRING);
// Create entries under CLSID
RegCreateKey(HKEY_LOCAL_MACHINE, wcCLSID, &hKey1);
if (hKey1 != NULL)
{
RegSetValueEx(hKey1, NULL, 0, REG_SZ, (BYTE *)pName,
(lstrlen(pName)+1) * sizeof(WCHAR));
RegCreateKey(hKey1, L"InprocServer32",&hKey2);
if (hKey2 != NULL)
{
GetModuleFileName(ghModule, wcModule, MAX_PATH);
RegSetValueEx(hKey2, NULL, 0, REG_SZ, (BYTE *) wcModule,
(lstrlen(wcModule)+1) * sizeof(WCHAR));
RegSetValueEx(hKey2, L"ThreadingModel", 0, REG_SZ,
(BYTE *)pModel, (lstrlen(pModel)+1) * sizeof(WCHAR));
CloseHandle(hKey2);
}
CloseHandle(hKey1);
}
// copy the sr.mof file into the wbem directories
// get the wbem directories from the registry
// dwRc = CopyMofFile();
lstrcpy(wcCLSID, L"SOFTWARE\\CLASSES\\CLSID\\");
lstrcat(wcCLSID, GUID_SRDiskCleanup);
RegCreateKey(HKEY_LOCAL_MACHINE, wcCLSID, &hKey1);
if (hKey1 != NULL)
{
RegSetValueEx(hKey1, NULL, 0, REG_SZ, (BYTE *)pName,
(lstrlen(pName)+1) * sizeof(WCHAR));
RegCreateKey(hKey1, L"InprocServer32", &hKey2);
if (hKey2 != NULL)
{
GetModuleFileName(ghModule, wcModule, MAX_PATH);
RegSetValueEx(hKey2, NULL, 0, REG_SZ, (BYTE *) wcModule,
(lstrlen(wcModule)+1) * sizeof(WCHAR));
RegSetValueEx(hKey2, L"ThreadingModel", 0, REG_SZ,
(BYTE *)pModel, (lstrlen(pModel)+1) * sizeof(WCHAR));
CloseHandle(hKey2);
}
RegCreateKey(hKey1, L"DefaultIcon", &hKey2);
if (hKey2 != NULL)
{
lstrcpy (wcModule, L"%SystemRoot%\\system32\\srclient.dll,0");
RegSetValueEx(hKey2, NULL, 0, REG_EXPAND_SZ, (BYTE *) wcModule,
(lstrlen(wcModule)+1) * sizeof(WCHAR));
CloseHandle(hKey2);
}
CloseHandle(hKey1);
}
RegCreateKey(HKEY_LOCAL_MACHINE,
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Explorer\\VolumeCaches"
L"\\System Restore", &hKey1);
if (hKey1 != NULL)
{
RegSetValueEx(hKey1, NULL, 0, REG_SZ, (BYTE *) GUID_SRDiskCleanup,
(lstrlen(GUID_SRDiskCleanup)+1) * sizeof(WCHAR));
}
UnattendedFileParameters();
return NOERROR;
}
//***************************************************************************
//
// DllUnregisterServer
//
// Purpose: Called when it is time to remove the registry entries.
//
// Return: NOERROR if registration successful, error otherwise.
//***************************************************************************
STDAPI DllUnregisterServer(void)
{
WCHAR wcID[128];
WCHAR wcCLSID[128];
HKEY hKey;
// Create the path using the CLSID
lstrcpy(wcCLSID, L"SOFTWARE\\CLASSES\\CLSID\\");
lstrcat(wcCLSID, GUIDSTRING);
// First delete the InProcServer subkey.
DWORD dwRet = RegOpenKey(HKEY_LOCAL_MACHINE, wcCLSID, &hKey);
if(dwRet == NO_ERROR)
{
RegDeleteKey(hKey, L"InProcServer32");
CloseHandle(hKey);
}
lstrcpy(wcCLSID, L"SOFTWARE\\CLASSES\\CLSID\\");
lstrcat(wcCLSID, GUID_SRDiskCleanup);
dwRet = RegOpenKey(HKEY_LOCAL_MACHINE, wcCLSID, &hKey);
if(dwRet == NO_ERROR)
{
RegDeleteKey(hKey, L"InProcServer32");
RegDeleteKey(hKey, L"DefaultIcon");
CloseHandle(hKey);
}
// then delete the clsid keys for both SystemRestoreProv and SRDiskCleanup
dwRet = RegOpenKey(HKEY_LOCAL_MACHINE, L"SOFTWARE\\CLASSES\\CLSID\\", &hKey);
if(dwRet == NO_ERROR)
{
RegDeleteKey(hKey, GUIDSTRING);
RegDeleteKey(hKey, GUID_SRDiskCleanup);
CloseHandle(hKey);
}
RegDeleteKey(HKEY_LOCAL_MACHINE,
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Explorer\\VolumeCaches"
L"\\System Restore");
return NOERROR;
}
void UnattendedFileParameters ()
{
HKEY hKey = NULL;
WCHAR * pwNull = L"";
DWORD dwAnswerLength = MAX_PATH;
LONG lAnswer = 0;
WCHAR wcsAnswerFile [MAX_PATH];
WCHAR wcsAnswer [MAX_PATH];
if (ERROR_SUCCESS != RegOpenKeyEx(HKEY_LOCAL_MACHINE, s_cszSRCfgRegKey, 0,
KEY_WRITE, &hKey))
return;
if (0 == GetSystemDirectoryW (wcsAnswerFile,MAX_PATH))
return;
lstrcatW (wcsAnswerFile, L"\\");
lstrcatW (wcsAnswerFile, WINNT_GUI_FILE_W);
/*
if( GetPrivateProfileString( s_cszSRUnattendedSection,
L"MaximumDataStoreSize",
pwNull,
wcsAnswer,
dwAnswerLength,
wcsAnswerFile ) )
{
if( lstrcmpW (pwNull, wcsAnswer ))
{
lAnswer = wcstol (wcsAnswer,NULL,10);
if (lAnswer > 0)
RegWriteDWORD(hKey, s_cszDSMax, (DWORD *) &lAnswer);
}
}
*/
if( GetPrivateProfileString( s_cszSRUnattendedSection,
L"RestorePointLife",
pwNull,
wcsAnswer,
dwAnswerLength,
wcsAnswerFile ) )
{
if( lstrcmpW (pwNull, wcsAnswer ))
{
lAnswer = wcstol (wcsAnswer,NULL,10);
lAnswer *= 24 * 3600; // convert days to seconds
if (lAnswer > 0)
RegWriteDWORD(hKey, s_cszRPLifeInterval, (DWORD *) &lAnswer);
}
}
if( GetPrivateProfileString( s_cszSRUnattendedSection,
L"CheckpointCalendarFrequency",
pwNull,
wcsAnswer,
dwAnswerLength,
wcsAnswerFile ) )
{
if( lstrcmpW (pwNull, wcsAnswer ))
{
lAnswer = wcstol (wcsAnswer,NULL,10);
lAnswer *= 24 * 3600; // convert days to seconds
if (lAnswer > 0)
RegWriteDWORD(hKey, s_cszRPGlobalInterval, (DWORD *) &lAnswer);
}
}
if( GetPrivateProfileString( s_cszSRUnattendedSection,
L"CheckPointSessionFrequency",
pwNull,
wcsAnswer,
dwAnswerLength,
wcsAnswerFile ) )
{
if( lstrcmpW (pwNull, wcsAnswer ))
{
lAnswer = wcstol (wcsAnswer,NULL,10);
lAnswer *= 3600; // convert hours to seconds
if (lAnswer > 0)
RegWriteDWORD(hKey, s_cszRPSessionInterval, (DWORD *) &lAnswer);
}
}
if( GetPrivateProfileString( s_cszSRUnattendedSection,
L"MaximumDataStorePercentOfDisk",
pwNull,
wcsAnswer,
dwAnswerLength,
wcsAnswerFile ) )
{
if( lstrcmpW (pwNull, wcsAnswer ))
{
lAnswer = wcstol (wcsAnswer,NULL,10);
if (lAnswer <= 100 && lAnswer > 0)
RegWriteDWORD(hKey, s_cszDiskPercent, (DWORD *) &lAnswer);
}
}
RegCloseKey (hKey);
return;
}
void CALLBACK
CreateFirstRunRp(
HWND hwnd,
HINSTANCE hinst,
LPSTR lpszCmdLine,
int nCmdShow)
{
RESTOREPOINTINFO RPInfo;
STATEMGRSTATUS SmgrStatus;
DWORD dwValue;
HANDLE hInit = NULL;
//
// first remove ourselves from Run key
//
HKEY hKey;
DWORD dwRet = RegOpenKey(HKEY_LOCAL_MACHINE,
L"Software\\Microsoft\\Windows\\CurrentVersion\\Run",
&hKey);
if (dwRet == ERROR_SUCCESS)
{
RegDeleteValue(hKey, L"SRFirstRun");
RegCloseKey(hKey);
}
//
// wait until the service has fully initialized
// query thrice at 10 second intervals
//
dwRet = WAIT_FAILED;
int i = 0;
while (i++ <= 3)
{
hInit = OpenEvent(SYNCHRONIZE, FALSE, s_cszSRInitEvent);
if (hInit == NULL)
{
if (i >= 3)
{
break;
}
Sleep(10000);
}
else
{
dwRet = WaitForSingleObject(hInit, 60*1000); // 1 minute
break;
}
}
//
// reset registry value CreateFirstRunRp
// so that service will create firstrun rp in the future
//
dwValue = 1;
if (ERROR_SUCCESS == RegOpenKey(HKEY_LOCAL_MACHINE,
s_cszSRRegKey,
&hKey))
{
RegWriteDWORD(hKey, s_cszCreateFirstRunRp, &dwValue);
RegCloseKey(hKey);
}
//
// try creating firstrun rp now
//
if (dwRet != WAIT_FAILED)
{
RPInfo.dwEventType = BEGIN_SYSTEM_CHANGE;
RPInfo.dwRestorePtType = FIRSTRUN;
if (ERROR_SUCCESS != SRLoadString(L"srrstr.dll", IDS_SYSTEM_CHECKPOINT_TEXT, RPInfo.szDescription, MAX_DESC_W))
{
lstrcpy(RPInfo.szDescription, s_cszSystemCheckpointName);
}
SRSetRestorePoint(&RPInfo, &SmgrStatus);
}
if (hInit)
{
CloseHandle(hInit);
}
return;
}
//***************************************************************************
//
// DllMain
//
// Purpose: Called by the operating system when processes and threads are
// initialized and terminated, or upon calls to the LoadLibrary
// and FreeLibrary functions
//
// Return: TRUE if load was successful, else FALSE
//***************************************************************************
BOOL APIENTRY DllMain ( HINSTANCE hInstDLL, // handle to dll module
DWORD fdwReason, // reason for calling function
LPVOID lpReserved ) // reserved
{
BOOL bRet = TRUE;
// Perform actions based on the reason for calling.
switch( fdwReason )
{
case DLL_PROCESS_ATTACH:
DisableThreadLibraryCalls(hInstDLL);
// Initialize once for each new process.
// Return FALSE to fail DLL load.
ghModule = hInstDLL;
bRet = CWbemProviderGlue::FrameworkLoginDLL(L"SYSTEMRESTORE");
break;
case DLL_THREAD_ATTACH:
// Do thread-specific initialization.
break;
case DLL_THREAD_DETACH:
// Do thread-specific cleanup.
break;
case DLL_PROCESS_DETACH:
// Perform any necessary cleanup.
break;
}
return bRet; // Sstatus of DLL_PROCESS_ATTACH.
}