Source code of Windows XP (NT5)
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.

722 lines
19 KiB

  1. //=============================================================================
  2. // MSConfig.cpp
  3. //
  4. // This contains the high level implementation of MSConfig - this class
  5. // creates all of the pages and displays a property sheet.
  6. //=============================================================================
  7. #include "stdafx.h"
  8. #include "MSConfig.h"
  9. #include <initguid.h>
  10. #include "MSConfig_i.c"
  11. #include "MSConfigState.h"
  12. #include "PageServices.h"
  13. #include "PageStartup.h"
  14. #include "PageBootIni.h"
  15. #include "PageIni.h"
  16. #include "PageGeneral.h"
  17. #include "MSConfigCtl.h"
  18. #include "AutoStartDlg.h"
  19. #ifdef _DEBUG
  20. #define new DEBUG_NEW
  21. #undef THIS_FILE
  22. static char THIS_FILE[] = __FILE__;
  23. #endif
  24. #define MSCONFIGDIR _T("%systemroot%\\pss")
  25. //-----------------------------------------------------------------------------
  26. // These global variables are pointers to each of the property pages shown
  27. // in MSConfig. They are made global so that each page can potential make
  28. // calls into other pages, to allow interaction.
  29. //-----------------------------------------------------------------------------
  30. CPageServices * ppageServices = NULL;
  31. CPageStartup * ppageStartup = NULL;
  32. CPageBootIni * ppageBootIni = NULL;
  33. CPageIni * ppageWinIni = NULL;
  34. CPageIni * ppageSystemIni = NULL;
  35. CPageGeneral * ppageGeneral = NULL;
  36. //-----------------------------------------------------------------------------
  37. // Other globals.
  38. //-----------------------------------------------------------------------------
  39. CMSConfigSheet * pMSConfigSheet = NULL; // global pointer to the property sheet
  40. /////////////////////////////////////////////////////////////////////////////
  41. // CMSConfigApp
  42. BEGIN_MESSAGE_MAP(CMSConfigApp, CWinApp)
  43. //{{AFX_MSG_MAP(CMSConfigApp)
  44. // NOTE - the ClassWizard will add and remove mapping macros here.
  45. // DO NOT EDIT what you see in these blocks of generated code!
  46. //}}AFX_MSG
  47. ON_COMMAND(ID_HELP, CWinApp::OnHelp)
  48. END_MESSAGE_MAP()
  49. //-----------------------------------------------------------------------------
  50. // Constructor. Nothing important here.
  51. //-----------------------------------------------------------------------------
  52. CMSConfigApp::CMSConfigApp()
  53. {
  54. }
  55. CMSConfigApp theApp;
  56. //-----------------------------------------------------------------------------
  57. // InitInstance is where we create the property sheet and show it (assuming
  58. // there isn't a command line flag to do otherwise).
  59. //-----------------------------------------------------------------------------
  60. BOOL fBasicControls = FALSE; // hide any advanced controls if true
  61. BOOL CMSConfigApp::InitInstance()
  62. {
  63. if (!InitATL())
  64. return FALSE;
  65. AfxEnableControlContainer();
  66. CCommandLineInfo cmdInfo;
  67. ParseCommandLine(cmdInfo);
  68. if (cmdInfo.m_bRunEmbedded || cmdInfo.m_bRunAutomated)
  69. {
  70. return TRUE;
  71. }
  72. // If this is not the first instance, exit. The call to FirstInstance
  73. // will activate the previous instance.
  74. if (!FirstInstance())
  75. return FALSE;
  76. // Standard initialization
  77. // If you are not using these features and wish to reduce the size
  78. // of your final executable, you should remove from the following
  79. // the specific initialization routines you do not need.
  80. #ifdef _AFXDLL
  81. Enable3dControls(); // Call this when using MFC in a shared DLL
  82. #else
  83. Enable3dControlsStatic(); // Call this when linking to MFC statically
  84. #endif
  85. // Process the command line to see if one of the following flags have been set:
  86. //
  87. // /n (where n is a number) startup showing the nth tab
  88. // /basic hide advanced features
  89. // /commit n make changes from tab number n permanent
  90. // /auto show the automatic launch dialog
  91. int nInitialTab = 0;
  92. int nCommitTab = 0;
  93. BOOL fShowAutoDialog = FALSE;
  94. CString strCommandLine(m_lpCmdLine);
  95. CString strFlag, strTemp;
  96. strCommandLine.MakeLower();
  97. while (!strCommandLine.IsEmpty())
  98. {
  99. // Get the next flag from the command line (starting at a / or -,
  100. // and containing the text to the end of the string or the next
  101. // instance of a / or -).
  102. int iFlag = strCommandLine.FindOneOf(_T("/-"));
  103. if (iFlag == -1)
  104. break;
  105. strFlag = strCommandLine.Mid(iFlag + 1);
  106. strFlag = strFlag.SpanExcluding(_T("/-"));
  107. strCommandLine = strCommandLine.Mid(iFlag + 1 + strFlag.GetLength());
  108. strFlag.TrimRight();
  109. // Check for the /auto flag.
  110. if (strFlag.Find(_T("auto")) == 0)
  111. fShowAutoDialog = TRUE;
  112. // Check for the "/basic" flag.
  113. if (strFlag.Compare(_T("basic")) == 0)
  114. {
  115. fBasicControls = TRUE;
  116. continue;
  117. }
  118. // Check for the "/commit n" flag.
  119. if (strFlag.Left(6) == CString(_T("commit")))
  120. {
  121. // Find out which tab number to commit. Skip all of the
  122. // non-numeric characters.
  123. strTemp = strFlag.SpanExcluding(_T("0123456789"));
  124. if (strTemp.GetLength() < strFlag.GetLength())
  125. {
  126. strFlag = strFlag.Mid(strTemp.GetLength());
  127. if (!strFlag.IsEmpty())
  128. {
  129. TCHAR c = strFlag[0];
  130. if (_istdigit(c))
  131. nCommitTab = _ttoi((LPCTSTR)strFlag);
  132. }
  133. }
  134. continue;
  135. }
  136. // Finally, check for the "/n" flag, where n is the number of
  137. // the tab to initially display.
  138. if (strFlag.GetLength() == 1)
  139. {
  140. TCHAR c = strFlag[0];
  141. if (_istdigit(c))
  142. nInitialTab = _ttoi((LPCTSTR)strFlag);
  143. }
  144. }
  145. // Show the automatic launch dialog. The user may make settings in this
  146. // dialog that will keep MSConfig from automatically launching.
  147. if (fShowAutoDialog)
  148. {
  149. CAutoStartDlg dlg;
  150. dlg.DoModal();
  151. if (dlg.m_checkDontShow)
  152. {
  153. SetAutoRun(FALSE);
  154. return FALSE;
  155. }
  156. }
  157. // Check to see if the user is going to be able to make changes using MSConfig
  158. // (if not an admin, probably not). If the user doesn't have the necessary
  159. // privileges, don't run. Bug 475796.
  160. BOOL fModifyServices = FALSE, fModifyRegistry = FALSE;
  161. // Check to see if the user will be able to modify services.
  162. SC_HANDLE sch = ::OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
  163. if (sch != NULL)
  164. {
  165. fModifyServices = TRUE;
  166. ::CloseServiceHandle(sch);
  167. }
  168. // Check to see if the user can modify the registry.
  169. HKEY hkey;
  170. if (ERROR_SUCCESS == ::RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Microsoft\\Shared Tools"), 0, KEY_WRITE, &hkey))
  171. {
  172. fModifyRegistry = TRUE;
  173. ::RegCloseKey(hkey);
  174. }
  175. // If the user can't do both actions, exit now.
  176. if (!fModifyServices || !fModifyRegistry)
  177. {
  178. CString strText, strCaption;
  179. strCaption.LoadString(IDS_DIALOGCAPTION);
  180. strText.LoadString(IDS_SERVICEACCESSDENIED);
  181. ::MessageBox(NULL, strText, strCaption, MB_OK | MB_ICONSTOP);
  182. return FALSE;
  183. }
  184. // This will load all the pages.
  185. BOOL fNeedsReboot = FALSE;
  186. InitializePages();
  187. // If the command line specifies to commit a change, we won't
  188. // show the dialog.
  189. if (nCommitTab > 1) // ignore zero (no flag) and one (general tab)
  190. {
  191. CPageBase * pPage = NULL;
  192. CString strTabCaption;
  193. if (NULL == ppageBootIni && nCommitTab >= 4)
  194. nCommitTab += 1; // adjust tab number if there is no BOOT.INI tab
  195. switch (nCommitTab)
  196. {
  197. case 2:
  198. pPage = dynamic_cast<CPageBase *>(ppageSystemIni);
  199. strTabCaption.LoadString(IDS_SYSTEMINI_CAPTION);
  200. break;
  201. case 3:
  202. pPage = dynamic_cast<CPageBase *>(ppageWinIni);
  203. strTabCaption.LoadString(IDS_WININI_CAPTION);
  204. break;
  205. case 4:
  206. pPage = dynamic_cast<CPageBase *>(ppageBootIni);
  207. strTabCaption.LoadString(IDS_BOOTINI_CAPTION);
  208. break;
  209. case 5:
  210. pPage = dynamic_cast<CPageBase *>(ppageServices);
  211. strTabCaption.LoadString(IDS_SERVICES_CAPTION);
  212. break;
  213. case 6:
  214. pPage = dynamic_cast<CPageBase *>(ppageStartup);
  215. strTabCaption.LoadString(IDS_STARTUP_CAPTION);
  216. break;
  217. }
  218. if (pPage)
  219. {
  220. CString strText, strCaption;
  221. strCaption.LoadString(IDS_DIALOGCAPTION);
  222. strText.Format(IDS_COMMITMESSAGE, strTabCaption);
  223. if (IDYES == ::MessageBox(NULL, strText, strCaption, MB_YESNO))
  224. pPage->CommitChanges();
  225. }
  226. }
  227. else
  228. fNeedsReboot = ShowPropertySheet(nInitialTab);
  229. CleanupPages();
  230. if (fNeedsReboot)
  231. Reboot();
  232. // Since the dialog has been closed, return FALSE so that we exit the
  233. // application, rather than start the application's message pump.
  234. return FALSE;
  235. }
  236. //-----------------------------------------------------------------------------
  237. // Create all of the property pages. This function also contains the logic to
  238. // exclude property pages under certain circumstances (for example, if there
  239. // is no BOOT.INI file, don't create that page). TBD.
  240. //-----------------------------------------------------------------------------
  241. void CMSConfigApp::InitializePages()
  242. {
  243. // The boot.ini tab shouldn't be added if the file doesn't exist (for
  244. // instance, on Win64).
  245. CString strBootINI(_T("c:\\boot.ini"));
  246. // Check the registry for a testing flag (which would mean we aren't
  247. // operating on the real BOOT.INI file).
  248. CRegKey regkey;
  249. if (ERROR_SUCCESS == regkey.Open(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Microsoft\\Shared Tools\\MSConfig")))
  250. {
  251. TCHAR szBoot[MAX_PATH];
  252. DWORD dwCount = MAX_PATH;
  253. if (ERROR_SUCCESS == regkey.QueryValue(szBoot, _T("boot.ini"), &dwCount))
  254. strBootINI = szBoot;
  255. }
  256. if (FileExists(strBootINI))
  257. ppageBootIni = new CPageBootIni;
  258. else
  259. ppageBootIni = NULL;
  260. ppageServices = new CPageServices;
  261. ppageStartup = new CPageStartup;
  262. ppageWinIni = new CPageIni;
  263. ppageSystemIni = new CPageIni;
  264. ppageGeneral = new CPageGeneral;
  265. ppageWinIni->SetTabInfo(_T("win.ini"));
  266. ppageSystemIni->SetTabInfo(_T("system.ini"));
  267. }
  268. //-----------------------------------------------------------------------------
  269. // Show the MSConfig property sheet. This function returns whether or not
  270. // the computer should be rebooted.
  271. //-----------------------------------------------------------------------------
  272. BOOL CMSConfigApp::ShowPropertySheet(int nInitialTab)
  273. {
  274. CMSConfigSheet sheet(IDS_DIALOGCAPTION, NULL, (nInitialTab > 0) ? nInitialTab - 1 : 0);
  275. // Add each of the pages to the property sheet.
  276. if (ppageGeneral) sheet.AddPage(ppageGeneral);
  277. if (ppageSystemIni) sheet.AddPage(ppageSystemIni);
  278. if (ppageWinIni) sheet.AddPage(ppageWinIni);
  279. if (ppageBootIni) sheet.AddPage(ppageBootIni);
  280. if (ppageServices) sheet.AddPage(ppageServices);
  281. if (ppageStartup) sheet.AddPage(ppageStartup);
  282. // Show the property sheet.
  283. pMSConfigSheet = &sheet;
  284. INT_PTR iReturn = sheet.DoModal();
  285. pMSConfigSheet = NULL;
  286. // Possibly set MSConfig to automatically run on boot, and
  287. // check to see if we need to restart.
  288. BOOL fRunMSConfigOnBoot = FALSE;
  289. BOOL fNeedToRestart = FALSE;
  290. CPageBase * apPages[5] =
  291. {
  292. ppageSystemIni,
  293. ppageWinIni,
  294. ppageBootIni,
  295. ppageServices,
  296. ppageStartup
  297. };
  298. for (int nPage = 0; nPage < 5; nPage++)
  299. if (apPages[nPage])
  300. {
  301. fRunMSConfigOnBoot |= (CPageBase::NORMAL != apPages[nPage]->GetAppliedTabState());
  302. fNeedToRestart |= apPages[nPage]->HasAppliedChanges();
  303. if (fRunMSConfigOnBoot && fNeedToRestart)
  304. break;
  305. }
  306. // If the user didn't click CANCEL, or the user applied a change, then
  307. // we should set whether or not MSConfig needs to automatically run on boot.
  308. if (fNeedToRestart || iReturn != IDCANCEL)
  309. SetAutoRun(fRunMSConfigOnBoot);
  310. return (fNeedToRestart);
  311. }
  312. //-----------------------------------------------------------------------------
  313. // Cleanup the global property pages.
  314. //-----------------------------------------------------------------------------
  315. void CMSConfigApp::CleanupPages()
  316. {
  317. if (ppageGeneral) delete ppageGeneral;
  318. if (ppageSystemIni) delete ppageSystemIni;
  319. if (ppageWinIni) delete ppageWinIni;
  320. if (ppageBootIni) delete ppageBootIni;
  321. if (ppageServices) delete ppageServices;
  322. if (ppageStartup) delete ppageStartup;
  323. }
  324. //-------------------------------------------------------------------------
  325. // This function will set the appropriate registry key to make msconfig run
  326. // on system start.
  327. //-------------------------------------------------------------------------
  328. void CMSConfigApp::SetAutoRun(BOOL fAutoRun)
  329. {
  330. LPCTSTR szRegKey = _T("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run");
  331. LPCTSTR szRegVal = _T("MSConfig");
  332. CRegKey regkey;
  333. if (ERROR_SUCCESS != regkey.Open(HKEY_LOCAL_MACHINE, szRegKey))
  334. return;
  335. if (fAutoRun)
  336. {
  337. TCHAR szModulePath[MAX_PATH];
  338. if (::GetModuleFileName(::GetModuleHandle(NULL), szModulePath, MAX_PATH) == 0)
  339. return;
  340. if (!FileExists(szModulePath))
  341. return;
  342. CString strNewVal = CString(szModulePath) + CString(_T(" ")) + CString(COMMANDLINE_AUTO);
  343. regkey.SetValue(strNewVal, szRegVal);
  344. }
  345. else
  346. regkey.DeleteValue(szRegVal);
  347. }
  348. //-----------------------------------------------------------------------------
  349. // Not much explanation needed here. The user is given an option to not
  350. // restart the system.
  351. //-----------------------------------------------------------------------------
  352. void CMSConfigApp::Reboot()
  353. {
  354. HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, GetCurrentProcessId());
  355. if (hProcess)
  356. {
  357. HANDLE hToken;
  358. if (OpenProcessToken(hProcess, TOKEN_ADJUST_PRIVILEGES, &hToken))
  359. {
  360. TOKEN_PRIVILEGES tp;
  361. tp.PrivilegeCount = 1;
  362. tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
  363. if (LookupPrivilegeValue(NULL, SE_SHUTDOWN_NAME, &tp.Privileges[0].Luid) && AdjustTokenPrivileges(hToken, FALSE, &tp, NULL, NULL, NULL))
  364. {
  365. CRebootDlg dlg;
  366. if (dlg.DoModal() == IDOK)
  367. ExitWindowsEx(EWX_REBOOT, 0);
  368. }
  369. else
  370. Message(IDS_USERSHOULDRESTART);
  371. }
  372. CloseHandle(hProcess);
  373. }
  374. }
  375. CMSConfigModule _Module;
  376. BEGIN_OBJECT_MAP(ObjectMap)
  377. END_OBJECT_MAP()
  378. LONG CMSConfigModule::Unlock()
  379. {
  380. AfxOleUnlockApp();
  381. return 0;
  382. }
  383. LONG CMSConfigModule::Lock()
  384. {
  385. AfxOleLockApp();
  386. return 1;
  387. }
  388. LPCTSTR CMSConfigModule::FindOneOf(LPCTSTR p1, LPCTSTR p2)
  389. {
  390. while (*p1 != NULL)
  391. {
  392. LPCTSTR p = p2;
  393. while (*p != NULL)
  394. {
  395. if (*p1 == *p)
  396. return CharNext(p1);
  397. p = CharNext(p);
  398. }
  399. p1++;
  400. }
  401. return NULL;
  402. }
  403. int CMSConfigApp::ExitInstance()
  404. {
  405. if (m_bATLInited)
  406. {
  407. _Module.RevokeClassObjects();
  408. _Module.Term();
  409. CoUninitialize();
  410. }
  411. return CWinApp::ExitInstance();
  412. }
  413. BOOL CMSConfigApp::InitATL()
  414. {
  415. m_bATLInited = TRUE;
  416. #if _WIN32_WINNT >= 0x0400
  417. HRESULT hRes = CoInitializeEx(NULL, COINIT_MULTITHREADED);
  418. #else
  419. HRESULT hRes = CoInitialize(NULL);
  420. #endif
  421. if (FAILED(hRes))
  422. {
  423. m_bATLInited = FALSE;
  424. return FALSE;
  425. }
  426. _Module.Init(ObjectMap, AfxGetInstanceHandle());
  427. _Module.dwThreadID = GetCurrentThreadId();
  428. LPTSTR lpCmdLine = GetCommandLine(); //this line necessary for _ATL_MIN_CRT
  429. TCHAR szTokens[] = _T("-/");
  430. BOOL bRun = TRUE;
  431. LPCTSTR lpszToken = _Module.FindOneOf(lpCmdLine, szTokens);
  432. while (lpszToken != NULL)
  433. {
  434. if (lstrcmpi(lpszToken, _T("UnregServer"))==0)
  435. {
  436. _Module.UpdateRegistryFromResource(IDR_MSCONFIG, FALSE);
  437. _Module.UnregisterServer(TRUE); //TRUE means typelib is unreg'd
  438. bRun = FALSE;
  439. break;
  440. }
  441. if (lstrcmpi(lpszToken, _T("RegServer"))==0)
  442. {
  443. _Module.UpdateRegistryFromResource(IDR_MSCONFIG, TRUE);
  444. _Module.RegisterServer(TRUE);
  445. bRun = FALSE;
  446. break;
  447. }
  448. lpszToken = _Module.FindOneOf(lpszToken, szTokens);
  449. }
  450. if (!bRun)
  451. {
  452. m_bATLInited = FALSE;
  453. _Module.Term();
  454. CoUninitialize();
  455. return FALSE;
  456. }
  457. hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
  458. REGCLS_MULTIPLEUSE);
  459. if (FAILED(hRes))
  460. {
  461. m_bATLInited = FALSE;
  462. CoUninitialize();
  463. return FALSE;
  464. }
  465. return TRUE;
  466. }
  467. //=============================================================================
  468. // Implement the utility functions described in msconfigstate.h
  469. //=============================================================================
  470. void Message(LPCTSTR szMessage, HWND hwndParent)
  471. {
  472. CString strCaption;
  473. strCaption.LoadString(IDS_APPCAPTION);
  474. if (hwndParent != NULL || pMSConfigSheet == NULL)
  475. ::MessageBox(hwndParent, szMessage, strCaption, MB_OK);
  476. else
  477. ::MessageBox(pMSConfigSheet->GetSafeHwnd(), szMessage, strCaption, MB_OK);
  478. }
  479. void Message(UINT uiMessage, HWND hwndParent)
  480. {
  481. CString strMessage;
  482. strMessage.LoadString(uiMessage);
  483. Message((LPCTSTR)strMessage, hwndParent);
  484. }
  485. HKEY GetRegKey(LPCTSTR szSubKey)
  486. {
  487. LPCTSTR szMSConfigKey = _T("SOFTWARE\\Microsoft\\Shared Tools\\MSConfig");
  488. CString strKey(szMSConfigKey);
  489. HKEY hkey = NULL;
  490. // Try to open the base MSConfig key. If it succeeds, and there is no
  491. // subkey to open, return the HKEY. Otherwise, we need to create the
  492. // base key.
  493. if (ERROR_SUCCESS == ::RegOpenKeyEx(HKEY_LOCAL_MACHINE, szMSConfigKey, 0, KEY_ALL_ACCESS, &hkey))
  494. {
  495. if (szSubKey == NULL)
  496. return hkey;
  497. ::RegCloseKey(hkey);
  498. }
  499. else
  500. {
  501. // Create the MSConfig key (and close it).
  502. HKEY hkeyBase;
  503. if (ERROR_SUCCESS == ::RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\Microsoft\\Shared Tools"), 0, KEY_ALL_ACCESS, &hkeyBase))
  504. {
  505. if (ERROR_SUCCESS == RegCreateKeyEx(hkeyBase, _T("MSConfig"), 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, NULL))
  506. ::RegCloseKey(hkey);
  507. ::RegCloseKey(hkeyBase);
  508. }
  509. }
  510. if (szSubKey)
  511. strKey += CString(_T("\\")) + CString(szSubKey);
  512. if (ERROR_SUCCESS != ::RegOpenKeyEx(HKEY_LOCAL_MACHINE, strKey, 0, KEY_ALL_ACCESS, &hkey))
  513. {
  514. // If we couldn't open the subkey, then we should try to create it.
  515. if (szSubKey)
  516. {
  517. HKEY hkeyBase;
  518. if (ERROR_SUCCESS == ::RegOpenKeyEx(HKEY_LOCAL_MACHINE, szMSConfigKey, 0, KEY_ALL_ACCESS, &hkeyBase))
  519. {
  520. if (ERROR_SUCCESS != RegCreateKeyEx(hkeyBase, szSubKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, NULL))
  521. hkey = NULL;
  522. ::RegCloseKey(hkeyBase);
  523. }
  524. }
  525. }
  526. return hkey;
  527. }
  528. HRESULT BackupFile(LPCTSTR szFilename, const CString & strAddedExtension, BOOL fOverwrite)
  529. {
  530. CString strFrom(szFilename);
  531. CString strTo(GetBackupName(szFilename, strAddedExtension));
  532. if (!fOverwrite && FileExists(strTo))
  533. return S_FALSE;
  534. ::SetFileAttributes(strTo, FILE_ATTRIBUTE_NORMAL);
  535. if (::CopyFile(strFrom, strTo, FALSE))
  536. {
  537. ::SetFileAttributes(strTo, FILE_ATTRIBUTE_NORMAL);
  538. return S_OK;
  539. }
  540. return E_FAIL;
  541. }
  542. CString strBackupDir; // global string containing the path of the backup directory
  543. const CString GetBackupName(LPCTSTR szFilename, const CString & strAddedExtension)
  544. {
  545. // There should be a directory for MSConfig files. Make sure it exists
  546. // (create it if it doesn't).
  547. if (strBackupDir.IsEmpty())
  548. {
  549. TCHAR szMSConfigDir[MAX_PATH];
  550. if (MAX_PATH > ::ExpandEnvironmentStrings(MSCONFIGDIR, szMSConfigDir, MAX_PATH))
  551. {
  552. strBackupDir = szMSConfigDir;
  553. if (!FileExists(strBackupDir))
  554. ::CreateDirectory(strBackupDir, NULL);
  555. }
  556. }
  557. CString strFrom(szFilename);
  558. int i = strFrom.ReverseFind(_T('\\'));
  559. CString strFile(strFrom.Mid(i + 1));
  560. CString strTo(strBackupDir + _T("\\") + strFile + strAddedExtension);
  561. return strTo;
  562. }
  563. HRESULT RestoreFile(LPCTSTR szFilename, const CString & strAddedExtension, BOOL fOverwrite)
  564. {
  565. CString strTo(szFilename);
  566. int i = strTo.ReverseFind(_T('\\'));
  567. CString strFile(strTo.Mid(i + 1));
  568. CString strFrom(strBackupDir + _T("\\") + strFile + strAddedExtension);
  569. if (!fOverwrite && FileExists(strTo))
  570. return S_FALSE;
  571. DWORD dwAttributes = ::GetFileAttributes(strTo);
  572. ::SetFileAttributes(strTo, FILE_ATTRIBUTE_NORMAL);
  573. if (::CopyFile(strFrom, strTo, FALSE))
  574. {
  575. ::SetFileAttributes(strTo, dwAttributes);
  576. return S_OK;
  577. }
  578. return E_FAIL;
  579. }