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.

1669 lines
47 KiB

  1. //+-------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. //
  5. // Copyright (C) Microsoft Corporation, 1996 - 1999
  6. //
  7. // File: rshx32.cpp
  8. //
  9. // Remote administration shell extension.
  10. //
  11. //--------------------------------------------------------------------------
  12. ///////////////////////////////////////////////////////////////////////////////
  13. // //
  14. // Include files //
  15. // //
  16. ///////////////////////////////////////////////////////////////////////////////
  17. #include "rshx32.h"
  18. #include <winnetwk.h> // WNetGetConnection
  19. #include <lm.h>
  20. #include <lmdfs.h> // NetDfsGetClientInfo
  21. #include <atlconv.h>
  22. #include <initguid.h>
  23. DEFINE_GUID(CLSID_NTFSSecurityExt, 0x1f2e5c40, 0x9550, 0x11ce, 0x99, 0xd2, 0x00, 0xaa, 0x00, 0x6e, 0x08, 0x6c);
  24. DEFINE_GUID(CLSID_PrintSecurityExt, 0xf37c5810, 0x4d3f, 0x11d0, 0xb4, 0xbf, 0x00, 0xaa, 0x00, 0xbb, 0xb7, 0x23);
  25. #define IID_PPV_ARG(IType, ppType) IID_##IType, reinterpret_cast<void**>(static_cast<IType**>(ppType))
  26. #define RSX_SECURITY_CHECKED 0x00000001L
  27. #define RSX_HAVE_SECURITY 0x00000002L
  28. #define DOBJ_RES_CONT 0x00000001L
  29. #define DOBJ_RES_ROOT 0x00000002L
  30. #define DOBJ_VOL_NTACLS 0x00000004L // NTFS or OFS
  31. class CRShellExtCF : public IClassFactory
  32. {
  33. protected:
  34. ULONG m_cRef;
  35. SE_OBJECT_TYPE m_seType;
  36. public:
  37. CRShellExtCF(SE_OBJECT_TYPE seType);
  38. ~CRShellExtCF();
  39. // IUnknown methods
  40. STDMETHODIMP QueryInterface(REFIID, void **);
  41. STDMETHODIMP_(ULONG) AddRef();
  42. STDMETHODIMP_(ULONG) Release();
  43. // IClassFactory methods
  44. STDMETHODIMP CreateInstance(LPUNKNOWN, REFIID, void **);
  45. STDMETHODIMP LockServer(BOOL);
  46. };
  47. class CRShellExt : public IShellExtInit, IShellPropSheetExt, IContextMenu
  48. {
  49. protected:
  50. ULONG m_cRef;
  51. SE_OBJECT_TYPE m_seType;
  52. IDataObject *m_lpdobj; // interface passed in by shell
  53. HRESULT m_hrSecurityCheck;
  54. DWORD m_dwSIFlags;
  55. LPTSTR m_pszServer;
  56. LPTSTR m_pszObject;
  57. HDPA m_hItemList;
  58. public:
  59. CRShellExt(SE_OBJECT_TYPE seType);
  60. ~CRShellExt();
  61. // IUnknown methods
  62. STDMETHODIMP QueryInterface(REFIID, void **);
  63. STDMETHODIMP_(ULONG) AddRef();
  64. STDMETHODIMP_(ULONG) Release();
  65. // IShellExtInit method
  66. STDMETHODIMP Initialize(LPCITEMIDLIST, LPDATAOBJECT, HKEY);
  67. // IShellPropSheetExt methods
  68. STDMETHODIMP AddPages(LPFNADDPROPSHEETPAGE, LPARAM);
  69. STDMETHODIMP ReplacePage(UINT, LPFNADDPROPSHEETPAGE, LPARAM);
  70. //IContextMenu methods
  71. STDMETHODIMP QueryContextMenu(HMENU hMenu,
  72. UINT indexMenu,
  73. UINT idCmdFirst,
  74. UINT idCmdLast,
  75. UINT uFlags);
  76. STDMETHODIMP InvokeCommand(LPCMINVOKECOMMANDINFO lpcmi);
  77. STDMETHODIMP GetCommandString(UINT_PTR idCmd,
  78. UINT uFlags,
  79. UINT *reserved,
  80. LPSTR pszName,
  81. UINT cchMax);
  82. private:
  83. STDMETHODIMP DoSecurityCheck(LPIDA pIDA);
  84. STDMETHODIMP CheckForSecurity(LPIDA pIDA);
  85. STDMETHODIMP CreateSI(LPSECURITYINFO *ppsi);
  86. STDMETHODIMP AddSecurityPage(LPFNADDPROPSHEETPAGE lpfnAddPage, LPARAM lParam);
  87. BOOL IsAddPrinterWizard() const;
  88. #if (_WIN32_WINNT >= 0x0500)
  89. STDMETHODIMP AddMountedVolumePage(LPFNADDPROPSHEETPAGE lpfnAddPage,
  90. LPARAM lParam);
  91. #endif
  92. };
  93. typedef CRShellExt* PRSHELLEXT;
  94. ///////////////////////////////////////////////////////////////////////////////
  95. // //
  96. // Global variables //
  97. // //
  98. ///////////////////////////////////////////////////////////////////////////////
  99. HINSTANCE g_hInstance = NULL;
  100. LONG g_cRefThisDll = 0;
  101. CLIPFORMAT g_cfShellIDList = 0;
  102. CLIPFORMAT g_cfPrinterGroup = 0;
  103. CLIPFORMAT g_cfMountedVolume = 0;
  104. HMODULE g_hAclui = NULL;
  105. ///////////////////////////////////////////////////////////////////////////////
  106. // //
  107. // Private prototypes //
  108. // //
  109. ///////////////////////////////////////////////////////////////////////////////
  110. void GetFileInfo(LPCTSTR pszPath,
  111. LPDWORD pdwFileType,
  112. LPTSTR pszServer,
  113. ULONG cchServer,
  114. LPTSTR *ppszAlternatePath);
  115. ///////////////////////////////////////////////////////////////////////////////
  116. // //
  117. // General routines //
  118. // //
  119. ///////////////////////////////////////////////////////////////////////////////
  120. /*++
  121. Routine Description:
  122. Dll's entry point.
  123. In order to service requests for file selection information from
  124. any of the file manager extensions to be included in this library,
  125. we must first register a window class to accept these requests.
  126. The Microsoft_Network provider transfers information via a private
  127. clipboard format called "Net Resource" which we must register.
  128. Arguments:
  129. Same as DllEntryPoint.
  130. Return Values:
  131. Same as DllEntryPoint.
  132. --*/
  133. STDAPI_(BOOL) DllMain(HINSTANCE hInstance, DWORD dwReason, void * /*lpReserved*/)
  134. {
  135. switch (dwReason)
  136. {
  137. case DLL_PROCESS_ATTACH:
  138. g_hInstance = hInstance;
  139. g_cfShellIDList = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_SHELLIDLIST);
  140. g_cfPrinterGroup = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_PRINTERGROUP);
  141. g_cfMountedVolume = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_MOUNTEDVOLUME);
  142. DebugProcessAttach();
  143. TraceSetMaskFromCLSID(CLSID_NTFSSecurityExt);
  144. #ifndef DEBUG
  145. DisableThreadLibraryCalls(hInstance);
  146. #endif
  147. break;
  148. case DLL_PROCESS_DETACH:
  149. if (g_hAclui)
  150. FreeLibrary(g_hAclui);
  151. DebugProcessDetach();
  152. break;
  153. case DLL_THREAD_DETACH:
  154. DebugThreadDetach();
  155. break;
  156. }
  157. return TRUE;
  158. }
  159. /*++
  160. Routine Description:
  161. Called by shell to create a class factory object.
  162. Arguments:
  163. rclsid - reference to class id specifier.
  164. riid - reference to interface id specifier.
  165. ppv - pointer to location to receive interface pointer.
  166. Return Values:
  167. Returns HRESULT signifying success or failure.
  168. --*/
  169. STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, void **ppv)
  170. {
  171. HRESULT hr;
  172. SE_OBJECT_TYPE seType;
  173. *ppv = NULL;
  174. if (IsEqualCLSID(rclsid, CLSID_NTFSSecurityExt))
  175. seType = SE_FILE_OBJECT;
  176. else if (IsEqualCLSID(rclsid, CLSID_PrintSecurityExt))
  177. seType = SE_PRINTER;
  178. else
  179. return CLASS_E_CLASSNOTAVAILABLE;
  180. CRShellExtCF *pShellExtCF = new CRShellExtCF(seType); // ref == 1
  181. if (!pShellExtCF)
  182. return E_OUTOFMEMORY;
  183. hr = pShellExtCF->QueryInterface(riid, ppv);
  184. pShellExtCF->Release(); // release initial ref
  185. return hr;
  186. }
  187. /*++
  188. Routine Description:
  189. Called by shell to find out if dll can be unloaded.
  190. Arguments:
  191. None.
  192. Return Values:
  193. Returns S_OK if dll can be unloaded, S_FALSE if not.
  194. --*/
  195. STDAPI DllCanUnloadNow()
  196. {
  197. return (g_cRefThisDll == 0 ? S_OK : S_FALSE);
  198. }
  199. STDAPI DllRegisterServer(void)
  200. {
  201. return CallRegInstall(g_hInstance, "DefaultInstall");
  202. }
  203. STDAPI DllUnregisterServer(void)
  204. {
  205. return CallRegInstall(g_hInstance, "DefaultUninstall");
  206. }
  207. ///////////////////////////////////////////////////////////////////////////////
  208. // //
  209. // Class factory object implementation //
  210. // //
  211. ///////////////////////////////////////////////////////////////////////////////
  212. CRShellExtCF::CRShellExtCF(SE_OBJECT_TYPE seType) : m_cRef(1), m_seType(seType)
  213. {
  214. InterlockedIncrement(&g_cRefThisDll);
  215. }
  216. CRShellExtCF::~CRShellExtCF()
  217. {
  218. InterlockedDecrement(&g_cRefThisDll);
  219. }
  220. ///////////////////////////////////////////////////////////////////////////////
  221. // //
  222. // Class factory object implementation (IUnknown) //
  223. // //
  224. ///////////////////////////////////////////////////////////////////////////////
  225. STDMETHODIMP_(ULONG) CRShellExtCF::AddRef()
  226. {
  227. return ++m_cRef;
  228. }
  229. STDMETHODIMP_(ULONG) CRShellExtCF::Release()
  230. {
  231. if (--m_cRef == 0)
  232. {
  233. delete this;
  234. return 0;
  235. }
  236. return m_cRef;
  237. }
  238. STDMETHODIMP CRShellExtCF::QueryInterface(REFIID riid, void ** ppv)
  239. {
  240. if (IsEqualIID(riid, IID_IUnknown) || IsEqualIID(riid, IID_IClassFactory))
  241. {
  242. *ppv = (IClassFactory *)this;
  243. m_cRef++;
  244. return S_OK;
  245. }
  246. else
  247. {
  248. *ppv = NULL;
  249. return E_NOINTERFACE;
  250. }
  251. }
  252. ///////////////////////////////////////////////////////////////////////////////
  253. // //
  254. // Class factory object implementation (IClassFactory) //
  255. // //
  256. ///////////////////////////////////////////////////////////////////////////////
  257. /*++
  258. Routine Description:
  259. Support for IClassFactory::CreateInstance.
  260. Arguments:
  261. pUnkOuter - pointer to controlling unknown.
  262. riid - reference to interface id specifier.
  263. ppvObj - pointer to location to receive interface pointer.
  264. Return Values:
  265. Returns HRESULT signifying success or failure.
  266. --*/
  267. STDMETHODIMP CRShellExtCF::CreateInstance(IUnknown *pUnkOuter, REFIID riid, void ** ppvObj)
  268. {
  269. *ppvObj = NULL;
  270. if (pUnkOuter)
  271. return CLASS_E_NOAGGREGATION;
  272. CRShellExt *pShellExt = new CRShellExt(m_seType);// ref count == 1
  273. if (!pShellExt)
  274. return E_OUTOFMEMORY;
  275. HRESULT hr = pShellExt->QueryInterface(riid, ppvObj);
  276. pShellExt->Release(); // release initial ref
  277. return hr;
  278. }
  279. /*++
  280. Routine Description:
  281. Support for IClassFactory::LockServer (not implemented).
  282. Arguments:
  283. fLock - true if lock count to be incremented.
  284. Return Values:
  285. Returns E_NOTIMPL.
  286. --*/
  287. STDMETHODIMP CRShellExtCF::LockServer(BOOL /*fLock*/)
  288. {
  289. return E_NOTIMPL;
  290. }
  291. ///////////////////////////////////////////////////////////////////////////////
  292. // //
  293. // Shell extension object implementation //
  294. // //
  295. ///////////////////////////////////////////////////////////////////////////////
  296. CRShellExt::CRShellExt(SE_OBJECT_TYPE seType) : m_cRef(1), m_seType(seType),
  297. m_dwSIFlags(SI_EDIT_ALL | SI_ADVANCED | SI_EDIT_EFFECTIVE), m_hrSecurityCheck((HRESULT)-1),
  298. m_hItemList(NULL)
  299. {
  300. InterlockedIncrement(&g_cRefThisDll);
  301. }
  302. CRShellExt::~CRShellExt()
  303. {
  304. DoRelease(m_lpdobj);
  305. LocalFreeString(&m_pszServer);
  306. LocalFreeString(&m_pszObject);
  307. LocalFreeDPA(m_hItemList);
  308. InterlockedDecrement(&g_cRefThisDll);
  309. }
  310. ///////////////////////////////////////////////////////////////////////////////
  311. // //
  312. // Shell extension object implementation (IUnknown) //
  313. // //
  314. ///////////////////////////////////////////////////////////////////////////////
  315. STDMETHODIMP_(ULONG)
  316. CRShellExt::AddRef()
  317. {
  318. return ++m_cRef;
  319. }
  320. STDMETHODIMP_(ULONG)
  321. CRShellExt::Release()
  322. {
  323. if (--m_cRef == 0)
  324. {
  325. delete this;
  326. return 0;
  327. }
  328. return m_cRef;
  329. }
  330. STDMETHODIMP CRShellExt::QueryInterface(REFIID riid, void ** ppv)
  331. {
  332. if (IsEqualIID(riid, IID_IShellExtInit) || IsEqualIID(riid, IID_IUnknown))
  333. {
  334. *ppv = (LPSHELLEXTINIT)this;
  335. m_cRef++;
  336. return S_OK;
  337. }
  338. else if (IsEqualIID(riid, IID_IContextMenu))
  339. {
  340. *ppv = (LPCONTEXTMENU)this;
  341. m_cRef++;
  342. return S_OK;
  343. }
  344. else if (IsEqualIID(riid, IID_IShellPropSheetExt))
  345. {
  346. *ppv = (LPSHELLPROPSHEETEXT)this;
  347. m_cRef++;
  348. return S_OK;
  349. }
  350. else
  351. {
  352. *ppv = NULL;
  353. return E_NOINTERFACE;
  354. }
  355. }
  356. ///////////////////////////////////////////////////////////////////////////////
  357. // //
  358. // Shell extension object implementation (IShellExtInit) //
  359. // //
  360. ///////////////////////////////////////////////////////////////////////////////
  361. /*++
  362. Routine Description:
  363. Support for IShellExtInit::Initialize.
  364. Arguments:
  365. pidlFolder - pointer to id list identifying parent folder.
  366. lpdobj - pointer to IDataObject interface for selected object(s).
  367. hKeyProgId - registry key handle.
  368. Return Values:
  369. Returns HRESULT signifying success or failure.
  370. --*/
  371. STDMETHODIMP CRShellExt::Initialize(LPCITEMIDLIST /*pidlFolder*/, IDataObject *lpdobj, HKEY /*hKeyProgID*/)
  372. {
  373. DoRelease(m_lpdobj);
  374. m_lpdobj = lpdobj; // processed in AddPages
  375. if (m_lpdobj)
  376. m_lpdobj->AddRef();
  377. return S_OK;
  378. }
  379. ///////////////////////////////////////////////////////////////////////////////
  380. // //
  381. // Shell extension object implementation (IShellPropSheetExt) //
  382. // //
  383. ///////////////////////////////////////////////////////////////////////////////
  384. /*++
  385. Routine Description:
  386. Support for IShellPropSheetExt::AddPages.
  387. Arguments:
  388. lpfnAddPage - pointer to function called to add a page.
  389. lParam - lParam parameter to be passed to lpfnAddPage.
  390. Return Values:
  391. Returns HRESULT signifying success or failure.
  392. --*/
  393. STDMETHODIMP
  394. CRShellExt::AddPages(LPFNADDPROPSHEETPAGE lpfnAddPage,
  395. LPARAM lParam)
  396. {
  397. HRESULT hr;
  398. STGMEDIUM medium = {0};
  399. FORMATETC fe = { g_cfShellIDList, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
  400. LPIDA pIDA = NULL;
  401. TraceEnter(TRACE_RSHX32, "CRShellExt::AddPages");
  402. if (IsSimpleUI())
  403. ExitGracefully(hr, E_FAIL, "No Security page in simple mode");
  404. //
  405. //Check if Security Tab is hidden by privacy policy
  406. //NTRAID#NTBUG9-223899-2001/03/06-hiteshr
  407. //
  408. if(IsUIHiddenByPrivacyPolicy())
  409. ExitGracefully(hr, E_FAIL, "Security Page is hidden by Privacy Policy");
  410. // Get the ID List data
  411. hr = m_lpdobj->GetData(&fe, &medium);
  412. #if (_WIN32_WINNT >= 0x0500)
  413. if (FAILED(hr) && m_seType == SE_FILE_OBJECT)
  414. TraceLeaveResult(AddMountedVolumePage(lpfnAddPage, lParam));
  415. #endif
  416. FailGracefully(hr, "Can't get ID List format from data object");
  417. pIDA = (LPIDA)GlobalLock(medium.hGlobal);
  418. TraceAssert(pIDA != NULL);
  419. // Only support single selection for printers
  420. if (m_seType == SE_PRINTER && pIDA->cidl != 1)
  421. ExitGracefully(hr, E_FAIL, "Printer multiple selection not supported");
  422. hr = DoSecurityCheck(pIDA);
  423. if (S_OK == hr)
  424. hr = AddSecurityPage(lpfnAddPage, lParam);
  425. exit_gracefully:
  426. if (pIDA)
  427. GlobalUnlock(medium.hGlobal);
  428. ReleaseStgMedium(&medium);
  429. TraceLeaveResult(hr);
  430. }
  431. /*++
  432. Routine Description:
  433. Support for IShellPropSheetExt::ReplacePages (not supported).
  434. Arguments:
  435. uPageID - page to replace.
  436. lpfnReplaceWith - pointer to function called to replace a page.
  437. lParam - lParam parameter to be passed to lpfnReplaceWith.
  438. Return Values:
  439. Returns E_FAIL.
  440. --*/
  441. STDMETHODIMP
  442. CRShellExt::ReplacePage(UINT /* uPageID */,
  443. LPFNADDPROPSHEETPAGE /* lpfnReplaceWith */,
  444. LPARAM /* lParam */)
  445. {
  446. return E_NOTIMPL;
  447. }
  448. ///////////////////////////////////////////////////////////////////////////////
  449. // //
  450. // Shell extension object implementation (IContextMenu) //
  451. // //
  452. ///////////////////////////////////////////////////////////////////////////////
  453. //
  454. // FUNCTION: IContextMenu::QueryContextMenu(HMENU, UINT, UINT, UINT, UINT)
  455. //
  456. // PURPOSE: Called by the shell just before the context menu is displayed.
  457. // This is where you add your specific menu items.
  458. //
  459. // PARAMETERS:
  460. // hMenu - Handle to the context menu
  461. // indexMenu - Index of where to begin inserting menu items
  462. // idCmdFirst - Lowest value for new menu ID's
  463. // idCmtLast - Highest value for new menu ID's
  464. // uFlags - Specifies the context of the menu event
  465. //
  466. // RETURN VALUE:
  467. // HRESULT signifying success or failure.
  468. //
  469. // COMMENTS:
  470. //
  471. STDMETHODIMP
  472. CRShellExt::QueryContextMenu(HMENU hMenu,
  473. UINT indexMenu,
  474. UINT idCmdFirst,
  475. UINT /*idCmdLast*/,
  476. UINT uFlags)
  477. {
  478. HRESULT hr = ResultFromShort(0);
  479. STGMEDIUM medium = {0};
  480. FORMATETC fe = { g_cfShellIDList, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
  481. if (uFlags & (CMF_DEFAULTONLY | CMF_VERBSONLY))
  482. return hr;
  483. TraceEnter(TRACE_RSHX32, "CRShellExt::QueryContextMenu");
  484. // Get the ID List data
  485. hr = m_lpdobj->GetData(&fe, &medium);
  486. if (SUCCEEDED(hr))
  487. {
  488. LPIDA pIDA = (LPIDA)GlobalLock(medium.hGlobal);
  489. TraceAssert(pIDA != NULL);
  490. // Only support single selection
  491. if (pIDA->cidl == 1)
  492. {
  493. if (S_OK == DoSecurityCheck(pIDA))
  494. {
  495. TCHAR szSecurity[32];
  496. if (LoadString(g_hInstance, IDS_SECURITY_MENU, szSecurity, ARRAYSIZE(szSecurity)))
  497. {
  498. MENUITEMINFO mii;
  499. mii.cbSize = sizeof(mii);
  500. mii.fMask = MIIM_TYPE | MIIM_ID;
  501. mii.fType = MFT_STRING;
  502. mii.wID = idCmdFirst;
  503. mii.dwTypeData = szSecurity;
  504. mii.cch = lstrlen(szSecurity);
  505. InsertMenuItem(hMenu, indexMenu, TRUE /*fByPosition*/, &mii);
  506. hr = ResultFromShort(1); // Return number of items we added
  507. }
  508. }
  509. }
  510. GlobalUnlock(medium.hGlobal);
  511. ReleaseStgMedium(&medium);
  512. }
  513. TraceLeaveResult(hr);
  514. }
  515. //
  516. // FUNCTION: IContextMenu::InvokeCommand(LPCMINVOKECOMMANDINFO)
  517. //
  518. // PURPOSE: Called by the shell after the user has selected on of the
  519. // menu items that was added in QueryContextMenu().
  520. //
  521. // PARAMETERS:
  522. // lpcmi - Pointer to an CMINVOKECOMMANDINFO structure
  523. //
  524. // RETURN VALUE:
  525. // HRESULT signifying success or failure.
  526. //
  527. // COMMENTS:
  528. //
  529. STDMETHODIMP
  530. CRShellExt::InvokeCommand(LPCMINVOKECOMMANDINFO lpcmi)
  531. {
  532. HRESULT hr = S_OK;
  533. STGMEDIUM medium;
  534. FORMATETC fe = { g_cfShellIDList, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
  535. // Don't support named verbs
  536. if (HIWORD(lpcmi->lpVerb))
  537. return E_NOTIMPL;
  538. TraceEnter(TRACE_RSHX32, "CRShellExt::InvokeCommand");
  539. // We only have one command, so we should get zero here
  540. TraceAssert(LOWORD(lpcmi->lpVerb) == 0);
  541. // This must be true for us to have added the command to the menu
  542. TraceAssert(S_OK == m_hrSecurityCheck);
  543. //
  544. // Call ShellExecuteEx to execute the "Properties" verb on this object, and
  545. // tell it to select the security property page.
  546. //
  547. // Get the ID List data
  548. hr = m_lpdobj->GetData(&fe, &medium);
  549. if (SUCCEEDED(hr))
  550. {
  551. LPIDA pIDA = (LPIDA)GlobalLock(medium.hGlobal);
  552. LPITEMIDLIST pidl;
  553. // We only support single selection for context menus
  554. TraceAssert(pIDA && pIDA->cidl == 1);
  555. // Build a fully qualified ID List for this object
  556. pidl = ILCombine((LPCITEMIDLIST)ByteOffset(pIDA, pIDA->aoffset[0]),
  557. (LPCITEMIDLIST)ByteOffset(pIDA, pIDA->aoffset[1]));
  558. if (pidl != NULL)
  559. {
  560. TCHAR szTitle[64];
  561. SHELLEXECUTEINFO sei =
  562. {
  563. sizeof(SHELLEXECUTEINFO),
  564. (lpcmi->fMask & (SEE_MASK_HOTKEY | SEE_MASK_ICON)) | SEE_MASK_INVOKEIDLIST,
  565. lpcmi->hwnd,
  566. c_szProperties, // lpVerb ("Properties")
  567. NULL, // lpFile
  568. szTitle, // lpParameters ("Security")
  569. NULL, // lpDirectory,
  570. lpcmi->nShow, // nShow
  571. NULL, // hInstApp
  572. (LPVOID)pidl, // lpIDList
  573. NULL, // lpClass
  574. NULL, // hkeyClass
  575. lpcmi->dwHotKey, // dwHotKey
  576. lpcmi->hIcon, // hIcon
  577. NULL // hProcess
  578. };
  579. LoadString(g_hInstance, IDS_PROPPAGE_TITLE, szTitle, ARRAYSIZE(szTitle));
  580. // Put up the properties dialog
  581. if (!ShellExecuteEx(&sei))
  582. {
  583. DWORD dwErr = GetLastError();
  584. hr = HRESULT_FROM_WIN32(dwErr);
  585. }
  586. ILFree(pidl);
  587. }
  588. GlobalUnlock(medium.hGlobal);
  589. ReleaseStgMedium(&medium);
  590. }
  591. #if 0
  592. //
  593. // SHObjectProperties builds a pidl to the object and then calls
  594. // ShellExecuteEx. Similar to above, but it does more work to obtain the
  595. // ID lists (which we already have).
  596. //
  597. SHObjectProperties(lpcmi->hwnd,
  598. m_seType == SE_PRINTER ? SHOP_PRINTERNAME : SHOP_FILEPATH,
  599. m_pszObject,
  600. TEXT("Security"));
  601. #endif
  602. TraceLeaveResult(hr);
  603. }
  604. //
  605. // FUNCTION: IContextMenu::GetCommandString(UINT, UINT, UINT, LPSTR, UINT)
  606. //
  607. // PURPOSE: Called by the shell after the user has selected on of the
  608. // menu items that was added in QueryContextMenu().
  609. //
  610. // PARAMETERS:
  611. // lpcmi - Pointer to an CMINVOKECOMMANDINFO structure
  612. //
  613. // RETURN VALUE:
  614. // HRESULT signifying success or failure.
  615. //
  616. // COMMENTS:
  617. //
  618. STDMETHODIMP
  619. CRShellExt::GetCommandString(UINT_PTR /*idCmd*/,
  620. UINT uFlags,
  621. LPUINT /*reserved*/,
  622. LPSTR pszName,
  623. UINT cchMax)
  624. {
  625. if (uFlags == GCS_HELPTEXT)
  626. {
  627. LoadString(g_hInstance, IDS_SECURITY_HELPSTRING, (LPTSTR)pszName, cchMax);
  628. return S_OK;
  629. }
  630. // Must be some other flag that we don't handle
  631. return E_NOTIMPL;
  632. }
  633. //
  634. // FUNCTION: CRShellExt::DoSecurityCheck(LPIDA)
  635. //
  636. // PURPOSE: Helper function called by the Property Sheet and Context Menu
  637. // extension code. Used to determine whether to add the menu item
  638. // or property sheet.
  639. //
  640. // PARAMETERS:
  641. // pIDA - pointer to ID List Array specifying selected objects
  642. //
  643. // RETURN VALUE: none
  644. //
  645. // COMMENTS:
  646. // The results are stored in m_hrSecurityCheck, m_dwSIFlags, m_pszServer, and m_pszObject
  647. //
  648. STDMETHODIMP CRShellExt::DoSecurityCheck(LPIDA pIDA)
  649. {
  650. if (((HRESULT)-1) == m_hrSecurityCheck)
  651. {
  652. if (m_seType == SE_PRINTER && IsAddPrinterWizard())
  653. m_hrSecurityCheck = HRESULT_FROM_WIN32(ERROR_NO_SECURITY_ON_OBJECT);
  654. else
  655. m_hrSecurityCheck = CheckForSecurity(pIDA);
  656. }
  657. return m_hrSecurityCheck;
  658. }
  659. //
  660. // PURPOSE: Helper function called by CRShellExt::DoSecurityCheck
  661. //
  662. // PARAMETERS: pIDA - pointer to ID List array
  663. //
  664. // RETURN VALUE: HRESULT - S_OK if ACL editing can proceed
  665. //
  666. // COMMENTS:
  667. // The results are stored in m_dwSIFlags, m_pszServer, and m_pszObject
  668. //
  669. STDMETHODIMP CRShellExt::CheckForSecurity(LPIDA pIDA)
  670. {
  671. HRESULT hr;
  672. TCHAR szServer[MAX_PATH];
  673. LPTSTR pszItem = NULL;
  674. // LPTSTR pszAlternate = NULL;
  675. DWORD dwFlags = 0;
  676. UINT cItems;
  677. IShellFolder2 * psf = NULL;
  678. LPCITEMIDLIST pidl;
  679. DWORD dwAttr;
  680. DWORD dwPrivs[] = { SE_SECURITY_PRIVILEGE, SE_TAKE_OWNERSHIP_PRIVILEGE };
  681. HANDLE hToken = INVALID_HANDLE_VALUE;
  682. ACCESS_MASK dwAccess = 0;
  683. UINT i;
  684. TraceEnter(TRACE_RSHX32, "CRShellExt::CheckForSecurity");
  685. TraceAssert(m_pszServer == NULL); // Shouldn't get called twice
  686. TraceAssert(pIDA != NULL);
  687. szServer[0] = TEXT('\0');
  688. cItems = pIDA->cidl;
  689. TraceAssert(cItems >= 1);
  690. //We don't show effective perm page for multiple selection
  691. if (cItems > 1)
  692. m_dwSIFlags &= ~SI_EDIT_EFFECTIVE;
  693. IShellFolder2 *psfRoot = NULL;
  694. LPCITEMIDLIST pidlFolder = (LPCITEMIDLIST)ByteOffset(pIDA, pIDA->aoffset[0]);
  695. hr = BindToObjectEx(NULL, pidlFolder, NULL, IID_PPV_ARG(IShellFolder2, &psfRoot));
  696. FailGracefully(hr, "Unable to bind to folder");
  697. TraceAssert(psfRoot);
  698. // Create list for item paths
  699. TraceAssert(NULL == m_hItemList);
  700. m_hItemList = DPA_Create(4);
  701. if (NULL == m_hItemList)
  702. ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create DPA");
  703. //
  704. // Get the first item and see if it supports security
  705. //
  706. LPCITEMIDLIST pidlItem = (LPCITEMIDLIST)ByteOffset(pIDA, pIDA->aoffset[1]);
  707. hr = BindToFolderIDListParent(psfRoot, pidlItem, IID_PPV_ARG(IShellFolder2, &psf), &pidl);
  708. FailGracefully(hr, "Unable to get item name");
  709. hr = IDA_GetItemName(psf, pidl, &pszItem);
  710. FailGracefully(hr, "Unable to get item name");
  711. dwAttr = SFGAO_FOLDER | SFGAO_STREAM | SFGAO_FILESYSTEM;
  712. hr = psf->GetAttributesOf(1, &pidl, &dwAttr);
  713. FailGracefully(hr, "Unable to get item attributes");
  714. DoRelease(psf);
  715. //
  716. //If ACLUI is invoked for filesystem and object is not of filesystem
  717. //return E_FAIL
  718. //
  719. if ((m_seType == SE_FILE_OBJECT) && !(dwAttr & SFGAO_FILESYSTEM))
  720. ExitGracefully(hr, E_FAIL, "Not a filesystem object");
  721. // in the case that an item is both folder and stream, assume its a stream (.zip, .cab file)
  722. // and not a container
  723. if ((dwAttr & (SFGAO_FOLDER | SFGAO_STREAM)) == SFGAO_FOLDER)
  724. dwFlags |= DOBJ_RES_CONT;
  725. //
  726. // Check access on the first item only. If we can write the DACL
  727. // on the first one, we will try (later) to write to all items
  728. // in the selection and report any errors at that time.
  729. //
  730. hToken = EnablePrivileges(dwPrivs, ARRAYSIZE(dwPrivs));
  731. switch (m_seType)
  732. {
  733. case SE_FILE_OBJECT:
  734. GetFileInfo(pszItem, &dwFlags, szServer, ARRAYSIZE(szServer), NULL);
  735. if (dwFlags & DOBJ_VOL_NTACLS)
  736. hr = CheckFileAccess(pszItem, &dwAccess);
  737. else
  738. hr = HRESULT_FROM_WIN32(ERROR_NO_SECURITY_ON_OBJECT);
  739. break;
  740. case SE_PRINTER:
  741. // Printers are containers (they contain documents)
  742. // and they don't have a parent (for acl editing purposes)
  743. dwFlags = DOBJ_RES_CONT | DOBJ_RES_ROOT;
  744. hr = CheckPrinterAccess(pszItem, &dwAccess, szServer, ARRAYSIZE(szServer));
  745. break;
  746. default:
  747. hr = E_UNEXPECTED;
  748. }
  749. FailGracefully(hr, "No access");
  750. // If we can't do anything security related, don't continue.
  751. if (!(dwAccess & ALL_SECURITY_ACCESS))
  752. ExitGracefully(hr, E_ACCESSDENIED, "No access");
  753. // Remember the server name
  754. if (TEXT('\0') != szServer[0])
  755. {
  756. hr = LocalAllocString(&m_pszServer, szServer);
  757. FailGracefully(hr, "LocalAlloc failed");
  758. }
  759. // Remember the item path
  760. DPA_AppendPtr(m_hItemList, pszItem);
  761. pszItem = NULL;
  762. if (!(dwAccess & WRITE_DAC))
  763. m_dwSIFlags |= SI_READONLY;
  764. if (!(dwAccess & WRITE_OWNER))
  765. {
  766. if (!(dwAccess & READ_CONTROL))
  767. m_dwSIFlags &= ~SI_EDIT_OWNER;
  768. else
  769. m_dwSIFlags |= SI_OWNER_READONLY;
  770. }
  771. if (!(dwAccess & ACCESS_SYSTEM_SECURITY))
  772. m_dwSIFlags &= ~SI_EDIT_AUDITS;
  773. //
  774. // Check the rest of the selection. If any part of a multiple
  775. // selection doesn't support ACLs or the selection isn't homogenous,
  776. // then we can't create the security page.
  777. //
  778. for (i = 2; i <= cItems; i++)
  779. {
  780. DWORD dw = 0;
  781. // We only do multiple selections for files
  782. TraceAssert(SE_FILE_OBJECT == m_seType);
  783. LPCITEMIDLIST pidlItem = (LPCITEMIDLIST)ByteOffset(pIDA, pIDA->aoffset[i]);
  784. hr = BindToFolderIDListParent(psfRoot, pidlItem, IID_PPV_ARG(IShellFolder2, &psf), &pidl);
  785. FailGracefully(hr, "Unable to get item name");
  786. hr = IDA_GetItemName(psf, pidl, &pszItem);
  787. FailGracefully(hr, "Unable to get item name");
  788. dwAttr = SFGAO_FOLDER | SFGAO_STREAM | SFGAO_FILESYSTEM;
  789. hr = psf->GetAttributesOf(1, &pidl, &dwAttr);
  790. FailGracefully(hr, "Unable to get item attributes");
  791. DoRelease(psf);
  792. //
  793. //If ACLUI is invoked for filesystem and object is not of filesystem
  794. //return E_FAIL
  795. //
  796. if ((m_seType == SE_FILE_OBJECT) && !(dwAttr & SFGAO_FILESYSTEM))
  797. ExitGracefully(hr, E_FAIL, "Not a filesystem object");
  798. if ((dwAttr & (SFGAO_FOLDER | SFGAO_STREAM)) == SFGAO_FOLDER)
  799. dw |= DOBJ_RES_CONT;
  800. if ((dw & DOBJ_RES_CONT) != (dwFlags & DOBJ_RES_CONT))
  801. ExitGracefully(hr, E_FAIL, "Incompatible multiple selection");
  802. GetFileInfo(pszItem, &dw, szServer, ARRAYSIZE(szServer), NULL);
  803. // Compare against first item. All flags and the server name
  804. // must match, otherwise we can't edit the ACLs.
  805. if (dw == dwFlags &&
  806. ((NULL == m_pszServer && TEXT('\0') == szServer[0]) ||
  807. (NULL != m_pszServer && 0 == lstrcmpi(m_pszServer, szServer))))
  808. {
  809. // Remember the item path
  810. DPA_AppendPtr(m_hItemList, pszItem);
  811. pszItem = NULL;
  812. }
  813. else
  814. ExitGracefully(hr, E_FAIL, "Incompatible multiple selection");
  815. }
  816. //
  817. // If everything has succeeded up to this point, save some flags
  818. // and the server and object name strings
  819. //
  820. if (dwFlags & DOBJ_RES_CONT)
  821. m_dwSIFlags |= SI_CONTAINER;
  822. //
  823. // For Root objects (e.g. "D:\") hide the ACL Protection checkbox,
  824. // since these objects don't appear to have parents.
  825. //
  826. if (dwFlags & DOBJ_RES_ROOT)
  827. m_dwSIFlags |= SI_NO_ACL_PROTECT;
  828. // Get the "Normal" display name to use as the object name
  829. hr = IDA_GetItemName(psfRoot, (LPCITEMIDLIST)ByteOffset(pIDA, pIDA->aoffset[1]),
  830. szServer, ARRAYSIZE(szServer), SHGDN_NORMAL);
  831. FailGracefully(hr, "Unable to get item name");
  832. if (cItems > 1)
  833. {
  834. int nLength = lstrlen(szServer);
  835. LoadString(g_hInstance, IDS_MULTISEL_ELLIPSIS, szServer + nLength, ARRAYSIZE(szServer) - nLength);
  836. }
  837. hr = LocalAllocString(&m_pszObject, szServer);
  838. exit_gracefully:
  839. ReleasePrivileges(hToken);
  840. DoRelease(psf);
  841. DoRelease(psfRoot);
  842. LocalFreeString(&pszItem);
  843. TraceLeaveResult(hr);
  844. }
  845. //
  846. // FUNCTION: CRShellExt::CreateSI(LPSECURITYINFO *)
  847. //
  848. // PURPOSE: Create a SecurityInformation object of the correct type
  849. //
  850. // PARAMETERS: ppsi - Location to store ISecurityInformation pointer
  851. //
  852. // RETURN VALUE: HRESULT signifying success or failure
  853. //
  854. // COMMENTS:
  855. //
  856. STDMETHODIMP
  857. CRShellExt::CreateSI(LPSECURITYINFO *ppsi)
  858. {
  859. HRESULT hr;
  860. CSecurityInformation *psi;
  861. TraceEnter(TRACE_RSHX32, "CRShellExt::CreateSI");
  862. TraceAssert(ppsi != NULL);
  863. *ppsi = NULL;
  864. switch (m_seType)
  865. {
  866. case SE_FILE_OBJECT:
  867. psi = new CNTFSSecurity(m_seType); // ref == 1
  868. break;
  869. case SE_PRINTER:
  870. psi = new CPrintSecurity(m_seType); // ref == 1
  871. break;
  872. default:
  873. TraceLeaveResult(E_UNEXPECTED);
  874. }
  875. if (psi == NULL)
  876. TraceLeaveResult(E_OUTOFMEMORY);
  877. hr = psi->Initialize(m_hItemList,
  878. m_dwSIFlags,
  879. m_pszServer,
  880. m_pszObject);
  881. if (SUCCEEDED(hr))
  882. {
  883. *ppsi = psi;
  884. // The SecurityInfo object takes responsibility for these
  885. m_hItemList = NULL;
  886. m_pszServer = NULL;
  887. m_pszObject = NULL;
  888. m_hrSecurityCheck = (HRESULT)-1;
  889. }
  890. else
  891. psi->Release();
  892. TraceLeaveResult(hr);
  893. }
  894. typedef HPROPSHEETPAGE (WINAPI *PFN_CREATESECPAGE)(LPSECURITYINFO);
  895. HPROPSHEETPAGE _CreateSecurityPage(LPSECURITYINFO psi)
  896. {
  897. HPROPSHEETPAGE hPage = NULL;
  898. const TCHAR szAclui[] = TEXT("aclui.dll");
  899. const char szCreateSecPage[] = "CreateSecurityPage";
  900. if (!g_hAclui)
  901. g_hAclui = LoadLibrary(szAclui);
  902. if (g_hAclui)
  903. {
  904. static PFN_CREATESECPAGE s_pfnCreateSecPage = NULL;
  905. if (!s_pfnCreateSecPage)
  906. s_pfnCreateSecPage = (PFN_CREATESECPAGE)GetProcAddress(g_hAclui, szCreateSecPage);
  907. if (s_pfnCreateSecPage)
  908. hPage = (*s_pfnCreateSecPage)(psi);
  909. }
  910. return hPage;
  911. }
  912. STDMETHODIMP
  913. CRShellExt::AddSecurityPage(LPFNADDPROPSHEETPAGE lpfnAddPage, LPARAM lParam)
  914. {
  915. HRESULT hr;
  916. LPSECURITYINFO psi;
  917. hr = CreateSI(&psi); // ref == 1
  918. if (SUCCEEDED(hr))
  919. {
  920. HPROPSHEETPAGE hPermPage = _CreateSecurityPage(psi);
  921. if (hPermPage)
  922. {
  923. if (!lpfnAddPage(hPermPage, lParam))
  924. DestroyPropertySheetPage(hPermPage);
  925. }
  926. else
  927. {
  928. DWORD dwErr = GetLastError();
  929. hr = HRESULT_FROM_WIN32(dwErr);
  930. }
  931. psi->Release(); // release initial ref
  932. }
  933. return hr;
  934. }
  935. //
  936. // PURPOSE: Check for the Add Printer wizard
  937. //
  938. // PARAMETERS: none
  939. //
  940. // RETURN VALUE: TRUE if the selected object is the Add Printer wizard,
  941. // FALSE otherwise
  942. //
  943. // COMMENTS:
  944. //
  945. BOOL CRShellExt::IsAddPrinterWizard() const
  946. {
  947. BOOL bRetval = FALSE;
  948. STGMEDIUM medium;
  949. FORMATETC fe = { g_cfPrinterGroup, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
  950. TCHAR szFile[MAX_PATH];
  951. TraceEnter(TRACE_RSHX32, "CRShellExt::IsAddPrinterWizard");
  952. TraceAssert(m_seType == SE_PRINTER);
  953. //
  954. // Fail the call if m_lpdobj is NULL.
  955. //
  956. if ( m_lpdobj && SUCCEEDED( m_lpdobj->GetData( &fe, &medium ) ) )
  957. {
  958. #if (_WIN32_WINNT < 0x0500)
  959. //
  960. // In NT 4.0 the printer context menus in the shell failed
  961. // to handle UNICODE strings properly during a call to DragQueryFile.
  962. // The strings returned were always ANSI. This has been fixed
  963. // in build 1393 and greater.
  964. //
  965. #ifdef UNICODE
  966. LPDROPFILES pdf = (LPDROPFILES)GlobalLock(medium.hGlobal);
  967. pdf->fWide = TRUE;
  968. GlobalUnlock(medium.hGlobal);
  969. #endif
  970. #endif
  971. //
  972. // Get the selected item name.
  973. //
  974. if ( DragQueryFile( (HDROP)medium.hGlobal, 0, szFile, ARRAYSIZE( szFile ) ) )
  975. {
  976. //
  977. // Check if this is the magic Add Printer Wizard shell object.
  978. // The check is not case sensitive and the string is not localized.
  979. //
  980. if ( 0 == lstrcmpi( szFile, TEXT("WinUtils_NewObject") ) )
  981. {
  982. TraceMsg("Found Add Printer wizard");
  983. bRetval = TRUE;
  984. }
  985. }
  986. //
  987. // Release the storage medium.
  988. //
  989. ReleaseStgMedium( &medium );
  990. }
  991. TraceLeaveValue(bRetval);
  992. }
  993. #if (_WIN32_WINNT >= 0x0500)
  994. //
  995. // FUNCTION: CRShellExt::AddMountedVolumePage()
  996. //
  997. // PURPOSE: Create Security page for mounted volume properties
  998. //
  999. // PARAMETERS: lpfnAddPage - pointer to function called to add a page.
  1000. // lParam - lParam parameter to be passed to lpfnAddPage.
  1001. //
  1002. // RETURN VALUE: HRESULT signifying success or failure
  1003. //
  1004. // COMMENTS:
  1005. //
  1006. STDMETHODIMP
  1007. CRShellExt::AddMountedVolumePage(LPFNADDPROPSHEETPAGE lpfnAddPage,
  1008. LPARAM lParam)
  1009. {
  1010. HRESULT hr = S_OK;
  1011. STGMEDIUM medium = {0};
  1012. FORMATETC fe = { g_cfMountedVolume, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
  1013. TCHAR szMountPoint[MAX_PATH];
  1014. TCHAR szVolumeID[MAX_PATH];
  1015. TCHAR szLabel[64];
  1016. LPTSTR pszVolID = NULL;
  1017. DWORD dwVolFlags = 0;
  1018. DWORD dwPrivs[] = { SE_SECURITY_PRIVILEGE, SE_TAKE_OWNERSHIP_PRIVILEGE };
  1019. HANDLE hToken = INVALID_HANDLE_VALUE;
  1020. ACCESS_MASK dwAccess = 0;
  1021. BOOL bHasSecurity = FALSE;
  1022. TraceEnter(TRACE_RSHX32, "CRShellExt::AddMountedVolumePage");
  1023. TraceAssert(m_seType == SE_FILE_OBJECT);
  1024. TraceAssert(m_lpdobj);
  1025. // Try to get the mounted volume host folder path
  1026. hr = m_lpdobj->GetData(&fe, &medium);
  1027. FailGracefully(hr, "Not a mounted volume");
  1028. // Get the host folder path
  1029. if (!DragQueryFile((HDROP)medium.hGlobal, 0, szMountPoint, ARRAYSIZE(szMountPoint)))
  1030. ExitGracefully(hr, E_FAIL, "Can't get mount point from storage medium");
  1031. PathAddBackslash(szMountPoint);
  1032. // Get the volume ID, which looks like
  1033. // "\\?\Volume{9e2df3f5-c7f1-11d1-84d5-000000000000}\"
  1034. if (!GetVolumeNameForVolumeMountPoint(szMountPoint, szVolumeID, ARRAYSIZE(szVolumeID)))
  1035. ExitGracefully(hr, E_FAIL, "GetVolumeNameForVolumeMountPoint failed");
  1036. if (GetVolumeInformation(szMountPoint, //szVolumeID,
  1037. szLabel,
  1038. ARRAYSIZE(szLabel),
  1039. NULL,
  1040. NULL,
  1041. &dwVolFlags,
  1042. NULL,
  1043. 0))
  1044. {
  1045. if (dwVolFlags & FS_PERSISTENT_ACLS)
  1046. {
  1047. bHasSecurity = TRUE;
  1048. }
  1049. }
  1050. else if (GetLastError() == ERROR_ACCESS_DENIED)
  1051. {
  1052. // If we can't get the volume information because we don't have
  1053. // access, then there must be security!
  1054. bHasSecurity = TRUE;
  1055. }
  1056. if (!bHasSecurity)
  1057. ExitGracefully(hr, E_FAIL, "Volume inaccessible or not NTFS");
  1058. hToken = EnablePrivileges(dwPrivs, ARRAYSIZE(dwPrivs));
  1059. hr = CheckFileAccess(szVolumeID, &dwAccess);
  1060. FailGracefully(hr, "Volume inaccessible");
  1061. // If we can't do anything security related, don't continue.
  1062. if (!(dwAccess & ALL_SECURITY_ACCESS))
  1063. ExitGracefully(hr, E_ACCESSDENIED, "No security access");
  1064. if (!(dwAccess & WRITE_DAC))
  1065. m_dwSIFlags |= SI_READONLY;
  1066. if (!(dwAccess & WRITE_OWNER))
  1067. {
  1068. if (!(dwAccess & READ_CONTROL))
  1069. m_dwSIFlags &= ~SI_EDIT_OWNER;
  1070. else
  1071. m_dwSIFlags |= SI_OWNER_READONLY;
  1072. }
  1073. if (!(dwAccess & ACCESS_SYSTEM_SECURITY))
  1074. m_dwSIFlags &= ~SI_EDIT_AUDITS;
  1075. m_dwSIFlags |= SI_CONTAINER | SI_NO_ACL_PROTECT;
  1076. if (!FormatStringID(&m_pszObject,
  1077. g_hInstance,
  1078. IDS_FMT_VOLUME_DISPLAY,
  1079. szLabel,
  1080. szMountPoint))
  1081. {
  1082. LocalAllocString(&m_pszObject, szLabel);
  1083. }
  1084. if (!m_pszObject)
  1085. ExitGracefully(hr, E_OUTOFMEMORY, "Unable to build volume display string");
  1086. m_hItemList = DPA_Create(1);
  1087. if (!m_hItemList)
  1088. ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create item list");
  1089. hr = LocalAllocString(&pszVolID, szVolumeID);
  1090. FailGracefully(hr, "Unable to copy volume ID string");
  1091. DPA_AppendPtr(m_hItemList, pszVolID);
  1092. pszVolID = NULL;
  1093. hr = AddSecurityPage(lpfnAddPage, lParam);
  1094. exit_gracefully:
  1095. ReleasePrivileges(hToken);
  1096. LocalFreeString(&pszVolID);
  1097. ReleaseStgMedium(&medium);
  1098. TraceLeaveResult(hr);
  1099. }
  1100. #endif // _WIN32_WINNT >= 0x0500
  1101. ///////////////////////////////////////////////////////////////////////////////
  1102. // //
  1103. // Miscellaneous helper functions //
  1104. // //
  1105. ///////////////////////////////////////////////////////////////////////////////
  1106. #if (_WIN32_WINNT < 0x0500)
  1107. #undef PathIsUNC
  1108. STDAPI_(BOOL)
  1109. PathIsUNC(LPCTSTR psz)
  1110. {
  1111. return (psz && psz[0] == TEXT('\\') && psz[1] == TEXT('\\'));
  1112. }
  1113. static const TCHAR c_szColonSlash[] = TEXT(":\\");
  1114. #undef PathIsRoot
  1115. STDAPI_(BOOL)
  1116. PathIsRoot(LPCTSTR pPath)
  1117. {
  1118. return (pPath && !lstrcmpi(pPath + 1, c_szColonSlash));
  1119. }
  1120. #endif // #if (_WIN32_WINNT < 0x0500)
  1121. BOOL
  1122. IsDfsPath(LPTSTR pszPath, // in
  1123. LPTSTR pszServer, // out
  1124. UINT cchServer, // in
  1125. LPTSTR pszAltPath) // out
  1126. {
  1127. BOOL bIsDfs = FALSE;
  1128. WCHAR szPath[MAX_PATH];
  1129. PDFS_INFO_3 pDI3 = NULL;
  1130. WCHAR szServer[MAX_PATH];
  1131. WCHAR szStorage[MAX_PATH];
  1132. USES_CONVERSION;
  1133. if (!PathIsUNC(pszPath))
  1134. return FALSE; // local machine
  1135. lstrcpynW(szPath, T2CW(pszPath), ARRAYSIZE(szPath));
  1136. // Check for DFS
  1137. for (;;)
  1138. {
  1139. DWORD dwErr;
  1140. __try
  1141. {
  1142. // This is delay-loaded by the linker, so
  1143. // must wrap with an exception handler.
  1144. dwErr = NetDfsGetClientInfo(szPath,
  1145. NULL,
  1146. NULL,
  1147. 3,
  1148. (LPBYTE*)&pDI3);
  1149. }
  1150. __except(EXCEPTION_EXECUTE_HANDLER)
  1151. {
  1152. return FALSE;
  1153. }
  1154. if (NERR_Success == dwErr)
  1155. {
  1156. for (ULONG i = 0; i < pDI3->NumberOfStorages; i++)
  1157. {
  1158. if (DFS_STORAGE_STATE_ONLINE & pDI3->Storage[i].State)
  1159. {
  1160. bIsDfs = TRUE;
  1161. szServer[0] = L'\\';
  1162. szServer[1] = L'\\';
  1163. lstrcpynW(&szServer[2], pDI3->Storage[i].ServerName, ARRAYSIZE(szServer)-2);
  1164. lstrcpynW(szStorage, szServer, ARRAYSIZE(szStorage));
  1165. PathAppendW(szStorage, pDI3->Storage[i].ShareName);
  1166. // If this server is active, quit looking
  1167. if (DFS_STORAGE_STATE_ACTIVE & pDI3->Storage[i].State)
  1168. break;
  1169. }
  1170. }
  1171. break;
  1172. }
  1173. else if (NERR_DfsNoSuchVolume == dwErr)
  1174. {
  1175. // If we're at the root, then we can't go any farther.
  1176. if (PathIsRoot(szPath))
  1177. break;
  1178. // Remove the last path element and try again, if nothing is
  1179. //removed, break, don't go in infinite loop
  1180. if (!PathRemoveFileSpec(szPath))
  1181. break;
  1182. }
  1183. else
  1184. {
  1185. // Some other error, bail
  1186. break;
  1187. }
  1188. }
  1189. if (bIsDfs)
  1190. {
  1191. lstrcpyn(pszServer, W2T(szServer), cchServer);
  1192. // Note that EntryPath has only a single leading backslash, hence +1
  1193. LPCTSTR pszEnd = pszPath + lstrlen(pDI3->EntryPath) + 1;
  1194. while (TEXT('\\') == *pszEnd)
  1195. pszEnd++;
  1196. PathCombine(pszAltPath, szStorage, pszEnd);
  1197. }
  1198. if (NULL != pDI3)
  1199. NetApiBufferFree(pDI3);
  1200. return bIsDfs;
  1201. }
  1202. void
  1203. GetVolumeInfo(LPCTSTR pszPath,
  1204. BOOL bIsFolder,
  1205. LPDWORD pdwFlags,
  1206. LPTSTR pszVolume,
  1207. ULONG cchVolume)
  1208. {
  1209. TCHAR szVolume[MAX_PATH];
  1210. TCHAR szVolumeID[MAX_PATH];
  1211. //
  1212. // The path can be DFS or contain volume mount points, so start
  1213. // with the full path and try GetVolumeInformation on successively
  1214. // shorter paths until it succeeds or we run out of path.
  1215. //
  1216. // However, if it's a volume mount point, we're interested in the
  1217. // the host folder's volume so back up one level to start. The
  1218. // child volume is handled separately (see AddMountedVolumePage).
  1219. //
  1220. lstrcpyn(szVolume, pszPath, ARRAYSIZE(szVolume));
  1221. if (!bIsFolder
  1222. || GetVolumeNameForVolumeMountPoint(szVolume, szVolumeID, ARRAYSIZE(szVolumeID)))
  1223. {
  1224. PathRemoveFileSpec(szVolume);
  1225. }
  1226. for (;;)
  1227. {
  1228. PathAddBackslash(szVolume); // GetVolumeInformation likes a trailing '\'
  1229. if (GetVolumeInformation(szVolume,
  1230. NULL,
  1231. NULL,
  1232. NULL,
  1233. NULL,
  1234. pdwFlags,
  1235. NULL,
  1236. 0))
  1237. {
  1238. break;
  1239. }
  1240. // Access denied implies that we've reached the deepest volume
  1241. // in the path; we just can't get the flags. It also implies
  1242. // security, so assume persistent acls.
  1243. if (ERROR_ACCESS_DENIED == GetLastError())
  1244. {
  1245. *pdwFlags = FS_PERSISTENT_ACLS;
  1246. break;
  1247. }
  1248. // If we're at the root, then we can't go any farther.
  1249. if (PathIsRoot(szVolume))
  1250. break;
  1251. // Remove the last path element and try again
  1252. PathRemoveBackslash(szVolume);
  1253. //if nothing is removed break instead of going in infinite loop
  1254. if (!PathRemoveFileSpec(szVolume))
  1255. break;
  1256. }
  1257. if (pszVolume)
  1258. {
  1259. PathRemoveBackslash(szVolume);
  1260. lstrcpyn(pszVolume, szVolume, cchVolume);
  1261. }
  1262. }
  1263. void
  1264. GetFileInfo(LPCTSTR pszPath,
  1265. LPDWORD pdwFileType,
  1266. LPTSTR pszServer,
  1267. ULONG cchServer,
  1268. LPTSTR *ppszAlternatePath)
  1269. {
  1270. DWORD dwVolumeFlags = 0;
  1271. TCHAR szVolume[MAX_PATH];
  1272. LPTSTR pszUNC = NULL;
  1273. TraceEnter(TRACE_RSHX32, "GetFileInfo");
  1274. TraceAssert(NULL != pszServer);
  1275. pszServer[0] = TEXT('\0');
  1276. if (ppszAlternatePath)
  1277. *ppszAlternatePath = NULL;
  1278. if (!PathIsUNC(pszPath) && S_OK == GetRemotePath(pszPath, &pszUNC))
  1279. pszPath = pszUNC;
  1280. if (PathIsRoot(pszPath))
  1281. *pdwFileType |= DOBJ_RES_ROOT;
  1282. GetVolumeInfo(pszPath,
  1283. *pdwFileType & DOBJ_RES_CONT,
  1284. &dwVolumeFlags,
  1285. szVolume,
  1286. ARRAYSIZE(szVolume));
  1287. if (dwVolumeFlags & FS_PERSISTENT_ACLS)
  1288. {
  1289. TCHAR szAltVolume[MAX_PATH];
  1290. *pdwFileType |= DOBJ_VOL_NTACLS;
  1291. if (IsDfsPath(szVolume, pszServer, cchServer, szAltVolume))
  1292. {
  1293. if (ppszAlternatePath)
  1294. {
  1295. LPCTSTR pszEnd = pszPath + lstrlen(szVolume);
  1296. while (TEXT('\\') == *pszEnd)
  1297. pszEnd++;
  1298. *ppszAlternatePath = (LPTSTR)LocalAlloc(LPTR, StringByteSize(szAltVolume)
  1299. + StringByteSize(pszEnd));
  1300. if (*ppszAlternatePath)
  1301. {
  1302. // PathCombine asserts that the buffer is at least MAX_PATH,
  1303. // so don't use it here. We know the buffer is big enough.
  1304. lstrcpy(*ppszAlternatePath, szAltVolume);
  1305. LPTSTR pszT = PathAddBackslash(*ppszAlternatePath);
  1306. if (pszT)
  1307. lstrcpy(pszT, pszEnd);
  1308. //PathCombine(*ppszAlternatePath, szAltVolume, pszEnd);
  1309. }
  1310. }
  1311. }
  1312. else if (PathIsUNC(szVolume))
  1313. {
  1314. LPTSTR pSlash = StrChr(&szVolume[2], TEXT('\\'));
  1315. if (pSlash)
  1316. cchServer = min(cchServer, (ULONG)(pSlash - szVolume) + 1);
  1317. lstrcpyn(pszServer, szVolume, cchServer);
  1318. }
  1319. }
  1320. LocalFreeString(&pszUNC);
  1321. TraceLeaveVoid();
  1322. }