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

326 lines
10 KiB
C++

#include "precomp.h"
#include <ntverp.h> //these are for
#include <common.ver> //ver_productversion_str
// private forward declarations
#define LOCK_FILENAME TEXT("LOCK")
#define MAX_LOCK_SPINS 10 // number of timeouts before we give up
#define LOCK_TIMEOUT 2000 // in milliseconds
static inline LPPROPSHEETCOOKIE getPropSheetCookie(HWND hDlg);
int SIEErrorMessageBox(HWND hDlg, UINT idErrStr, UINT uiFlags /* = 0 */)
{
static TCHAR s_szTitle[128];
TCHAR szMessage[MAX_PATH];
if (ISNULL(s_szTitle))
LoadString(g_hInstance, IDS_SIE_NAME, s_szTitle, countof(s_szTitle));
if (LoadString(g_hInstance, idErrStr, szMessage, countof(szMessage)) == 0)
LoadString(g_hUIInstance, idErrStr, szMessage, countof(szMessage));
return MessageBox(hDlg, szMessage, s_szTitle, uiFlags ? uiFlags : MB_OK | MB_SETFOREGROUND | MB_ICONEXCLAMATION);
}
void CreateWorkDir(LPCTSTR pcszInsFile, LPCTSTR pcszFeatureDir, LPTSTR pszWorkDir,
LPCTSTR pcszCabDir /* = NULL */, BOOL fCreate /* = TRUE */)
{
StrCpy(pszWorkDir, pcszInsFile);
PathRemoveFileSpec(pszWorkDir);
if (pcszCabDir != NULL)
PathAppend(pszWorkDir, pcszCabDir);
PathAppend(pszWorkDir, pcszFeatureDir);
if (fCreate)
{
if (!PathFileExists(pszWorkDir))
PathCreatePath(pszWorkDir);
}
}
UINT CALLBACK PropSheetPageProc(HWND hwnd, UINT uMsg, LPPROPSHEETPAGE ppsp)
{
UNREFERENCED_PARAMETER(hwnd);
if ((uMsg == PSPCB_RELEASE) && (ppsp->lParam != NULL))
{
CoTaskMemFree((LPVOID)ppsp->lParam);
ppsp->lParam = NULL;
}
return TRUE;
}
void SetPropSheetCookie(HWND hDlg, LPARAM lParam)
{
LPPROPSHEETCOOKIE lpPropSheetCookie = (LPPROPSHEETCOOKIE)(((LPPROPSHEETPAGE)lParam)->lParam);
TCHAR szTitle[MAX_PATH];
INT iNamePrefID = lpPropSheetCookie->lpResultItem->iNamePrefID;
BOOL fPrefTitle = (!InsIsKeyEmpty(IS_BRANDING, IK_GPE_ONETIME_GUID, lpPropSheetCookie->pCS->GetInsFile())) &&
(iNamePrefID != -1);
SetWindowLongPtr(hDlg, DWLP_USER, (LONG_PTR)lpPropSheetCookie);
// overload the page title to get rid of the "properties" suffix
if (LoadString(g_hUIInstance, fPrefTitle ?
iNamePrefID : lpPropSheetCookie->lpResultItem->iNameID,
szTitle, countof(szTitle)) != 0)
SetWindowText(GetParent(hDlg), szTitle);
}
LPCTSTR GetInsFile(LPVOID lpVoid)
{
LPPROPSHEETCOOKIE lpPropSheetCookie = getPropSheetCookie((HWND)lpVoid);
return lpPropSheetCookie->pCS->GetInsFile();
}
void ShowHelpTopic(LPVOID lpVoid)
{
LPPROPSHEETCOOKIE lpPropSheetCookie = getPropSheetCookie((HWND)lpVoid);
if (lpPropSheetCookie && lpPropSheetCookie->lpResultItem)
{
WCHAR wszHelpTopic[MAX_PATH];
StrCpyW(wszHelpTopic, HELP_FILENAME TEXT("::/"));
StrCatW(wszHelpTopic, lpPropSheetCookie->lpResultItem->pcszHelpTopic);
MMCPropertyHelp((LPOLESTR)wszHelpTopic);
}
}
BOOL AcquireWriteCriticalSection(HWND hDlg, CComponentData * pCDCurrent /* = NULL */,
BOOL fCreateCookie /* = TRUE */)
{
CComponentData * pCD;
TCHAR szLockFile[MAX_PATH];
LPTSTR pszLockName;
HANDLE hLock;
DWORD dwRet;
BOOL fRet = TRUE;
if (hDlg != NULL)
{
LPPROPSHEETCOOKIE lpPropSheetCookie = getPropSheetCookie(hDlg);
pCD = lpPropSheetCookie->pCS->GetCompData();
}
else
if (pCDCurrent)
pCD = pCDCurrent;
else
{
fRet = false;
goto exit;
}
StrCpy(szLockFile, pCD->GetInsFile());
PathRemoveFileSpec(szLockFile);
PathAppend(szLockFile, TEXT("LOCK"));
PathCreatePath(szLockFile);
pszLockName = PathAddBackslash(szLockFile);
StrCpy(pszLockName, LOCK_FILENAME);
// (pritobla): Delete the lock file before acquiring it.
// If someone else has acquired it, the delete would fail, which is correct behavior.
// If the file was left behind (probably because the machine crashed the last time the file was created),
// then the delete would succeed and subsequently, the CreateFile would succeed.
DeleteFile(szLockFile);
hLock = CreateFile(szLockFile, GENERIC_WRITE, 0, NULL, CREATE_NEW,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, NULL);
if (hLock == INVALID_HANDLE_VALUE)
{
HANDLE hNotify;
DWORD dwSpins = 0;
*pszLockName = TEXT('\0');
hNotify = FindFirstChangeNotification(szLockFile, FALSE, FILE_NOTIFY_CHANGE_FILE_NAME);
if (hNotify == INVALID_HANDLE_VALUE)
{
fRet = FALSE;
goto exit;
}
StrCpy(pszLockName, LOCK_FILENAME);
do
{
while (((dwRet = MsgWaitForMultipleObjects(1, &hNotify, FALSE, LOCK_TIMEOUT, QS_ALLINPUT)) != WAIT_OBJECT_0) &&
(dwRet != WAIT_TIMEOUT))
{
MSG msg;
while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
dwSpins++;
hLock = CreateFile(szLockFile, GENERIC_WRITE, 0, NULL, CREATE_NEW,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, NULL);
if ((hLock != INVALID_HANDLE_VALUE) || (dwSpins == MAX_LOCK_SPINS))
break;
}
while (FindNextChangeNotification(hNotify));
if (!FindCloseChangeNotification(hNotify) || (hLock == INVALID_HANDLE_VALUE))
{
if (hLock != INVALID_HANDLE_VALUE)
CloseHandle(hLock);
fRet = FALSE;
goto exit;
}
}
pCD->SetLockHandle(hLock);
if (fCreateCookie)
{
// now that we have the lock, create our cookie file in our root GPO dir
*(pszLockName-1) = TEXT('\0');
PathRemoveFileSpec(szLockFile);
PathAppend(szLockFile, IEAK_GPE_COOKIE_FILE);
hLock = CreateFile(szLockFile, GENERIC_WRITE, 0, NULL, CREATE_NEW,
FILE_ATTRIBUTE_NORMAL, NULL);
if (hLock != INVALID_HANDLE_VALUE)
CloseHandle(hLock);
else
{
ASSERT(FALSE);
}
}
exit:
if (!fRet)
{
TCHAR szTitle[MAX_PATH], szMsg[MAX_PATH];
LoadString(g_hInstance, IDS_SIE_NAME, szTitle, countof(szTitle));
LoadString(g_hInstance, IDS_ERROR_SAVE, szMsg, countof(szMsg));
MessageBox(hDlg, szMsg, szTitle, MB_OK | MB_SETFOREGROUND | MB_ICONEXCLAMATION);
}
return fRet;
}
void ReleaseWriteCriticalSection(CComponentData * pCD, BOOL fDeleteCookie, BOOL fApplyPolicy,
BOOL bMachine /* = FALSE */, BOOL bAdd /* = FALSE */,
GUID *pGuidExtension /* = NULL */, GUID *pGuidSnapin /* = NULL */)
{
HANDLE hLock;
if (fDeleteCookie)
{
TCHAR szCookieFile[MAX_PATH];
StrCpy(szCookieFile, pCD->GetInsFile());
PathRemoveFileSpec(szCookieFile);
PathAppend(szCookieFile, IEAK_GPE_COOKIE_FILE);
DeleteFile(szCookieFile);
if (fApplyPolicy)
pCD->SignalPolicyChanged(bMachine, bAdd, pGuidExtension, pGuidSnapin);
}
if ((hLock = pCD->GetLockHandle()) != INVALID_HANDLE_VALUE)
{
pCD->SetLockHandle(INVALID_HANDLE_VALUE);
CloseHandle(hLock);
}
else
{
ASSERT(FALSE);
}
}
void SignalPolicyChanged(HWND hDlg, BOOL bMachine, BOOL bAdd, GUID *pGuidExtension,
GUID *pGuidSnapin, BOOL fAdvanced /* = FALSE */)
{
LPPROPSHEETCOOKIE lpPropSheetCookie = getPropSheetCookie(hDlg);
TCHAR szCookieFile[MAX_PATH];
TCHAR szGuid[128];
GUID guid;
USES_CONVERSION;
StrCpy(szCookieFile, lpPropSheetCookie->pCS->GetInsFile());
WritePrivateProfileString(BRANDING, GPVERKEY, A2CT(VER_PRODUCTVERSION_STR), szCookieFile);
//clear other keys so we're sure this is GP
WritePrivateProfileString(BRANDING, PMVERKEY, NULL, szCookieFile);
WritePrivateProfileString(BRANDING, IK_WIZVERSION, NULL, szCookieFile);
// write out a new guid for one time branding to the ins file if there was already one
// there to signify apply only once is checked
if (!InsIsKeyEmpty(IS_BRANDING, IK_GPE_ONETIME_GUID, szCookieFile))
{
if (CoCreateGuid(&guid) == NOERROR)
CoStringFromGUID(guid, szGuid, countof(szGuid));
else
szGuid[64] = TEXT('\0');
InsWriteString(IS_BRANDING, IK_GPE_ONETIME_GUID, szGuid, szCookieFile);
}
// write out a separate guid to track adms since they are always preferences
if (fAdvanced)
{
if (CoCreateGuid(&guid) == NOERROR)
CoStringFromGUID(guid, szGuid, countof(szGuid));
else
szGuid[64] = TEXT('\0');
InsWriteString(IS_BRANDING, IK_GPE_ADM_GUID, szGuid, szCookieFile);
}
InsFlushChanges(szCookieFile);
ReleaseWriteCriticalSection(lpPropSheetCookie->pCS->GetCompData(), TRUE, TRUE, bMachine, bAdd,
pGuidExtension, pGuidSnapin);
}
LPCTSTR GetCurrentAdmFile(LPVOID lpVoid)
{
LPPROPSHEETCOOKIE lpPropSheetCookie = getPropSheetCookie((HWND)lpVoid);
return lpPropSheetCookie->lpResultItem->pszDesc;
}
LPTSTR res2Str(int nIDString, LPTSTR pszBuffer, UINT cbBuffer)
{
if (pszBuffer == NULL || cbBuffer == 0)
return NULL;
*pszBuffer = TEXT('\0');
if (LoadString(g_hInstance, nIDString, pszBuffer, cbBuffer) == 0)
LoadString(g_hUIInstance, nIDString, pszBuffer, cbBuffer);
return pszBuffer;
}
// private helper APIs for this file
static inline LPPROPSHEETCOOKIE getPropSheetCookie(HWND hDlg)
{
return (LPPROPSHEETCOOKIE)GetWindowLongPtr(hDlg, DWLP_USER);
}