Leaked source code of windows server 2003
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.

786 lines
26 KiB

  1. ///////////////////////////////////////////////////////////////////////////////
  2. /* File: policy.cpp
  3. Description: Handles disk quota policy issues for both a GPE client
  4. extension and a server MMC policy snapin (see snapin.cpp).
  5. ProgressGroupPolicy is called by winlogon to process disk quota policy
  6. on the client machine. ProcessGroupPolicy instantiates a CDiskQuotaPolicy
  7. object to handle the loading and application of disk quota policy.
  8. The CDiskQuotaPolicy object is also instantiated by the MMC
  9. disk quota policy snapin to save quota policy information to the
  10. registry.
  11. A good deal of this module, especially in CDiskQuotaPolicy::Apply( ),
  12. is devoted to reporting errors to the NT event log. This is necessary
  13. because much of this code runs without UI from within winlogon. That's
  14. also why there's a lot of debugger spew.
  15. Revision History:
  16. Date Description Programmer
  17. -------- --------------------------------------------------- ----------
  18. 02/14/98 Initial creation. BrianAu
  19. 11/24/98 Added event logging settings to policy. BrianAu
  20. 11/30/98 Replaced ProcessGPO function with BrianAu
  21. ProcessGroupPolicy to support GPO interface changes.
  22. */
  23. ///////////////////////////////////////////////////////////////////////////////
  24. #include "pch.h"
  25. #pragma hdrstop
  26. #include <userenv.h>
  27. #include <gpedit.h>
  28. #include <dskquota.h>
  29. #include "policy.h"
  30. #include "registry.h"
  31. #include "guidsp.h"
  32. #include "msg.h"
  33. #include "resource.h"
  34. //
  35. // Global NT event log object.
  36. //
  37. CEventLog g_Log;
  38. //
  39. // Name of the disk quota dll.
  40. //
  41. const TCHAR g_szDskquotaDll[] = TEXT("dskquota.dll");
  42. //
  43. // Exported function called by winlogon to update policy on the client machine.
  44. //
  45. // This function is registered as a GPO extension (see selfreg.inf).
  46. //
  47. DWORD
  48. ProcessGroupPolicy(
  49. DWORD dwFlags,
  50. HANDLE hUserToken,
  51. HKEY hkeyRoot,
  52. PGROUP_POLICY_OBJECT pDeletedGPOList,
  53. PGROUP_POLICY_OBJECT pChangedGPOList,
  54. ASYNCCOMPLETIONHANDLE pHandle,
  55. BOOL *pbAbort,
  56. PFNSTATUSMESSAGECALLBACK pStatusCallback
  57. )
  58. {
  59. HRESULT hr = ERROR_SUCCESS;
  60. DBGTRACE((DM_POLICY, DL_HIGH, TEXT("ProcessGroupPolicy")));
  61. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\tdwFlags......: 0x%08X"), dwFlags));
  62. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\thUserToken...: 0x%08X"), hUserToken));
  63. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\thKeyRoot.....: 0x%08X"), hkeyRoot));
  64. //
  65. // FEATURE: Need to add support for pDeletedGPOList
  66. // If pDeletedGPOList is non-null, you should
  67. // reset the disk quotas back to their defaults first
  68. // and then apply the new settings below if appropriate
  69. //
  70. if (pChangedGPOList)
  71. {
  72. hr = g_Log.Initialize(TEXT("DiskQuota"));
  73. if (FAILED(hr))
  74. {
  75. DBGERROR((TEXT("Error 0x%08X initializing NT event log."), hr));
  76. //
  77. // Continue without event log.
  78. //
  79. }
  80. //
  81. // Only process policy info when...
  82. //
  83. // 1. Not deleting policy.
  84. //
  85. try
  86. {
  87. DBGPRINT((DM_POLICY, DL_HIGH, TEXT("Set quota policy - START.")));
  88. autoptr<CDiskQuotaPolicy> ptrPolicy(new CDiskQuotaPolicy(NULL,
  89. hkeyRoot,
  90. 0 != (GPO_INFO_FLAG_VERBOSE & dwFlags),
  91. pbAbort));
  92. DISKQUOTAPOLICYINFO dqpi;
  93. ZeroMemory(&dqpi, sizeof(dqpi));
  94. //
  95. // Load policy info from the registry and apply to local volumes.
  96. //
  97. hr = ptrPolicy->Load(&dqpi);
  98. if (SUCCEEDED(hr))
  99. {
  100. hr = ptrPolicy->Apply(&dqpi);
  101. }
  102. DBGPRINT((DM_POLICY, DL_HIGH, TEXT("Set quota policy - FINISHED.")));
  103. }
  104. catch(CAllocException& e)
  105. {
  106. DBGERROR((TEXT("Insufficient memory in ProcessGroupPolicy")));
  107. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE,
  108. 0,
  109. MSG_E_POLICY_OUTOFMEMORY);
  110. hr = E_OUTOFMEMORY;
  111. }
  112. }
  113. return hr;
  114. }
  115. //-----------------------------------------------------------------------------
  116. // CDiskQuotaPolicy
  117. //-----------------------------------------------------------------------------
  118. //
  119. // Location of disk quota policy information in the registry. The "PolicyData"
  120. // value name is somewhat arbitrary. The policy key name string however must
  121. // coordinate with other system policy locations in the registry. It should
  122. // not change unless you have a good reason to do so and you understand the
  123. // consequences.
  124. //
  125. const TCHAR CDiskQuotaPolicy::m_szRegKeyPolicy[] = REGSTR_KEY_POLICYDATA;
  126. CDiskQuotaPolicy::CDiskQuotaPolicy(
  127. LPGPEINFORMATION pGPEInfo,
  128. HKEY hkeyRoot,
  129. bool bVerboseEventLog,
  130. BOOL *pbAbort
  131. ) : m_cRef(0),
  132. m_pGPEInfo(pGPEInfo),
  133. m_hkeyRoot(hkeyRoot),
  134. m_pbAbort(pbAbort),
  135. m_bRootKeyOpened(false),
  136. m_bVerboseEventLog(bVerboseEventLog)
  137. {
  138. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::CDiskQuotaPolicy")));
  139. }
  140. CDiskQuotaPolicy::~CDiskQuotaPolicy(
  141. void
  142. )
  143. {
  144. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::~CDiskQuotaPolicy")));
  145. if (NULL != m_hkeyRoot && m_bRootKeyOpened)
  146. RegCloseKey(m_hkeyRoot);
  147. if (NULL != m_pGPEInfo)
  148. m_pGPEInfo->Release();
  149. }
  150. HRESULT
  151. CDiskQuotaPolicy::QueryInterface(
  152. REFIID riid,
  153. LPVOID *ppvOut
  154. )
  155. {
  156. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::~QueryInterface")));
  157. HRESULT hr = E_NOINTERFACE;
  158. *ppvOut = NULL;
  159. if (IID_IUnknown == riid || IID_IDiskQuotaPolicy == riid)
  160. {
  161. *ppvOut = this;
  162. ((LPUNKNOWN)*ppvOut)->AddRef();
  163. hr = NOERROR;
  164. }
  165. return hr;
  166. }
  167. ULONG
  168. CDiskQuotaPolicy::AddRef(
  169. void
  170. )
  171. {
  172. DBGTRACE((DM_POLICY, DL_LOW, TEXT("CDiskQuotaPolicy::AddRef")));
  173. return InterlockedIncrement(&m_cRef);
  174. }
  175. ULONG
  176. CDiskQuotaPolicy::Release(
  177. void
  178. )
  179. {
  180. DBGTRACE((DM_POLICY, DL_LOW, TEXT("CDiskQuotaPolicy::Release")));
  181. ASSERT( 0 != m_cRef );
  182. ULONG cRef = InterlockedDecrement(&m_cRef);
  183. if ( 0 == cRef )
  184. {
  185. delete this;
  186. }
  187. return cRef;
  188. }
  189. //
  190. // Caller can init with either:
  191. //
  192. // 1. Ptr to IGPEInformation interface. The snap in should initialize this
  193. // way since it has a pointer to an IGPEInformation
  194. // interface (LPGPEINFORMATION).
  195. //
  196. // 2. HKEY retrieved from IGPEInformation interface or from Group Policy
  197. // notification. ProcessGroupPolicy should initialize this way since it is given
  198. // the root key from winlogon.
  199. //
  200. // Can also init with both but pGPEInfo will be ignored if hkeyRoot is provided.
  201. //
  202. HRESULT
  203. CDiskQuotaPolicy::Initialize(
  204. LPGPEINFORMATION pGPEInfo,
  205. HKEY hkeyRoot
  206. )
  207. {
  208. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::Initialize")));
  209. if (NULL != m_pGPEInfo || NULL != m_hkeyRoot)
  210. return S_FALSE; // Already intialized
  211. m_hkeyRoot = hkeyRoot;
  212. m_pGPEInfo = pGPEInfo;
  213. if (NULL != m_pGPEInfo)
  214. m_pGPEInfo->AddRef();
  215. return S_OK;
  216. }
  217. //
  218. // Fill in a DISKQUOTAPOLICYINFO structure with default data.
  219. //
  220. void
  221. CDiskQuotaPolicy::InitPolicyInfo(
  222. LPDISKQUOTAPOLICYINFO pInfo
  223. )
  224. {
  225. pInfo->llDefaultQuotaThreshold = (LONGLONG)-1; // No limit.
  226. pInfo->llDefaultQuotaLimit = (LONGLONG)-1; // No limit.
  227. pInfo->dwQuotaState = 0;
  228. pInfo->dwQuotaLogFlags = 0;
  229. pInfo->bRemovableMedia = 0;
  230. }
  231. //
  232. // Initialize and load policy information into a DISKQUOTAPOLICYINFO structure.
  233. // If reg values don't exist, default values are used.
  234. //
  235. void
  236. CDiskQuotaPolicy::LoadPolicyInfo(
  237. const RegKey& key,
  238. LPDISKQUOTAPOLICYINFO pInfo
  239. )
  240. {
  241. DWORD dwValue = DWORD(-1);
  242. const struct
  243. {
  244. LPCTSTR pszValue; // Name of the "value" reg value.
  245. LPCTSTR pszUnits; // Name of the "units" reg value.
  246. LONGLONG *pValue; // Address of destination for computed limit or threshold.
  247. } rgValUnits[] = {
  248. { REGSTR_VAL_POLICY_LIMIT, REGSTR_VAL_POLICY_LIMITUNITS, &(pInfo->llDefaultQuotaLimit) },
  249. { REGSTR_VAL_POLICY_THRESHOLD, REGSTR_VAL_POLICY_THRESHOLDUNITS, &(pInfo->llDefaultQuotaThreshold) }
  250. };
  251. //
  252. // Initialize with defaults.
  253. //
  254. InitPolicyInfo(pInfo);
  255. //
  256. // Load the limit and threshold values along with their respective "units"
  257. // factor. The factor is a number [1..6] that represents the required
  258. // multiplier to convert the "value" to a byte value.
  259. //
  260. // 1 = KB, 2 = MB, 3 = GB, 4 = TB, 5 = PB, 6 = EB
  261. //
  262. // Bytes = value << (factor * 10).
  263. //
  264. // Given: value = 250
  265. // factor = 2 (MB)
  266. //
  267. // Bytes = 250 << 20
  268. // = 262,144,000
  269. // = 250 MB
  270. //
  271. for (int i = 0; i < ARRAYSIZE(rgValUnits); i++)
  272. {
  273. dwValue = DWORD(-1);
  274. DWORD dwUnits = DWORD(-1);
  275. key.GetValue(rgValUnits[i].pszValue, &dwValue);
  276. key.GetValue(rgValUnits[i].pszUnits, &dwUnits);
  277. *(rgValUnits[i].pValue) = LONGLONG(-1);
  278. // must cast dwValue to "int" since DWORDs are never negative
  279. // Don't need to test dwUnits for negative since negative DWORDs
  280. // turn into huge positive values, so it will fail the "6 >= dwUnits"
  281. // test.
  282. if (0 <= (int)dwValue && 6 >= dwUnits)
  283. {
  284. *(rgValUnits[i].pValue) = LONGLONG(dwValue) << (10 * dwUnits);
  285. }
  286. }
  287. //
  288. // This logic for setting the dwQuotaState member is the same as that
  289. // used in VolumePropPage::QuotaStateFromControls (volprop.cpp).
  290. //
  291. DWORD dwEnable = 0;
  292. DWORD dwEnforce = 0;
  293. key.GetValue(REGSTR_VAL_POLICY_ENABLE, &dwEnable);
  294. key.GetValue(REGSTR_VAL_POLICY_ENFORCE, &dwEnforce);
  295. if (dwEnable)
  296. {
  297. if (dwEnforce)
  298. {
  299. pInfo->dwQuotaState = DISKQUOTA_STATE_ENFORCE;
  300. }
  301. else
  302. {
  303. pInfo->dwQuotaState = DISKQUOTA_STATE_TRACK;
  304. }
  305. }
  306. else
  307. {
  308. pInfo->dwQuotaState = DISKQUOTA_STATE_DISABLED;
  309. }
  310. //
  311. // Get event logging settings.
  312. //
  313. DWORD dwLog = 0;
  314. key.GetValue(REGSTR_VAL_POLICY_LOGLIMIT, &dwLog);
  315. DISKQUOTA_SET_LOG_USER_LIMIT(pInfo->dwQuotaLogFlags, dwLog);
  316. dwLog = 0;
  317. key.GetValue(REGSTR_VAL_POLICY_LOGTHRESHOLD, &dwLog);
  318. DISKQUOTA_SET_LOG_USER_THRESHOLD(pInfo->dwQuotaLogFlags, dwLog);
  319. //
  320. // Determine if policy is to be applied to removable as well as fixed
  321. // media.
  322. //
  323. if (SUCCEEDED(key.GetValue(REGSTR_VAL_POLICY_REMOVABLEMEDIA, &dwValue)))
  324. {
  325. pInfo->bRemovableMedia = boolify(dwValue);
  326. }
  327. }
  328. //
  329. // Load machine policy information from the registry. See comment
  330. // in CDiskQuotaPolicy::Save( ) for registry location information.
  331. //
  332. HRESULT
  333. CDiskQuotaPolicy::Load(
  334. LPDISKQUOTAPOLICYINFO pInfo
  335. )
  336. {
  337. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::Load")));
  338. DBGASSERT((NULL != pInfo));
  339. if (NULL == m_pGPEInfo && NULL == m_hkeyRoot)
  340. {
  341. DBGERROR((TEXT("Policy object not initialized")));
  342. return E_FAIL; // Not initialized.
  343. }
  344. HRESULT hr = E_FAIL;
  345. if (NULL == m_hkeyRoot &&
  346. SUCCEEDED(hr = m_pGPEInfo->GetRegistryKey(GPO_SECTION_MACHINE, &m_hkeyRoot)))
  347. {
  348. m_bRootKeyOpened = true;
  349. }
  350. if (NULL != m_hkeyRoot)
  351. {
  352. DBGPRINT((DM_POLICY, DL_LOW, TEXT("Opening reg key 0x%08X \"%s\""), m_hkeyRoot, m_szRegKeyPolicy));
  353. RegKey key(m_hkeyRoot, m_szRegKeyPolicy);
  354. hr = key.Open(KEY_READ);
  355. if (SUCCEEDED(hr))
  356. {
  357. DBGPRINT((DM_POLICY, DL_LOW, TEXT("Reading disk quota policy information.")));
  358. LoadPolicyInfo(key, pInfo);
  359. if (m_bVerboseEventLog)
  360. {
  361. //
  362. // Report successful information retrieval.
  363. //
  364. g_Log.ReportEvent(EVENTLOG_INFORMATION_TYPE, 0, MSG_I_POLICY_INFOLOADED);
  365. }
  366. }
  367. else if (ERROR_FILE_NOT_FOUND != HRESULT_CODE(hr)) // Key doesn't always exist.
  368. {
  369. DBGERROR((TEXT("Error 0x%08X opening policy reg key"), hr));
  370. g_Log.Push(hr, CEventLog::eFmtHex);
  371. g_Log.Push(m_szRegKeyPolicy),
  372. g_Log.Push(hr, CEventLog::eFmtSysErr);
  373. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_GPEREGKEYOPEN);
  374. }
  375. }
  376. else
  377. {
  378. DBGERROR((TEXT("m_hkeyRoot is NULL")));
  379. g_Log.Push(hr, CEventLog::eFmtHex);
  380. g_Log.Push(hr, CEventLog::eFmtSysErr);
  381. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_GPEREGKEYROOT);
  382. }
  383. return hr;
  384. }
  385. //
  386. // Apply policy information to all local NTFS volumes. Removable media
  387. // are optional per a value in the policy information structure.
  388. //
  389. HRESULT
  390. CDiskQuotaPolicy::Apply(
  391. LPCDISKQUOTAPOLICYINFO pInfo
  392. )
  393. {
  394. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::Apply")));
  395. DBGASSERT((NULL != pInfo));
  396. HRESULT hr = NOERROR;
  397. BOOL bAborted = m_pbAbort ? *m_pbAbort : FALSE;
  398. try
  399. {
  400. if (!bAborted)
  401. {
  402. CString strVolCompleted;
  403. //
  404. // Get list of drives on which to set policy.
  405. //
  406. CArray<CString> rgstrDrives;
  407. hr = GetDriveNames(&rgstrDrives, pInfo->bRemovableMedia);
  408. int cDrives = rgstrDrives.Count();
  409. if (SUCCEEDED(hr) && 0 < cDrives)
  410. {
  411. //
  412. // Get the disk quota class factory. This way we don't
  413. // call CoCreateInstance for each drive. Only call it once then
  414. // call the class factory's CreateInstance for each drive.
  415. // Should be more efficient.
  416. //
  417. com_autoptr<IClassFactory> pcf;
  418. hr = CoCreateInstance(CLSID_DiskQuotaControl,
  419. NULL,
  420. CLSCTX_INPROC_SERVER,
  421. IID_IClassFactory,
  422. reinterpret_cast<void **>(pcf.getaddr()));
  423. if (SUCCEEDED(hr))
  424. {
  425. bAborted = m_pbAbort ? *m_pbAbort : FALSE;
  426. for (int i = 0; i < cDrives && !bAborted; i++)
  427. {
  428. DBGPRINT((DM_POLICY, DL_MID, TEXT("Setting policy for \"%s\""), rgstrDrives[i].Cstr()));
  429. //
  430. // Get a quota control object and initialize it for drive[i].
  431. // Init with read/write access.
  432. //
  433. com_autoptr<IDiskQuotaControl> pdqc;
  434. hr = pcf->CreateInstance(NULL,
  435. IID_IDiskQuotaControl,
  436. reinterpret_cast<void **>(pdqc.getaddr()));
  437. if (SUCCEEDED(hr))
  438. {
  439. hr = pdqc->Initialize(rgstrDrives[i], TRUE);
  440. if (SUCCEEDED(hr))
  441. {
  442. //
  443. // Set the quota information on the volume.
  444. //
  445. if (FAILED(hr = pdqc->SetQuotaLogFlags(pInfo->dwQuotaLogFlags)))
  446. {
  447. DBGERROR((TEXT("Error 0x%08X setting log flags"), hr));
  448. goto setpolerr;
  449. }
  450. if (FAILED(hr = pdqc->SetDefaultQuotaThreshold(pInfo->llDefaultQuotaThreshold)))
  451. {
  452. DBGERROR((TEXT("Error 0x%08X setting default threshold"), hr));
  453. goto setpolerr;
  454. }
  455. if (FAILED(hr = pdqc->SetDefaultQuotaLimit(pInfo->llDefaultQuotaLimit)))
  456. {
  457. DBGERROR((TEXT("Error 0x%08X setting default limit"), hr));
  458. goto setpolerr;
  459. }
  460. //
  461. // Set state last in case we're enabling quotas. That way
  462. // any rebuild activity will come after the other settings have
  463. // been set.
  464. //
  465. if (FAILED(hr = pdqc->SetQuotaState(DISKQUOTA_STATE_MASK & pInfo->dwQuotaState)))
  466. {
  467. DBGERROR((TEXT("Error 0x%08X setting quota state"), hr));
  468. goto setpolerr;
  469. }
  470. goto setpolsuccess;
  471. }
  472. else
  473. DBGERROR((TEXT("Error 0x%08X initializing vol \"%s\""),
  474. hr, rgstrDrives[i].Cstr()));
  475. setpolerr:
  476. //
  477. // Record error for this particular volume.
  478. //
  479. g_Log.Push(hr, CEventLog::eFmtHex);
  480. g_Log.Push(rgstrDrives[i].Cstr());
  481. g_Log.Push(hr, CEventLog::eFmtSysErr);
  482. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_SETQUOTA);
  483. setpolsuccess:
  484. if (m_bVerboseEventLog && SUCCEEDED(hr))
  485. {
  486. //
  487. // Append name to list of drives that have been successful.
  488. //
  489. strVolCompleted += rgstrDrives[i];
  490. strVolCompleted += CString(TEXT(" "));
  491. }
  492. pdqc = NULL; // This releases pdqc.
  493. }
  494. else
  495. {
  496. DBGERROR((TEXT("CreateInstance failed with error 0x%08X"), hr));
  497. g_Log.Push(hr, CEventLog::eFmtHex);
  498. g_Log.Push(hr, CEventLog::eFmtSysErr);
  499. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_CREATEQUOTACONTROL);
  500. }
  501. }
  502. pcf = NULL; // This releases pcf.
  503. }
  504. else
  505. {
  506. DBGERROR((TEXT("CoCreateInstance failed with error 0x%08X"), hr));
  507. g_Log.Push(hr, CEventLog::eFmtHex);
  508. g_Log.Push(hr, CEventLog::eFmtSysErr);
  509. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_CREATECLASSFACTORY);
  510. }
  511. }
  512. else
  513. {
  514. DBGERROR((TEXT("Error 0x%08X getting drive name list"), hr));
  515. g_Log.Push(hr, CEventLog::eFmtHex);
  516. g_Log.Push(hr, CEventLog::eFmtSysErr);
  517. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_GETDRIVELIST);
  518. }
  519. if (m_bVerboseEventLog && 0 < strVolCompleted.Length())
  520. {
  521. //
  522. // Log successful completions by listing volumes
  523. // and applied policy values.
  524. //
  525. CString s;
  526. LONGLONG llValue;
  527. g_Log.Push(strVolCompleted);
  528. g_Log.Push(!DISKQUOTA_IS_DISABLED(pInfo->dwQuotaState));
  529. g_Log.Push(DISKQUOTA_IS_ENFORCED(pInfo->dwQuotaState));
  530. llValue = pInfo->llDefaultQuotaThreshold;
  531. if (LONGLONG(-1) != llValue)
  532. {
  533. XBytes::FormatByteCountForDisplay(llValue, s.GetBuffer(40), 40);
  534. s.ReleaseBuffer();
  535. }
  536. else
  537. {
  538. s.Format(g_hInstDll, IDS_NO_LIMIT);
  539. }
  540. g_Log.Push(s);
  541. llValue = pInfo->llDefaultQuotaLimit;
  542. if (LONGLONG(-1) != llValue)
  543. {
  544. XBytes::FormatByteCountForDisplay(llValue, s.GetBuffer(40), 40);
  545. s.ReleaseBuffer();
  546. }
  547. else
  548. {
  549. s.Format(g_hInstDll, IDS_NO_LIMIT);
  550. }
  551. g_Log.Push(s);
  552. g_Log.ReportEvent(EVENTLOG_INFORMATION_TYPE, 0, MSG_I_POLICY_FINISHED);
  553. }
  554. }
  555. }
  556. catch(CAllocException& e)
  557. {
  558. DBGERROR((TEXT("Insufficient memory.")));
  559. hr = E_OUTOFMEMORY;
  560. g_Log.ReportEvent(EVENTLOG_ERROR_TYPE, 0, MSG_E_POLICY_OUTOFMEMORY);
  561. }
  562. if (bAborted)
  563. {
  564. g_Log.ReportEvent(EVENTLOG_INFORMATION_TYPE, 0, MSG_I_POLICY_ABORTED);
  565. }
  566. return hr;
  567. }
  568. //
  569. // Build a list of drives to which policy can be applied.
  570. //
  571. HRESULT
  572. CDiskQuotaPolicy::GetDriveNames( // [ static ]
  573. CArray<CString> *prgstrDrives, // Output array of name strings.
  574. bool bRemovableMedia // Include removable media?
  575. )
  576. {
  577. DBGTRACE((DM_POLICY, DL_MID, TEXT("CDiskQuotaPolicy::GetDriveNames")));
  578. DBGASSERT((NULL != prgstrDrives));
  579. HRESULT hr = NOERROR;
  580. //
  581. // Get buffer size required to hold drive name strings.
  582. //
  583. int cch = GetLogicalDriveStrings(0, NULL);
  584. //
  585. // Allocate buffer and get the strings.
  586. //
  587. array_autoptr<TCHAR> ptrDrives(new TCHAR[cch + 1]);
  588. if (0 < GetLogicalDriveStrings(cch, ptrDrives.get()))
  589. {
  590. //
  591. // Iterate over all of the drive name strings. Append to the
  592. // string array each that can accept policy.
  593. //
  594. DblNulTermListIter iter(ptrDrives.get());
  595. LPCTSTR pszDrive;
  596. while(iter.Next(&pszDrive))
  597. {
  598. if (S_OK == OkToApplyPolicy(pszDrive, bRemovableMedia))
  599. {
  600. prgstrDrives->Append(CString(pszDrive));
  601. }
  602. }
  603. }
  604. else
  605. {
  606. DWORD dwErr = GetLastError();
  607. DBGERROR((TEXT("GetLogicalDriveStrings failed with error %d"), dwErr));
  608. hr = HRESULT_FROM_WIN32(dwErr);
  609. }
  610. return hr;
  611. }
  612. //
  613. // Returns: S_OK = OK to set policy on drive.
  614. // S_FALSE = Not OK to set policy
  615. // Other = Error occured. Not OK to set policy.
  616. //
  617. HRESULT
  618. CDiskQuotaPolicy::OkToApplyPolicy( // [ static ]
  619. LPCTSTR pszDrive, // Drive (volume) name string.
  620. bool bRemovableMedia // Include removable media?
  621. )
  622. {
  623. DBGTRACE((DM_POLICY, DL_LOW, TEXT("CDiskQuotaPolicy::OkToApplyPolicy")));
  624. HRESULT hr = S_FALSE;
  625. //
  626. // Primary filter is drive type.
  627. //
  628. UINT uDriveType = GetDriveType(pszDrive);
  629. switch(uDriveType)
  630. {
  631. case DRIVE_UNKNOWN:
  632. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" is UNKNOWN"), pszDrive));
  633. return S_FALSE;
  634. case DRIVE_NO_ROOT_DIR:
  635. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" has no root dir"), pszDrive));
  636. return S_FALSE;
  637. case DRIVE_REMOTE:
  638. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" is REMOTE"), pszDrive));
  639. return S_FALSE;
  640. case DRIVE_CDROM:
  641. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" is CDROM"), pszDrive));
  642. return S_FALSE;
  643. case DRIVE_RAMDISK:
  644. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" is RAMDISK"), pszDrive));
  645. return S_FALSE;
  646. case DRIVE_REMOVABLE:
  647. //
  648. // Removable is allowable if policy says it is. It should be
  649. // disallowed by default since using disk quota on removable media
  650. // doesn't make a lot of sense in most situations.
  651. //
  652. if (!bRemovableMedia)
  653. {
  654. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" is REMOVABLE"), pszDrive));
  655. return S_FALSE;
  656. }
  657. //
  658. // Fall through...
  659. //
  660. case DRIVE_FIXED:
  661. //
  662. // Fixed drives are always acceptable.
  663. //
  664. break;
  665. default:
  666. DBGERROR((TEXT("Unknown drive type %d for \"%s\""), uDriveType, pszDrive));
  667. return S_FALSE;
  668. }
  669. //
  670. // Next filter is support for NTFS quotas. We do the drive-type check first
  671. // because it doesn't require hitting the disks. GetVolumeInformation does
  672. // hit the disk so we only want to do it if necessary.
  673. //
  674. DWORD dwFlags = 0;
  675. if (GetVolumeInformation(pszDrive, NULL, 0, NULL, NULL, &dwFlags, NULL, 0))
  676. {
  677. if (FILE_VOLUME_QUOTAS & dwFlags)
  678. {
  679. DBGPRINT((DM_POLICY, DL_LOW, TEXT("Ok to set policy on \"%s\""), pszDrive));
  680. hr = S_OK;
  681. }
  682. else
  683. {
  684. DBGPRINT((DM_POLICY, DL_LOW, TEXT("\"%s\" doesn't support NTFS quotas"), pszDrive));
  685. }
  686. }
  687. else
  688. {
  689. DWORD dwErr = GetLastError();
  690. DBGERROR((TEXT("GetVolumeInformation failed with error %d for \"%s\""),
  691. dwErr, pszDrive));
  692. hr = HRESULT_FROM_WIN32(dwErr);
  693. }
  694. return hr;
  695. }