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.

1993 lines
60 KiB

  1. //+-------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation, 1992 - 1996.
  5. //
  6. // File: expdf.cxx
  7. //
  8. // Contents: Exposed DocFile implementation
  9. //
  10. // Notes:
  11. // The CExposedDocFile class is the implementation
  12. // of IStorage. It implements IPropertySetStorage
  13. // by inheriting from CPropertySetStorage. CPropertySetStorage
  14. // implements all the functionality of IPropertySetStorage.
  15. //
  16. // Note that this interface is solely UNICODE, the ASCII layer
  17. // support which is present if _UNICODE is not defined, provides
  18. // the overloaded functions that handles the ASCII to Unicode
  19. // conversion.
  20. //
  21. //---------------------------------------------------------------------------
  22. // Initialize all the GUID's in ref.hxx
  23. #ifdef INITGUID
  24. #error "Something is Wrong: INIT_GUID should not be defined yet"
  25. #else
  26. #define INITGUID
  27. #include "h/ref.hxx"
  28. #endif
  29. // enable memory leak detection if neccessary
  30. #include "h/dbg.hxx"
  31. #include "exphead.cxx"
  32. #include "expdf.hxx"
  33. #include "expst.hxx"
  34. #include "expiter.hxx"
  35. #include "logfile.hxx"
  36. #include "h/rexpdf.hxx"
  37. #include "h/docfilep.hxx"
  38. // Check for proper single-instance flags
  39. #define NOT_SINGLE(md) (((md) & STGM_DENY) != STGM_SHARE_EXCLUSIVE)
  40. #define EnforceSingle(mode) (NOT_SINGLE(mode) ? STG_E_INVALIDFUNCTION : S_OK)
  41. //+--------------------------------------------------------------
  42. //
  43. // Member: CExposedDocFile::CExposedDocFile, public
  44. //
  45. // Synopsis: Constructor
  46. //
  47. // Arguments: [pdf] - Public DocFile
  48. // [pdfb] - DocFile basis
  49. // [ppc] - Context
  50. // [fOwnContext] - Whether this object owns the context
  51. //
  52. //---------------------------------------------------------------
  53. CExposedDocFile::CExposedDocFile(CExposedDocFile *pdfParent,
  54. CDocFile *pdf,
  55. DFLAGS const df,
  56. DFLUID luid,
  57. ILockBytes *pilbBase,
  58. CDfName const *pdfn,
  59. CMStream* pmsBase,
  60. CDFBasis *pdfb)
  61. #ifdef NEWPROPS
  62. #ifdef _MSC_VER
  63. #pragma warning(disable: 4355)
  64. #endif // _MSC_VER
  65. : CPropertySetStorage(this)
  66. #ifdef _MSC_VER
  67. #pragma warning(default: 4355)
  68. #endif // _MSC_VER
  69. #endif // NEWPROPS
  70. {
  71. olDebugOut((DEB_ITRACE, "In CExposedDocFile::CExposedDocFile(%p)\n",
  72. pdf));
  73. _pdfb = pdfb;
  74. _pdfb->AddRef();
  75. _pdf = pdf;
  76. _df = df;
  77. _luid = luid;
  78. _pdfParent = pdfParent;
  79. // note: we don't addref here 'cos it is only done in the root
  80. _pilbBase = pilbBase;
  81. _pmsBase = pmsBase;
  82. if (pdfn) _dfn.Set(pdfn->GetLength(), pdfn->GetBuffer());
  83. else _dfn.Set((WORD)0, (BYTE*)NULL);
  84. if (!IsRoot())
  85. _pdfParent->AddChild(this);
  86. _fDirty = FALSE;
  87. _cReferences = 1;
  88. _ulAccessLockBase = 0;
  89. _sig = CEXPOSEDDOCFILE_SIG;
  90. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::CExposedDocFile\n"));
  91. }
  92. //+--------------------------------------------------------------
  93. //
  94. // Member: CExposedDocFile::~CExposedDocFile, public
  95. //
  96. // Synopsis: Destructor
  97. //
  98. //---------------------------------------------------------------
  99. CExposedDocFile::~CExposedDocFile(void)
  100. {
  101. olDebugOut((DEB_TRACE, "In CExposedDocFile::~CExposedDocFile\n"));
  102. olAssert(_cReferences == 0);
  103. if (_pdfb) _pdfb->Release();
  104. _sig = CEXPOSEDDOCFILE_SIGDEL;
  105. if (SUCCEEDED(CheckReverted()))
  106. {
  107. if (IsRoot()) {
  108. olDebugOut((DEB_TRACE, "Destr called for root\n"));
  109. olAssert(_pilbBase==NULL);
  110. }
  111. else {
  112. _pdfParent->ReleaseChild(this);
  113. }
  114. _cilChildren.DeleteByName(NULL);
  115. if (_pdf) _pdf->Release();
  116. }
  117. olDebugOut((DEB_TRACE, "Out CExposedDocFile::~CExposedDocFile\n"));
  118. }
  119. //+--------------------------------------------------------------
  120. //
  121. // Member: CExposedDocFile::Release, public
  122. //
  123. // Synopsis: Releases resources for a CExposedDocFile
  124. //
  125. // Returns: Appropriate status code
  126. //
  127. //---------------------------------------------------------------
  128. STDMETHODIMP_(ULONG) CExposedDocFile::Release(void)
  129. {
  130. LONG lRet;
  131. olLog(("%p::In CExposedDocFile::Release()\n", this));
  132. olDebugOut((DEB_ITRACE, "In CExposedDocFile::Release()\n"));
  133. TRY
  134. {
  135. if (FAILED(Validate())) return 0;
  136. olAssert(_cReferences > 0);
  137. lRet = AtomicDec(&_cReferences);
  138. if (_pdf && !P_TRANSACTED(_df) && SUCCEEDED(CheckReverted()))
  139. {
  140. TIME_T tm;
  141. olVerSucc(DfGetTOD(&tm));
  142. olVerSucc(_pdf->SetTime(WT_ACCESS, tm));
  143. #ifdef NEWPROPS
  144. olVerSucc(FlushBufferedData());
  145. #endif
  146. if (IsDirty())
  147. {
  148. olVerSucc(DfGetTOD(&tm));
  149. olVerSucc(_pdf->SetTime(WT_MODIFICATION, tm));
  150. if (!IsRoot())
  151. _pdfParent->SetDirty();
  152. olAssert(P_WRITE(_df) &&
  153. aMsg("Dirty & Direct but no write access"));
  154. SetClean();
  155. }
  156. if (IsRoot() && P_WRITE(_df))
  157. {
  158. SCODE sc;
  159. sc = _pmsBase->Flush(0);
  160. #if DBG == 1
  161. if (FAILED(sc))
  162. {
  163. olDebugOut((DEB_ERROR,
  164. "ILockBytes::Flush() failed in release path "
  165. "with error %lx\n", sc));
  166. }
  167. #endif
  168. }
  169. }
  170. if (lRet == 0)
  171. {
  172. delete this;
  173. }
  174. else if (lRet < 0)
  175. lRet = 0;
  176. }
  177. CATCH(CException, e)
  178. {
  179. UNREFERENCED_PARM(e);
  180. lRet = 0;
  181. }
  182. END_CATCH
  183. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::Release()\n"));
  184. olLog(("%p::Out CExposedDocFile::Release(). ret == %lu\n", this, lRet));
  185. FreeLogFile();
  186. return (ULONG)lRet;
  187. }
  188. //+--------------------------------------------------------------
  189. //
  190. // Member: CExposedDocFile::CheckCopyTo, private
  191. //
  192. // Synopsis: Checks for CopyTo legality
  193. //
  194. // Returns: Appropriate status code
  195. //
  196. //---------------------------------------------------------------
  197. inline SCODE CExposedDocFile::CheckCopyTo(void)
  198. {
  199. // it is an error to copy a parent to child
  200. return _pdfb->GetCopyBase() != NULL &&
  201. IsAtOrAbove(_pdfb->GetCopyBase()) ? STG_E_ACCESSDENIED : S_OK;
  202. }
  203. //+--------------------------------------------------------------
  204. //
  205. // Member: CExposedDocFile::ConvertInternalStream, private
  206. //
  207. // Synopsis: Converts an internal stream to a storage
  208. //
  209. // Arguments: [pwcsName] - Name
  210. // [pdfExp] - Destination docfile
  211. //
  212. // Returns: Appropriate status code
  213. //
  214. //---------------------------------------------------------------
  215. static WCHAR const wcsIllegalName[] = {'\\','\0'};
  216. SCODE CExposedDocFile::ConvertInternalStream(CExposedDocFile *pdfExp)
  217. {
  218. CExposedStream *pstFrom=NULL, *pstTo=NULL;
  219. SCODE sc=S_OK;
  220. CDfName const dfnIllegal(wcsIllegalName);
  221. CDfName const dfnContents(wcsContents);
  222. olDebugOut((DEB_ITRACE, "In CExposedDocFile::ConvertInternalStream(%p)\n",
  223. pdfExp));
  224. olChk(GetExposedStream(&dfnIllegal, DF_READWRITE | DF_DENYALL,
  225. &pstFrom));
  226. olChkTo(EH_pstFrom,
  227. pdfExp->CreateExposedStream(&dfnContents, DF_WRITE | DF_DENYALL,
  228. &pstTo));
  229. olChkTo(EH_pstTo, CopyStreamToStream(pstFrom->GetDirectStream(),
  230. pstTo->GetDirectStream()));
  231. olChkTo(EH_pstTo, DestroyEntry(&dfnIllegal, FALSE));
  232. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::ConvertInternalStream\n"));
  233. // Fall through
  234. EH_pstTo:
  235. pstTo->Release();
  236. EH_pstFrom:
  237. pstFrom->Release();
  238. EH_Err:
  239. return sc;
  240. }
  241. //+---------------------------------------------------------------------------
  242. //
  243. // Member: CExposedDocFile::CreateEntry, private
  244. //
  245. // Synopsis: Creates elements, used in CreateStream, CreateStorage and
  246. // for properties
  247. //
  248. // Arguments: [pwcsName] - Name
  249. // [dwType] - Entry type
  250. // [grfMode] - Access mode
  251. // [ppv] - Object return
  252. //
  253. // Returns: Appropriate status code
  254. //
  255. // Modifies: [ppv]
  256. //
  257. //----------------------------------------------------------------------------
  258. SCODE CExposedDocFile::CreateEntry(WCHAR const *pwcsName,
  259. DWORD dwType,
  260. DWORD grfMode,
  261. void **ppv)
  262. {
  263. SCODE sc;
  264. SEntryBuffer eb;
  265. CDfName dfn;
  266. BOOL fRenamed = FALSE;
  267. CExposedStream *pstExp;
  268. CExposedDocFile *pdfExp;
  269. olDebugOut((DEB_ITRACE, "In CExposedDocFile::CreateEntry:%p("
  270. "%ws, %lX, %lX, %p)\n",
  271. this, pwcsName, dwType, grfMode, ppv));
  272. olChk(EnforceSingle(grfMode));
  273. olChk(CheckReverted());
  274. dfn.Set(pwcsName);
  275. if (grfMode & (STGM_CREATE | STGM_CONVERT))
  276. {
  277. if (FAILED(sc = _pdf->IsEntry(&dfn, &eb)))
  278. {
  279. if (sc != STG_E_FILENOTFOUND)
  280. olErr(EH_Err, sc);
  281. }
  282. else if (eb.dwType == dwType && (grfMode & STGM_CREATE))
  283. olChk(DestroyEntry(&dfn, FALSE));
  284. else if (eb.dwType == STGTY_STREAM && (grfMode & STGM_CONVERT) &&
  285. dwType == STGTY_STORAGE)
  286. {
  287. CDfName const dfnIllegal(wcsIllegalName);
  288. olChk(RenameEntry(&dfn, &dfnIllegal));
  289. fRenamed = TRUE;
  290. }
  291. else
  292. olErr(EH_Err, STG_E_FILEALREADYEXISTS);
  293. }
  294. if (REAL_STGTY(dwType) == STGTY_STREAM)
  295. {
  296. olChk(CreateExposedStream(&dfn, ModeToDFlags(grfMode), &pstExp));
  297. *ppv = pstExp;
  298. }
  299. else
  300. {
  301. olAssert(REAL_STGTY(dwType) == STGTY_STORAGE);
  302. olChk(CreateExposedDocFile(&dfn, ModeToDFlags(grfMode), &pdfExp));
  303. // If we've renamed the original stream for conversion, convert
  304. if (fRenamed)
  305. {
  306. olChkTo(EH_pdfExpInit, ConvertInternalStream(pdfExp));
  307. sc = STG_S_CONVERTED;
  308. }
  309. *ppv = pdfExp;
  310. }
  311. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::CreateEntry\n"));
  312. return sc;
  313. EH_pdfExpInit:
  314. pdfExp->Release();
  315. olVerSucc(DestroyEntry(&dfn, TRUE));
  316. EH_Err:
  317. return sc;
  318. }
  319. //+---------------------------------------------------------------------------
  320. //
  321. // Member: CExposedDocFile::OpenEntry, private
  322. //
  323. // Synopsis: Opens elements, used in OpenStream, OpenStorage and
  324. // for properties
  325. //
  326. // Arguments: [pwcsName] - Name
  327. // [dwType] - Entry type
  328. // [grfMode] - Access mode
  329. // [ppv] - Object return
  330. //
  331. // Returns: Appropriate status code
  332. //
  333. // Modifies: [ppv]
  334. //
  335. //----------------------------------------------------------------------------
  336. SCODE CExposedDocFile::OpenEntry(WCHAR const *pwcsName,
  337. DWORD dwType,
  338. DWORD grfMode,
  339. void **ppv)
  340. {
  341. CDfName dfn;
  342. SCODE sc;
  343. CExposedDocFile *pdfExp;
  344. CExposedStream *pstExp;
  345. olDebugOut((DEB_ITRACE, "In CExposedDocFile::OpenEntry:%p("
  346. "%ws, %lX, %lX, %p)\n", this, pwcsName, dwType, grfMode, ppv));
  347. olChk(EnforceSingle(grfMode));
  348. dfn.Set(pwcsName);
  349. if (REAL_STGTY(dwType) == STGTY_STREAM)
  350. {
  351. olChk(GetExposedStream(&dfn, ModeToDFlags(grfMode), &pstExp));
  352. *ppv = pstExp;
  353. }
  354. else
  355. {
  356. olAssert(REAL_STGTY(dwType) == STGTY_STORAGE);
  357. olChk(GetExposedDocFile(&dfn, ModeToDFlags(grfMode), &pdfExp));
  358. *ppv = pdfExp;
  359. }
  360. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::OpenEntry\n"));
  361. return S_OK;
  362. EH_Err:
  363. return sc;
  364. }
  365. //+--------------------------------------------------------------
  366. //
  367. // Member: CExposedDocFile::CreateStream, public
  368. //
  369. // Synopsis: Creates a stream
  370. //
  371. // Arguments: [pwcsName] - Name
  372. // [grfMode] - Permissions
  373. // [reserved1]
  374. // [reserved2]
  375. // [ppstm] - Stream return
  376. //
  377. // Returns: Appropriate status code
  378. //
  379. // Modifies: [ppstm]
  380. //
  381. //---------------------------------------------------------------
  382. TSTDMETHODIMP CExposedDocFile::CreateStream(WCHAR const *pwcsName,
  383. DWORD grfMode,
  384. DWORD reserved1,
  385. DWORD reserved2,
  386. IStream **ppstm)
  387. {
  388. SCODE sc;
  389. olDebugOut((DEB_ITRACE, "In CExposedDocFile::CreateStream("
  390. "%ws, %lX, %lu, %lu, %p)\n", pwcsName, grfMode, reserved1,
  391. reserved2, ppstm));
  392. olLog(("%p::In CExposedDocFile::CreateStream(%ws, %lX, %lu, %lu, %p)\n",
  393. this, pwcsName, grfMode, reserved1, reserved2, ppstm));
  394. TRY
  395. {
  396. olChk(ValidateOutPtrBuffer(ppstm));
  397. *ppstm = NULL;
  398. olChk(CheckWName(pwcsName));
  399. if (reserved1 != 0 || reserved2 != 0)
  400. olErr(EH_Err, STG_E_INVALIDPARAMETER);
  401. olChk(VerifyPerms(grfMode));
  402. if (grfMode & (STGM_CONVERT | STGM_TRANSACTED | STGM_PRIORITY |
  403. STGM_DELETEONRELEASE))
  404. olErr(EH_Err, STG_E_INVALIDFUNCTION);
  405. olChk(Validate());
  406. olChk(CheckCopyTo());
  407. sc = CreateEntry(pwcsName, STGTY_STREAM, grfMode, (void **)ppstm);
  408. }
  409. CATCH(CException, e)
  410. {
  411. sc = e.GetErrorCode();
  412. }
  413. END_CATCH
  414. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::CreateStream => %p\n",
  415. *ppstm));
  416. EH_Err:
  417. olLog(("%p::Out CExposedDocFile::CreateStream(). "
  418. "*ppstm == %p, ret == %lx\n", this, SAFE_DREF(ppstm), sc));
  419. return sc;
  420. }
  421. //+--------------------------------------------------------------
  422. //
  423. // Member: CExposedDocFile::OpenStream, public
  424. //
  425. // Synopsis: Opens an existing stream
  426. //
  427. // Arguments: [pwcsName] - Name
  428. // [reserved1]
  429. // [grfMode] - Permissions
  430. // [reserved2]
  431. // [ppstm] - Stream return
  432. //
  433. // Returns: Appropriate status code
  434. //
  435. // Modifies: [ppstm]
  436. //
  437. //---------------------------------------------------------------
  438. TSTDMETHODIMP CExposedDocFile::OpenStream(WCHAR const *pwcsName,
  439. void *reserved1,
  440. DWORD grfMode,
  441. DWORD reserved2,
  442. IStream **ppstm)
  443. {
  444. SCODE sc;
  445. olDebugOut((DEB_ITRACE, "In CExposedDocFile::OpenStream("
  446. "%ws, %p, %lX, %lu, %p)\n", pwcsName, reserved1,
  447. grfMode, reserved2, ppstm));
  448. olLog(("%p::In CExposedDocFile::OpenStream(%ws, %lu %lX, %lu, %p)\n",
  449. this, pwcsName, reserved1, grfMode, reserved2, ppstm));
  450. TRY
  451. {
  452. olChk(ValidateOutPtrBuffer(ppstm));
  453. *ppstm = NULL;
  454. olChk(CheckWName(pwcsName));
  455. if (reserved1 != NULL || reserved2 != 0)
  456. olErr(EH_Err, STG_E_INVALIDPARAMETER);
  457. olChk(VerifyPerms(grfMode));
  458. if (grfMode & (STGM_TRANSACTED | STGM_PRIORITY |
  459. STGM_DELETEONRELEASE))
  460. olErr(EH_Err, STG_E_INVALIDFUNCTION);
  461. olChk(Validate());
  462. sc = OpenEntry(pwcsName, STGTY_STREAM, grfMode, (void **)ppstm);
  463. }
  464. CATCH(CException, e)
  465. {
  466. sc = e.GetErrorCode();
  467. }
  468. END_CATCH
  469. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::OpenStream => %p\n",
  470. SAFE_DREF(ppstm)));
  471. EH_Err:
  472. olLog(("%p::Out CExposedDocFile::OpenStream(). "
  473. "*ppstm == %p, ret == %lx\n", this, SAVE_DREF(ppstm), sc));
  474. return sc;
  475. }
  476. //+--------------------------------------------------------------
  477. //
  478. // Member: CExposedDocFile::CreateStorage, public
  479. //
  480. // Synopsis: Creates an embedded DocFile
  481. //
  482. // Arguments: [pwcsName] - Name
  483. // [grfMode] - Permissions
  484. // [reserved1]
  485. // [reserved2]
  486. // [ppstg] - New DocFile return
  487. //
  488. // Returns: Appropriate status code
  489. //
  490. // Modifies: [ppstg]
  491. //
  492. //---------------------------------------------------------------
  493. TSTDMETHODIMP CExposedDocFile::CreateStorage(WCHAR const *pwcsName,
  494. DWORD grfMode,
  495. DWORD reserved1,
  496. DWORD reserved2,
  497. IStorage **ppstg)
  498. {
  499. SCODE sc;
  500. olLog(("%p::In CExposedDocFile::CreateStorage(%ws, %lX, %lu, %lu, %p)\n",
  501. this, pwcsName, grfMode, reserved1, reserved2, ppstg));
  502. olDebugOut((DEB_ITRACE, "In CExposedDocFile::CreateStorage:%p("
  503. "%ws, %lX, %lu, %lu, %p)\n", this, pwcsName, grfMode,
  504. reserved1, reserved2, ppstg));
  505. TRY
  506. {
  507. olChk(ValidateOutPtrBuffer(ppstg));
  508. *ppstg = NULL;
  509. olChk(CheckWName(pwcsName));
  510. if (reserved1 != 0 || reserved2 != 0)
  511. olErr(EH_Err, STG_E_INVALIDPARAMETER);
  512. olChk(VerifyPerms(grfMode));
  513. if (grfMode & (STGM_PRIORITY | STGM_DELETEONRELEASE))
  514. olErr(EH_Err, STG_E_INVALIDFUNCTION);
  515. olChk(Validate());
  516. olChk(CheckCopyTo());
  517. sc = CreateEntry(pwcsName, STGTY_STORAGE, grfMode, (void **)ppstg);
  518. }
  519. CATCH(CException, e)
  520. {
  521. sc = e.GetErrorCode();
  522. }
  523. END_CATCH
  524. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::CreateStorage => %p\n",
  525. *ppstg));
  526. EH_Err:
  527. olLog(("%p::Out CExposedDocFile::CreateStorage(). "
  528. "*ppstg == %p, ret == %lX\n", this, *ppstg, sc));
  529. return sc;
  530. }
  531. //+--------------------------------------------------------------
  532. //
  533. // Member: CExposedDocFile::OpenStorage, public
  534. //
  535. // Synopsis: Gets an existing embedded DocFile
  536. //
  537. // Arguments: [pwcsName] - Name
  538. // [pstgPriority] - Priority reopens
  539. // [grfMode] - Permissions
  540. // [snbExclude] - Priority reopens
  541. // [reserved]
  542. // [ppstg] - DocFile return
  543. //
  544. // Returns: Appropriate status code
  545. //
  546. // Modifies: [ppstg]
  547. //
  548. //---------------------------------------------------------------
  549. TSTDMETHODIMP CExposedDocFile::OpenStorage(WCHAR const *pwcsName,
  550. IStorage *pstgPriority,
  551. DWORD grfMode,
  552. SNBW snbExclude,
  553. DWORD reserved,
  554. IStorage **ppstg)
  555. {
  556. SCODE sc;
  557. CExposedDocFile *pdfExp;
  558. olLog(("%p::In CExposedDocFile::OpenStorage(%ws, %p, %lX, %p, %lu, %p)\n",
  559. this, pwcsName, pstgPriority, grfMode, snbExclude, reserved,
  560. ppstg));
  561. olDebugOut((DEB_ITRACE, "In CExposedDocFile::OpenStorage:%p("
  562. "%ws, %p, %lX, %p, %lu, %p)\n", this, pwcsName, pstgPriority,
  563. grfMode, snbExclude, reserved, ppstg));
  564. TRY
  565. {
  566. #ifdef _UNICODE // for UNICODE API's we have to do validation,
  567. // else it had been done in the ascii layer
  568. olChk(CheckWName(pwcsName));
  569. #endif
  570. olChk(ValidateOutPtrBuffer(ppstg));
  571. *ppstg = NULL;
  572. if (reserved != 0)
  573. olErr(EH_Err, STG_E_INVALIDPARAMETER);
  574. olChk(VerifyPerms(grfMode));
  575. if (pstgPriority != NULL ||
  576. (grfMode & (STGM_PRIORITY | STGM_DELETEONRELEASE)))
  577. olErr(EH_Err, STG_E_INVALIDFUNCTION);
  578. olChk(Validate());
  579. if (snbExclude != NULL)
  580. olErr(EH_Err, STG_E_INVALIDPARAMETER);
  581. olChk(OpenEntry(pwcsName, STGTY_STORAGE, grfMode, (void **)&pdfExp));
  582. *ppstg = pdfExp;
  583. }
  584. CATCH(CException, e)
  585. {
  586. sc = e.GetErrorCode();
  587. }
  588. END_CATCH
  589. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::OpenStorage => %p\n",
  590. *ppstg));
  591. EH_Err:
  592. olLog(("%p::Out CExposedDocFile::OpenStorage(). "
  593. "*ppstg == %p, ret == %lX\n", this, SAFE_DREF(ppstg), sc));
  594. return sc;
  595. }
  596. //+---------------------------------------------------------------------------
  597. //
  598. // Member: CExposedDocFile::MakeCopyFlags, public
  599. //
  600. // Synopsis: Translates IID array into bit fields
  601. //
  602. // Arguments: [ciidExclude] - Count of IIDs
  603. // [rgiidExclude] - IIDs not to copy
  604. //
  605. // Returns: Appropriate status code
  606. //
  607. //----------------------------------------------------------------------------
  608. DWORD CExposedDocFile::MakeCopyFlags(DWORD ciidExclude,
  609. IID const *rgiidExclude)
  610. {
  611. DWORD dwCopyFlags;
  612. olDebugOut((DEB_ITRACE, "In CExposedDocFile::MakeCopyFlags(%lu, %p)\n",
  613. ciidExclude, rgiidExclude));
  614. // Copy everything by default
  615. dwCopyFlags = COPY_ALL;
  616. for (; ciidExclude > 0; ciidExclude--, rgiidExclude++)
  617. if (IsEqualIID(*rgiidExclude, IID_IStorage))
  618. dwCopyFlags &= ~COPY_STORAGES;
  619. else if (IsEqualIID(*rgiidExclude, IID_IStream))
  620. dwCopyFlags &= ~COPY_STREAMS;
  621. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::MakeCopyFlags\n"));
  622. return dwCopyFlags;
  623. }
  624. //+--------------------------------------------------------------
  625. //
  626. // Member: CExposedDocFile::CopyTo, public
  627. //
  628. // Synopsis: Makes a copy of a DocFile
  629. //
  630. // Arguments: [ciidExclude] - Length of rgiid array
  631. // [rgiidExclude] - Array of IIDs to exclude
  632. // [snbExclude] - Names to exclude
  633. // [pstgDest] - Parent of copy
  634. //
  635. // Returns: Appropriate status code
  636. //
  637. //---------------------------------------------------------------
  638. TSTDMETHODIMP CExposedDocFile::CopyTo(DWORD ciidExclude,
  639. IID const *rgiidExclude,
  640. SNBW snbExclude,
  641. IStorage *pstgDest)
  642. {
  643. SCODE sc;
  644. DWORD i;
  645. olLog(("%p::In CExposedDocFile::CopyTo(%lu, %p, %p, %p)\n",
  646. this, ciidExclude, rgiidExclude, snbExclude, pstgDest));
  647. olDebugOut((DEB_ITRACE, "In CExposedDocFile::Copy(%lu, %p, %p, %p)\n",
  648. ciidExclude, rgiidExclude, snbExclude, pstgDest));
  649. TRY
  650. {
  651. if (snbExclude)
  652. olChk(ValidateSNBW(snbExclude));
  653. olChk(ValidateInterface(pstgDest, IID_IStorage));
  654. if (rgiidExclude)
  655. {
  656. olAssert(sizeof(IID)*ciidExclude <= 0xffffUL);
  657. olChk(ValidateBuffer(rgiidExclude,
  658. (size_t)(sizeof(IID)*ciidExclude)));
  659. for (i = 0; i<ciidExclude; i++)
  660. olChk(ValidateIid(rgiidExclude[i]));
  661. }
  662. olChk(Validate());
  663. olChk(CheckReverted());
  664. olAssert(_pdfb->GetCopyBase() == NULL);
  665. _pdfb->SetCopyBase(this);
  666. #ifdef NEWPROPS
  667. // Flush all descendant property set buffers so that their
  668. // underlying Streams (which are about to be copied) are
  669. // up to date.
  670. olChk(FlushBufferedData());
  671. #endif
  672. olChk(CopyDocFileToIStorage(GetDF(), pstgDest, snbExclude,
  673. MakeCopyFlags(ciidExclude, rgiidExclude)));
  674. }
  675. CATCH(CException, e)
  676. {
  677. sc = e.GetErrorCode();
  678. }
  679. END_CATCH
  680. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::Copy\n"));
  681. EH_Err:
  682. _pdfb->SetCopyBase(NULL);
  683. olLog(("%p::Out ExposedDocFile::CopyTo(). ret == %lX\n",this, sc));
  684. return sc;
  685. }
  686. //+--------------------------------------------------------------
  687. //
  688. // Member: CExposedDocFile::Commit, public
  689. //
  690. // Synopsis: Commits transacted changes
  691. //
  692. // Arguments: [dwFlags] - DFC_*
  693. //
  694. // Returns: Appropriate status code
  695. //
  696. //---------------------------------------------------------------
  697. STDMETHODIMP CExposedDocFile::Commit(DWORD dwFlags)
  698. {
  699. SCODE sc=S_OK;
  700. TIME_T tm;
  701. olLog(("%p::In CExposedDocFile::Commit(%lX)\n",this, dwFlags));
  702. olDebugOut((DEB_ITRACE, "In CExposedDocFile::Commit(%lX)\n", dwFlags));
  703. TRY
  704. {
  705. if (!VALID_COMMIT(dwFlags))
  706. olErr(EH_Err, STG_E_INVALIDFLAG);
  707. olChk(Validate());
  708. olChk(CheckReverted());
  709. if (!P_WRITE(_df)) olErr(EH_Err, STG_E_ACCESSDENIED);
  710. if (IsDirty())
  711. {
  712. olChk(DfGetTOD(&tm));
  713. olChk(_pdf->SetTime(WT_MODIFICATION, tm));
  714. olChk(_pmsBase->Flush(FLUSH_CACHE(dwFlags)));
  715. if (!IsRoot()) _pdfParent->SetDirty();
  716. }
  717. olChk(DfGetTOD(&tm));
  718. olChk(_pdf->SetTime(WT_ACCESS, tm));
  719. #ifdef NEWPROPS
  720. olChk(FlushBufferedData());
  721. #endif
  722. }
  723. CATCH(CException, e)
  724. {
  725. sc = e.GetErrorCode();
  726. }
  727. END_CATCH
  728. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::Commit\n"));
  729. EH_Err:
  730. olLog(("%p::Out CExposedDocFile::Commit(). ret == %lx\n",this, sc));
  731. return ResultFromScode(sc);
  732. }
  733. //+--------------------------------------------------------------
  734. //
  735. // Member: CExposedDocFile::Revert, public
  736. //
  737. // Synopsis: Reverts transacted changes
  738. //
  739. // Returns: S_OK - for direct mode files, this function
  740. // has not effect
  741. //
  742. //---------------------------------------------------------------
  743. STDMETHODIMP CExposedDocFile::Revert(void)
  744. {
  745. // we don't supported transacted files
  746. // it is stated in the OLE documentation that for direct
  747. // files, this method has no effect
  748. return ResultFromScode(S_OK);
  749. }
  750. //+--------------------------------------------------------------
  751. //
  752. // Member: CExposedDocFile::EnumElements, public
  753. //
  754. // Synopsis: Starts an iterator
  755. //
  756. // Arguments: [reserved1]
  757. // [reserved2]
  758. // [reserved3]
  759. // [ppenm] - Enumerator return
  760. //
  761. // Returns: Appropriate status code
  762. //
  763. // Modifies: [ppenm]
  764. //
  765. //---------------------------------------------------------------
  766. STDMETHODIMP CExposedDocFile::EnumElements(DWORD reserved1,
  767. void *reserved2,
  768. DWORD reserved3,
  769. IEnumSTATSTG **ppenm)
  770. {
  771. SCODE sc;
  772. CExposedIterator *pdiExp;
  773. CDfName dfnTemp;
  774. olLog(("%p::In CExposedDocFile::EnumElements(%lu, %p, %lu, %p)\n",
  775. this, reserved1, reserved2, reserved3, ppenm));
  776. olDebugOut((DEB_ITRACE, "In CExposedDocFile::EnumElements(%p)\n",
  777. ppenm));
  778. TRY
  779. {
  780. olChk(ValidateOutPtrBuffer(ppenm));
  781. *ppenm = NULL;
  782. if (reserved1 != 0 || reserved2 != NULL || reserved3 != 0)
  783. olErr(EH_Err, STG_E_INVALIDPARAMETER);
  784. olChk(Validate());
  785. olChk(CheckReverted());
  786. if (!P_READ(_df))
  787. olErr(EH_Err, STG_E_ACCESSDENIED);
  788. olMem(pdiExp = new CExposedIterator(this, &dfnTemp));
  789. *ppenm = pdiExp;
  790. }
  791. CATCH(CException, e)
  792. {
  793. sc = e.GetErrorCode();
  794. }
  795. END_CATCH
  796. EH_Err:
  797. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::EnumElements => %p\n",
  798. SAFE_DREF(ppenm)));
  799. olLog(("%p::Out CExposedDocFile::EnumElements(). ret == %lx\n",this, sc));
  800. return ResultFromScode(sc);
  801. }
  802. //+--------------------------------------------------------------
  803. //
  804. // Member: CExposedDocFile::DestroyElement, public
  805. //
  806. // Synopsis: Permanently deletes an element of a DocFile
  807. //
  808. // Arguments: [pwcsName] - Name of element
  809. //
  810. // Returns: Appropriate status code
  811. //
  812. //---------------------------------------------------------------
  813. TSTDMETHODIMP CExposedDocFile::DestroyElement(WCHAR const *pwcsName)
  814. {
  815. SCODE sc;
  816. CDfName dfn;
  817. olLog(("%p::In CExposedDocFile::DestroyElement(%ws)\n", this, pwcsName));
  818. olDebugOut((DEB_ITRACE, "In CExposedDocFile::DestroyElement(%ws)\n",
  819. pwcsName));
  820. TRY
  821. {
  822. olChk(CheckWName(pwcsName));
  823. olChk(Validate());
  824. dfn.Set(pwcsName);
  825. olChk(DestroyEntry(&dfn, FALSE));
  826. }
  827. CATCH(CException, e)
  828. {
  829. sc = e.GetErrorCode();
  830. }
  831. END_CATCH
  832. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::DestroyElement\n"));
  833. EH_Err:
  834. olLog(("%p::Out CExposedDocFile::DestroyElement(). ret == %lx\n",
  835. this, sc));
  836. return sc;
  837. }
  838. //+--------------------------------------------------------------
  839. //
  840. // Member: CExposedDocFile::MoveElementTo, public
  841. //
  842. // Synopsis: Move an element of a DocFile to an IStorage
  843. //
  844. // Arguments: [pwcsName] - Current name
  845. // [ptcsNewName] - New name
  846. //
  847. // Returns: Appropriate status code
  848. //
  849. // Algorithm: Open source as storage or stream (whatever works)
  850. // Create appropriate destination
  851. // Copy source to destination
  852. // Set create time of destination equal to create time of source
  853. // If appropriate, delete source
  854. //
  855. //---------------------------------------------------------------
  856. TSTDMETHODIMP CExposedDocFile::MoveElementTo(WCHAR const *pwcsName,
  857. IStorage *pstgParent,
  858. TCHAR const *ptcsNewName,
  859. DWORD grfFlags)
  860. {
  861. IUnknown *punksrc = NULL;
  862. SCODE sc;
  863. olLog(("%p::In CExposedDocFile::MoveElementTo(%ws, %p, %s, %lu)\n",
  864. this, pwcsName, pstgParent, ptcsNewName, grfFlags));
  865. olDebugOut((DEB_ITRACE, "In CExposedDocFile::MoveElementTo("
  866. "%ws, %p, %s, %lu)\n",
  867. pwcsName, pstgParent, ptcsNewName, grfFlags));
  868. TRY
  869. {
  870. IUnknown *punkdst;
  871. IStorage *pstgsrc;
  872. STATSTG statstg;
  873. olChk(CheckWName(pwcsName));
  874. olChk(Validate());
  875. olChk(VerifyMoveFlags(grfFlags));
  876. // determine source type (determine its type)
  877. sc = OpenStorage(pwcsName, (IStorage*)NULL,
  878. STGM_DIRECT| STGM_READ| STGM_SHARE_EXCLUSIVE,
  879. (SNBW)NULL, (DWORD)NULL, &pstgsrc);
  880. if (SUCCEEDED(sc))
  881. {
  882. HRESULT hr;
  883. // It's a storage
  884. punksrc = pstgsrc;
  885. IStorage *pstgdst;
  886. olHChkTo(EH_UnkSrc, pstgsrc->Stat(&statstg, STATFLAG_NONAME));
  887. hr = pstgParent->CreateStorage(ptcsNewName,
  888. STGM_DIRECT |
  889. STGM_WRITE |
  890. STGM_SHARE_EXCLUSIVE
  891. | STGM_FAILIFTHERE,
  892. 0, 0, &pstgdst);
  893. if (DfGetScode(hr) == STG_E_FILEALREADYEXISTS &&
  894. grfFlags == STGMOVE_COPY)
  895. {
  896. hr = pstgParent->OpenStorage(ptcsNewName,
  897. NULL,
  898. STGM_DIRECT |
  899. STGM_WRITE |
  900. STGM_SHARE_EXCLUSIVE,
  901. NULL,
  902. 0, &pstgdst);
  903. }
  904. olHChkTo(EH_UnkSrc, hr);
  905. punkdst = pstgdst;
  906. sc = DfGetScode(pstgsrc->CopyTo(0, NULL, NULL, pstgdst));
  907. }
  908. else if (sc == STG_E_FILENOTFOUND)
  909. {
  910. // Try opening it as a stream
  911. IStream *pstmsrc, *pstmdst;
  912. olChk(OpenStream(pwcsName, (void *)NULL,
  913. STGM_DIRECT | STGM_READ | STGM_SHARE_EXCLUSIVE,
  914. (DWORD)NULL, &pstmsrc));
  915. // It's a stream
  916. punksrc = pstmsrc;
  917. olHChkTo(EH_UnkSrc, pstmsrc->Stat(&statstg, STATFLAG_NONAME));
  918. olHChkTo(EH_UnkSrc,
  919. pstgParent->CreateStream(ptcsNewName,
  920. STGM_DIRECT |
  921. STGM_WRITE |
  922. STGM_SHARE_EXCLUSIVE |
  923. (grfFlags == STGMOVE_MOVE ?
  924. STGM_FAILIFTHERE :
  925. STGM_CREATE),
  926. 0, 0, &pstmdst));
  927. punkdst = pstmdst;
  928. ULARGE_INTEGER cb;
  929. ULISetLow (cb, 0xffffffff);
  930. ULISetHigh(cb, 0xffffffff);
  931. sc = DfGetScode(pstmsrc->CopyTo(pstmdst, cb, NULL, NULL));
  932. }
  933. else
  934. olChk(sc);
  935. punkdst->Release();
  936. if (SUCCEEDED(sc))
  937. {
  938. // Make destination create time match source create time
  939. // Note that we don't really care if this call succeeded.
  940. pstgParent->SetElementTimes(ptcsNewName, &statstg.ctime,
  941. NULL, NULL);
  942. if ((grfFlags & STGMOVE_COPY) == STGMOVE_MOVE)
  943. sc = DestroyElement(pwcsName);
  944. }
  945. if (FAILED(sc))
  946. {
  947. // The copy/move failed, so get rid of the partial result.
  948. pstgParent->DestroyElement(ptcsNewName);
  949. }
  950. }
  951. CATCH(CException, e)
  952. {
  953. sc = e.GetErrorCode();
  954. }
  955. END_CATCH
  956. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::MoveElementTo\n"));
  957. // Fall through
  958. EH_UnkSrc:
  959. if (punksrc)
  960. punksrc->Release();
  961. EH_Err:
  962. olLog(("%p::Out CExposedDocFile::MoveElementTo(). ret == %lx\n",
  963. this, sc));
  964. return sc;
  965. }
  966. //+--------------------------------------------------------------
  967. //
  968. // Member: CExposedDocFile::RenameElement, public
  969. //
  970. // Synopsis: Renames an element of a DocFile
  971. //
  972. // Arguments: [pwcsName] - Current name
  973. // [pwcsNewName] - New name
  974. //
  975. // Returns: Appropriate status code
  976. //
  977. //---------------------------------------------------------------
  978. TSTDMETHODIMP CExposedDocFile::RenameElement(WCHAR const *pwcsName,
  979. WCHAR const *pwcsNewName)
  980. {
  981. SCODE sc;
  982. CDfName dfnOld, dfnNew;
  983. olLog(("%p::In CExposedDocFile::RenameElement(%ws, %ws)\n",
  984. this, pwcsName, pwcsNewName));
  985. olDebugOut((DEB_ITRACE, "In CExposedDocFile::RenameElement(%ws, %ws)\n",
  986. pwcsName, pwcsNewName));
  987. TRY
  988. {
  989. olChk(Validate());
  990. olChk(CheckWName(pwcsName));
  991. olChk(CheckWName(pwcsNewName));
  992. dfnOld.Set(pwcsName);
  993. dfnNew.Set(pwcsNewName);
  994. olChk(RenameEntry(&dfnOld, &dfnNew));
  995. }
  996. CATCH(CException, e)
  997. {
  998. sc = e.GetErrorCode();
  999. }
  1000. END_CATCH
  1001. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::RenameElement\n"));
  1002. EH_Err:
  1003. olLog(("%p::Out CExposedDocFile::RenameElement(). ret == %lx\n",
  1004. this, sc));
  1005. return sc;
  1006. }
  1007. //+--------------------------------------------------------------
  1008. //
  1009. // Member: CExposedDocFile::SetElementTimes, public
  1010. //
  1011. // Synopsis: Sets element time stamps
  1012. //
  1013. // Arguments: [pwcsName] - Name
  1014. // [pctime] - create time
  1015. // [patime] - access time
  1016. // [pmtime] - modify time
  1017. //
  1018. // Returns: Appropriate status code
  1019. //
  1020. //---------------------------------------------------------------
  1021. TSTDMETHODIMP CExposedDocFile::SetElementTimes(WCHAR const *pwcsName,
  1022. FILETIME const *pctime,
  1023. FILETIME const *patime,
  1024. FILETIME const *pmtime)
  1025. {
  1026. SCODE sc;
  1027. CDfName dfn;
  1028. CDocFile *pdf;
  1029. olLog(("%p::In CExposedDocFile::SetElementTimes(%ws, %p, %p, %p)\n",
  1030. this, pwcsName, pctime, patime, pmtime));
  1031. olDebugOut((DEB_ITRACE, "In CExposedDocFile::SetElementTimes:%p("
  1032. "%ws, %p, %p, %p)\n", this, pwcsName, pctime, patime, pmtime));
  1033. TRY
  1034. {
  1035. if (pwcsName)
  1036. olChk(CheckWName(pwcsName));
  1037. else // function is meant to work on root storage
  1038. {
  1039. olAssert(FALSE &&
  1040. aMsg("SetTimes on root storage is not supported!\n"));
  1041. // SetElementTimes on root storage is not portable
  1042. // since it calls set filetimes.
  1043. return ResultFromScode(STG_E_UNIMPLEMENTEDFUNCTION);
  1044. }
  1045. olChk(Validate());
  1046. if (pctime)
  1047. olChk(ValidateBuffer(pctime, sizeof(FILETIME)));
  1048. if (patime)
  1049. olChk(ValidateBuffer(patime, sizeof(FILETIME)));
  1050. if (pmtime)
  1051. olChk(ValidateBuffer(pmtime, sizeof(FILETIME)));
  1052. dfn.Set(pwcsName);
  1053. olChk(CheckReverted());
  1054. if (!P_WRITE(_df) || _cilChildren.FindByName(&dfn) != NULL)
  1055. olErr(EH_Err, STG_E_ACCESSDENIED);
  1056. olChk(_pdf->GetDocFile(&dfn, DF_WRITE, &pdf));
  1057. if (pctime)
  1058. olChkTo(EH_pdf, pdf->SetTime(WT_CREATION, *pctime));
  1059. if (pmtime)
  1060. olChkTo(EH_pdf, pdf->SetTime(WT_MODIFICATION, *pmtime));
  1061. if (patime)
  1062. olChkTo(EH_pdf, pdf->SetTime(WT_ACCESS, *patime));
  1063. SetDirty();
  1064. }
  1065. CATCH(CException, e)
  1066. {
  1067. sc = e.GetErrorCode();
  1068. }
  1069. END_CATCH
  1070. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::SetElementTimes\n"));
  1071. // Fall thru
  1072. EH_pdf:
  1073. pdf->Release();
  1074. EH_Err:
  1075. olLog(("%p::Out CExposedDocFile::SetElementTimes(). ret == %lx\n",
  1076. this, sc));
  1077. return sc;
  1078. }
  1079. //+--------------------------------------------------------------
  1080. //
  1081. // Member: CExposedDocFile::SetClass, public
  1082. //
  1083. // Synopsis: Sets storage class
  1084. //
  1085. // Arguments: [clsid] - class id
  1086. //
  1087. // Returns: Appropriate status code
  1088. //
  1089. //---------------------------------------------------------------
  1090. STDMETHODIMP CExposedDocFile::SetClass(REFCLSID clsid)
  1091. {
  1092. SCODE sc;
  1093. olLog(("%p::In CExposedDocFile::SetClass(?)\n", this));
  1094. olDebugOut((DEB_ITRACE, "In CExposedDocFile::SetClass:%p(?)\n", this));
  1095. TRY
  1096. {
  1097. olChk(Validate());
  1098. olChk(ValidateBuffer(&clsid, sizeof(CLSID)));
  1099. olChk(CheckReverted());
  1100. if (!P_WRITE(_df))
  1101. olErr(EH_Err, STG_E_ACCESSDENIED);
  1102. olChk(_pdf->SetClass(clsid));
  1103. }
  1104. CATCH(CException, e)
  1105. {
  1106. sc = e.GetErrorCode();
  1107. }
  1108. END_CATCH
  1109. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::SetClass\n"));
  1110. EH_Err:
  1111. olLog(("%p::Out CExposedDocFile::SetClass(). ret == %lx\n",
  1112. this, sc));
  1113. return ResultFromScode(sc);
  1114. }
  1115. //+--------------------------------------------------------------
  1116. //
  1117. // Member: CExposedDocFile::SetStateBits, public
  1118. //
  1119. // Synopsis: Sets state bits
  1120. //
  1121. // Arguments: [grfStateBits] - state bits
  1122. // [grfMask] - state bits mask
  1123. //
  1124. // Returns: Appropriate status code
  1125. //
  1126. //---------------------------------------------------------------
  1127. STDMETHODIMP CExposedDocFile::SetStateBits(DWORD grfStateBits, DWORD grfMask)
  1128. {
  1129. SCODE sc;
  1130. olLog(("%p::In CExposedDocFile::SetStateBits(%lu, %lu)\n", this,
  1131. grfStateBits, grfMask));
  1132. olDebugOut((DEB_ITRACE, "In CExposedDocFile::SetStateBits:%p("
  1133. "%lu, %lu)\n", this, grfStateBits, grfMask));
  1134. TRY
  1135. {
  1136. olChk(Validate());
  1137. olChk(CheckReverted());
  1138. if (!P_WRITE(_df))
  1139. olErr(EH_Err, STG_E_ACCESSDENIED);
  1140. olChk(_pdf->SetStateBits(grfStateBits, grfMask));
  1141. SetDirty();
  1142. }
  1143. CATCH(CException, e)
  1144. {
  1145. sc = e.GetErrorCode();
  1146. }
  1147. END_CATCH
  1148. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::SetStateBits\n"));
  1149. // fall thru
  1150. EH_Err:
  1151. olLog(("%p::Out CExposedDocFile::SetStateBits(). ret == %lx\n",
  1152. this, sc));
  1153. return ResultFromScode(sc);
  1154. }
  1155. //+--------------------------------------------------------------
  1156. //
  1157. // Member: CExposedDocFile::Stat, public virtual
  1158. //
  1159. // Synopsis: Fills in a buffer of information about this object
  1160. //
  1161. // Arguments: [pstatstg] - Buffer
  1162. //
  1163. // Returns: Appropriate status code
  1164. //
  1165. // Modifies: [pstatstg]
  1166. //
  1167. //---------------------------------------------------------------
  1168. TSTDMETHODIMP CExposedDocFile::Stat(STATSTGW *pstatstg, DWORD grfStatFlag)
  1169. {
  1170. SCODE sc;
  1171. // root storage should be handled by virtual funcs in CRootExposedDocFile
  1172. olAssert(!IsRoot());
  1173. olLog(("%p::In CExposedDocFile::Stat(%p)\n", this, pstatstg));
  1174. olDebugOut((DEB_ITRACE, "In CExposedDocFile::Stat(%p)\n", pstatstg));
  1175. TRY
  1176. {
  1177. olChkTo(EH_RetSc, ValidateOutBuffer(pstatstg, sizeof(STATSTGW)));
  1178. olChk(VerifyStatFlag(grfStatFlag));
  1179. olChk(CheckReverted());
  1180. olChk(_pdf->GetTime(WT_CREATION, &pstatstg->ctime));
  1181. olChk(_pdf->GetTime(WT_MODIFICATION, &pstatstg->mtime));
  1182. pstatstg->atime.dwLowDateTime = pstatstg->atime.dwHighDateTime = 0;
  1183. olChk(_pdf->GetClass(&pstatstg->clsid));
  1184. olChk(_pdf->GetStateBits(&pstatstg->grfStateBits));
  1185. pstatstg->pwcsName = NULL;
  1186. if ((grfStatFlag & STATFLAG_NONAME) == 0)
  1187. {
  1188. olChk(DfAllocWCS((WCHAR *)_dfn.GetBuffer(), &pstatstg->pwcsName));
  1189. wcscpy(pstatstg->pwcsName, (WCHAR *)_dfn.GetBuffer());
  1190. }
  1191. pstatstg->grfMode = DFlagsToMode(_df);
  1192. pstatstg->type = STGTY_STORAGE;
  1193. ULISet32(pstatstg->cbSize, 0); // irelevant for storage obj
  1194. pstatstg->grfLocksSupported = 0;
  1195. pstatstg->reserved = 0;
  1196. }
  1197. CATCH(CException, e)
  1198. {
  1199. sc = e.GetErrorCode();
  1200. }
  1201. END_CATCH
  1202. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::Stat\n"));
  1203. EH_Err:
  1204. if (FAILED(sc))
  1205. memset(pstatstg, 0, sizeof(STATSTGW));
  1206. EH_RetSc:
  1207. olLog(("%p::Out CExposedDocFile::Stat(). *pstatstg == %p ret == %lx\n",
  1208. this, *pstatstg, sc));
  1209. return sc;
  1210. }
  1211. //+--------------------------------------------------------------
  1212. //
  1213. // Member: CExposedDocFile::AddRef, public
  1214. //
  1215. // Synopsis: Increments the ref count
  1216. //
  1217. // Returns: Appropriate status code
  1218. //
  1219. //---------------------------------------------------------------
  1220. STDMETHODIMP_(ULONG) CExposedDocFile::AddRef(void)
  1221. {
  1222. ULONG ulRet;
  1223. olLog(("%p::In CExposedDocFile::AddRef()\n", this));
  1224. olDebugOut((DEB_ITRACE, "In CExposedDocFile::AddRef()\n"));
  1225. TRY
  1226. {
  1227. if (FAILED(Validate()))
  1228. return 0;
  1229. AtomicInc(&_cReferences);
  1230. ulRet = _cReferences;
  1231. }
  1232. CATCH(CException, e)
  1233. {
  1234. UNREFERENCED_PARM(e);
  1235. ulRet = 0;
  1236. }
  1237. END_CATCH
  1238. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::AddRef\n"));
  1239. olLog(("%p::Out CExposedDocFile::AddRef(). ret == %lu\n", this, ulRet));
  1240. return ulRet;
  1241. }
  1242. //+--------------------------------------------------------------
  1243. //
  1244. // Member: CExposedDocFile::QueryInterface, public
  1245. //
  1246. // Synopsis: Returns an object for the requested interface
  1247. //
  1248. // Arguments: [iid] - Interface ID
  1249. // [ppvObj] - Object return
  1250. //
  1251. // Returns: Appropriate status code
  1252. //
  1253. // Modifies: [ppvObj]
  1254. //
  1255. //---------------------------------------------------------------
  1256. STDMETHODIMP CExposedDocFile::QueryInterface(REFIID iid, void **ppvObj)
  1257. {
  1258. SCODE sc;
  1259. olLog(("%p::In CExposedDocFile::QueryInterface(?, %p)\n",
  1260. this, ppvObj));
  1261. olDebugOut((DEB_ITRACE, "In CExposedDocFile::QueryInterface(?, %p)\n",
  1262. ppvObj));
  1263. TRY
  1264. {
  1265. olChk(ValidateOutPtrBuffer(ppvObj));
  1266. *ppvObj = NULL;
  1267. olChk(ValidateIid(iid));
  1268. olChk(Validate());
  1269. olChk(CheckReverted());
  1270. if ( IsEqualIID(iid, IID_IStorage) || IsEqualIID(iid, IID_IUnknown) )
  1271. {
  1272. olChk(CExposedDocFile::AddRef());
  1273. *ppvObj = (IStorage*) this;
  1274. }
  1275. #ifdef NEWPROPS
  1276. else if (IsEqualIID(iid, IID_IPropertySetStorage))
  1277. {
  1278. olChk(CExposedDocFile::AddRef());
  1279. *ppvObj = (IPropertySetStorage *) this;
  1280. }
  1281. #endif
  1282. else
  1283. olErr(EH_Err, E_NOINTERFACE);
  1284. sc = S_OK;
  1285. }
  1286. CATCH(CException, e)
  1287. {
  1288. sc = e.GetErrorCode();
  1289. }
  1290. END_CATCH
  1291. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::QueryInterface => %p\n",
  1292. ppvObj));
  1293. EH_Err:
  1294. olLog(("%p::Out CExposedDocFile::QueryInterface(). *ppvObj == %p ret == %lx\n",
  1295. this, SAFE_DREF(ppvObj), sc));
  1296. return ResultFromScode(sc);
  1297. }
  1298. //+--------------------------------------------------------------
  1299. //
  1300. // Method: CExposedDocFile::CopyDStreamToIStream, private
  1301. //
  1302. // Synopsis: Copies a CDirectStream to an IStream
  1303. //
  1304. // Arguments: [pstFrom] - CDirectStream
  1305. // [pstTo] - IStream
  1306. //
  1307. // Returns: Appropriate status code
  1308. //
  1309. //---------------------------------------------------------------
  1310. SCODE CExposedDocFile::CopyDStreamToIStream(CDirectStream *pstFrom,
  1311. IStream *pstTo)
  1312. {
  1313. BYTE *pbBuffer;
  1314. SCODE sc;
  1315. ULONG cbRead, cbWritten, cbPos, cbSizeLow;
  1316. ULARGE_INTEGER cbSize;
  1317. // This is part of CopyTo and therefore we are allowed to
  1318. // fail with out-of-memory
  1319. olMem(pbBuffer = new BYTE[STREAMBUFFERSIZE]);
  1320. // Set destination size for contiguity
  1321. pstFrom->GetSize(&cbSizeLow);
  1322. ULISet32(cbSize, cbSizeLow);
  1323. // Don't need to SetAccess0 here because pstTo is an IStream
  1324. olHChk(pstTo->SetSize(cbSize));
  1325. // Copy between streams
  1326. cbPos = 0;
  1327. for (;;)
  1328. {
  1329. olChk(pstFrom->ReadAt(cbPos, pbBuffer, STREAMBUFFERSIZE,
  1330. (ULONG STACKBASED *)&cbRead));
  1331. if (cbRead == 0) // EOF
  1332. break;
  1333. // Don't need to SetAccess0 here because pstTo is an IStream
  1334. olHChk(pstTo->Write(pbBuffer, cbRead, &cbWritten));
  1335. if (cbRead != cbWritten)
  1336. olErr(EH_Err, STG_E_WRITEFAULT);
  1337. cbPos += cbWritten;
  1338. }
  1339. sc = S_OK;
  1340. EH_Err:
  1341. delete [] pbBuffer;
  1342. return sc;
  1343. }
  1344. //+--------------------------------------------------------------
  1345. //
  1346. // Method: CExposedDocFile::CopyDocFileToIStorage, private
  1347. //
  1348. // Synopsis: Copies a docfile's contents to an IStorage
  1349. //
  1350. // Arguments: [pdfFrom] - From
  1351. // [pstgTo] - To
  1352. // [snbExclude] - Names to not copy
  1353. // [dwCopyFlags] - Bitwise flags for types of objects to copy
  1354. //
  1355. // Returns: Appropriate status code
  1356. //
  1357. //---------------------------------------------------------------
  1358. SCODE CExposedDocFile::CopyDocFileToIStorage(CDocFile *pdfFrom,
  1359. IStorage *pstgTo,
  1360. SNBW snbExclude,
  1361. DWORD dwCopyFlags)
  1362. {
  1363. PDocFileIterator *pdfi;
  1364. SIterBuffer ib;
  1365. CDirectStream *pstFrom;
  1366. IStream *pstTo;
  1367. CDocFile *pdfFromChild;
  1368. IStorage *pstgToChild;
  1369. SCODE sc;
  1370. TCHAR atcName[CWCSTORAGENAME];
  1371. CLSID clsid;
  1372. DWORD grfStateBits;
  1373. olDebugOut((DEB_ITRACE, "In CopyDocFileToIStorage:%p(%p, %p, %p, %lX)\n",
  1374. this, pdfFrom, pstgTo, snbExclude, dwCopyFlags));
  1375. olChk(pdfFrom->GetClass(&clsid));
  1376. olHChk(pstgTo->SetClass(clsid));
  1377. olChk(pdfFrom->GetStateBits(&grfStateBits));
  1378. olHChk(pstgTo->SetStateBits(grfStateBits, 0xffffffff));
  1379. olChk(pdfFrom->GetIterator(&pdfi));
  1380. for (;;)
  1381. {
  1382. sc = pdfi->BufferGetNext(&ib);
  1383. if (sc == STG_E_NOMOREFILES)
  1384. break;
  1385. else if (FAILED(sc))
  1386. olErr(EH_pdfi, sc);
  1387. if (snbExclude && NameInSNB(&ib.dfnName, snbExclude) == S_OK)
  1388. continue;
  1389. if ((ib.type == STGTY_STORAGE && (dwCopyFlags & COPY_STORAGES) == 0) ||
  1390. (ib.type == STGTY_STREAM && (dwCopyFlags & COPY_STREAMS) == 0)
  1391. )
  1392. continue;
  1393. switch(ib.type)
  1394. {
  1395. case STGTY_STORAGE:
  1396. // Embedded DocFile, create destination and recurse
  1397. sc = pdfFrom->GetDocFile(&ib.dfnName, DF_READ,
  1398. ib.type, &pdfFromChild);
  1399. olChkTo(EH_pdfi, sc);
  1400. WTOT(atcName, (WCHAR *)ib.dfnName.GetBuffer(), CWCSTORAGENAME);
  1401. // Don't need to SetAccess0 here because pstgTo is an IStorage.
  1402. sc = DfGetScode(pstgTo->CreateStorage(atcName, STGM_WRITE |
  1403. STGM_SHARE_EXCLUSIVE |
  1404. STGM_FAILIFTHERE,
  1405. 0, 0, &pstgToChild));
  1406. if (sc == STG_E_FILEALREADYEXISTS)
  1407. olHChkTo(EH_Get, pstgTo->OpenStorage(atcName, NULL,
  1408. STGM_WRITE |
  1409. STGM_SHARE_EXCLUSIVE,
  1410. NULL, 0, &pstgToChild));
  1411. else if (FAILED(sc))
  1412. olErr(EH_Get, sc);
  1413. olChkTo(EH_Create,
  1414. CopyDocFileToIStorage(pdfFromChild, pstgToChild, NULL,
  1415. dwCopyFlags));
  1416. pdfFromChild->Release();
  1417. pstgToChild->Release();
  1418. break;
  1419. case STGTY_STREAM:
  1420. sc = pdfFrom->GetStream(&ib.dfnName, DF_READ, ib.type, &pstFrom);
  1421. olChkTo(EH_pdfi, sc);
  1422. WTOT(atcName, (WCHAR *)ib.dfnName.GetBuffer(), CWCSTORAGENAME);
  1423. // Don't need to SetAccess0 here because pstgTo is an IStorage.
  1424. olHChkTo(EH_Get,
  1425. pstgTo->CreateStream(atcName, STGM_WRITE |
  1426. STGM_SHARE_EXCLUSIVE |
  1427. STGM_CREATE,
  1428. 0, 0, &pstTo));
  1429. olChkTo(EH_Create, CopyDStreamToIStream(pstFrom, pstTo));
  1430. pstFrom->Release();
  1431. pstTo->Release();
  1432. break;
  1433. default:
  1434. olAssert(!aMsg("Unknown type in CopyDocFileToIStorage"));
  1435. break;
  1436. }
  1437. }
  1438. pdfi->Release();
  1439. olDebugOut((DEB_ITRACE, "Out CopyDocFileToIStorage\n"));
  1440. return S_OK;
  1441. EH_Create:
  1442. if (ib.type == STGTY_STORAGE)
  1443. pstgToChild->Release();
  1444. else
  1445. pstTo->Release();
  1446. olVerSucc(pstgTo->DestroyElement(atcName));
  1447. EH_Get:
  1448. if (ib.type == STGTY_STORAGE)
  1449. pdfFromChild->Release();
  1450. else
  1451. pstFrom->Release();
  1452. EH_pdfi:
  1453. pdfi->Release();
  1454. EH_Err:
  1455. return sc;
  1456. }
  1457. //+---------------------------------------------------------------------------
  1458. //
  1459. // Member: CExposedDocFile::SwitchToFile, public
  1460. //
  1461. // Synopsis: Switches the underlying file to another file
  1462. //
  1463. // Arguments: [ptcsFile] - New file name
  1464. //
  1465. // Returns: Appropriate status code
  1466. //
  1467. //----------------------------------------------------------------------------
  1468. STDMETHODIMP CExposedDocFile::SwitchToFile(TCHAR *ptcsFile)
  1469. {
  1470. UNREFERENCED_PARM(ptcsFile);
  1471. olAssert(FALSE && aMsg("Unimplemented Function called!\n"));
  1472. return ResultFromScode(STG_E_UNIMPLEMENTEDFUNCTION);
  1473. }
  1474. //+---------------------------------------------------------------------------
  1475. //
  1476. // Member: CExposedDocFile::CreateExposedStream, private
  1477. //
  1478. // Synopsis: Creates an Exposed Stream
  1479. // This is a private function that creates the exposed stream.
  1480. // It is splitted out as a function so that the code can be
  1481. // reused.
  1482. //
  1483. // Arguments: [pdfnName] name of entry
  1484. // [df] doc file flags
  1485. // [ppStream] returned ExposedStream
  1486. //
  1487. // Returns: Appropriate status code
  1488. //
  1489. //----------------------------------------------------------------------------
  1490. SCODE CExposedDocFile::CreateExposedStream( CDfName const *pdfnName,
  1491. DFLAGS const df,
  1492. CExposedStream **ppStream)
  1493. {
  1494. CExposedStream *pstExp = NULL;
  1495. CDirectStream *pstDirect = NULL;
  1496. SCODE sc = S_OK;
  1497. olChk(CheckReverted());
  1498. if (!P_WRITE(_df)) olErr(EH_Err, STG_E_ACCESSDENIED);
  1499. olChk(_cilChildren.IsDenied(pdfnName, df, _df));
  1500. olChk(_pdf->CreateStream(pdfnName, df, DF_NOLUID, &pstDirect));
  1501. // As soon as we have a base we dirty ourself (in case
  1502. // we get an error later) so that we'll flush properly.
  1503. SetDirty();
  1504. olMemTo(EH_pst, pstExp = new CExposedStream());
  1505. olChkTo(EH_pstExp, pstExp->Init(pstDirect, this,
  1506. df, pdfnName, (ULONG)NULL));
  1507. *ppStream = pstExp;
  1508. return S_OK;
  1509. EH_pstExp:
  1510. delete pstExp;
  1511. EH_pst:
  1512. pstDirect->Release();
  1513. olVerSucc(DestroyEntry(pdfnName, TRUE));
  1514. EH_Err:
  1515. return sc;
  1516. }
  1517. //+---------------------------------------------------------------------------
  1518. //
  1519. // Member: CExposedDocFile::GetExposedStream, private
  1520. //
  1521. // Synopsis: Gets an existing exposed stream
  1522. // This is a private function that gets the exposed stream.
  1523. // It is splitted out as a function so that the code can be
  1524. // reused.
  1525. //
  1526. // Arguments: [pdfnName] name of entry
  1527. // [df] doc file flags
  1528. // [ppStream] returned ExposedStream
  1529. //
  1530. // Returns: Appropriate status code
  1531. //
  1532. //----------------------------------------------------------------------------
  1533. SCODE CExposedDocFile::GetExposedStream( CDfName const *pdfnName,
  1534. DFLAGS const df,
  1535. CExposedStream **ppStream)
  1536. {
  1537. CExposedStream *pstExp = NULL;
  1538. CDirectStream *pstDirect = NULL;
  1539. SCODE sc = S_OK;
  1540. olChk(CheckReverted());
  1541. if (!P_READ(_df)) olErr(EH_Err, STG_E_ACCESSDENIED);
  1542. // Check permissions
  1543. olChk(_cilChildren.IsDenied(pdfnName, df, _df));
  1544. olChk(_pdf->GetStream(pdfnName, df, DF_NOLUID, &pstDirect));
  1545. olMemTo(EH_pst, pstExp = new CExposedStream());
  1546. olChkTo(EH_pstExp, pstExp->Init(pstDirect, this,
  1547. df, pdfnName, (ULONG)NULL));
  1548. *ppStream = pstExp;
  1549. return S_OK;
  1550. EH_pstExp:
  1551. delete pstExp;
  1552. EH_pst:
  1553. pstDirect->Release();
  1554. EH_Err:
  1555. return sc;
  1556. }
  1557. //+--------------------------------------------------------------
  1558. //
  1559. // Member: CExposedDocFile::DestroyEntry, private
  1560. //
  1561. // Synopsis: Destroys an entry and removes it from the children
  1562. // list.
  1563. //
  1564. //---------------------------------------------------------------
  1565. SCODE CExposedDocFile::DestroyEntry( CDfName const *pdfnName,
  1566. BOOL fClean)
  1567. {
  1568. SCODE sc=S_OK;
  1569. olChk(CheckReverted());
  1570. if (!P_WRITE(_df)) olErr(EH_Err, STG_E_ACCESSDENIED);
  1571. olChk(_pdf->DestroyEntry(pdfnName, fClean));
  1572. _cilChildren.DeleteByName(pdfnName);
  1573. SetDirty();
  1574. // Fall through
  1575. EH_Err:
  1576. return sc;
  1577. }
  1578. //+--------------------------------------------------------------
  1579. //
  1580. // Member: CExposedDocFile::CreateExposedDocFile, private
  1581. //
  1582. // Synopsis: Creates an embedded DocFile
  1583. //
  1584. // Arguments: [pdfnName] - Name
  1585. // [df] - Permissions
  1586. // [ppdfDocFile] - New DocFile return
  1587. //
  1588. // Returns: Appropriate status code
  1589. //
  1590. // Modifies: [ppdfDocFile]
  1591. //
  1592. //---------------------------------------------------------------
  1593. SCODE CExposedDocFile::CreateExposedDocFile(CDfName const *pdfnName,
  1594. DFLAGS const df,
  1595. CExposedDocFile **ppdfDocFile)
  1596. {
  1597. SCODE sc;
  1598. CDocFile *pdf=NULL;
  1599. SEntryBuffer eb;
  1600. olChk(CheckReverted());
  1601. if (!P_WRITE(_df)) olErr(EH_Err, STG_E_ACCESSDENIED);
  1602. olChk(_cilChildren.IsDenied(pdfnName, df, _df));
  1603. olChkTo(EH_Reserve, _pdf->CreateDocFile(pdfnName, df, DF_NOLUID,
  1604. &pdf));
  1605. // As soon as we have a base we dirty ourself (in case
  1606. // we get an error later) so that we'll flush properly.
  1607. SetDirty();
  1608. eb.luid = pdf->GetLuid();
  1609. olAssert(eb.luid != DF_NOLUID && aMsg("DocFile id is DF_NOLUID!"));
  1610. olMemTo(EH_pdf,
  1611. *ppdfDocFile = new CExposedDocFile(this, pdf, df, eb.luid,
  1612. _pilbBase, pdfnName, _pmsBase, _pdfb));
  1613. return S_OK;
  1614. EH_pdf:
  1615. pdf->Release();
  1616. olVerSucc(_pdf->DestroyEntry(pdfnName, TRUE));
  1617. return sc;
  1618. EH_Reserve:
  1619. EH_Err:
  1620. return sc;
  1621. }
  1622. //+--------------------------------------------------------------
  1623. //
  1624. // Member: CExposedDocFile::GetExposedDocFile, private
  1625. //
  1626. // Synopsis: Retrieves an embedded DocFile
  1627. //
  1628. // Arguments: [pdfnName] - Name
  1629. // [df] - Permissions
  1630. // [ppdfDocFile] - New DocFile return
  1631. //
  1632. // Returns: Appropriate status code
  1633. //
  1634. // Modifies: [ppdfDocFile]
  1635. //
  1636. //---------------------------------------------------------------
  1637. SCODE CExposedDocFile::GetExposedDocFile( CDfName const *pdfnName,
  1638. DFLAGS const df,
  1639. CExposedDocFile **ppdfDocFile)
  1640. {
  1641. CDocFile *pdf;
  1642. SCODE sc;
  1643. SEntryBuffer eb;
  1644. olChk(CheckReverted());
  1645. if (!P_READ(_df)) olErr(EH_Err, STG_E_ACCESSDENIED);
  1646. // Check to see if an instance with DENY_* exists
  1647. olChk(_cilChildren.IsDenied(pdfnName, df, _df));
  1648. olChk(_pdf->GetDocFile(pdfnName, df, &pdf));
  1649. eb.luid = pdf->GetLuid();
  1650. olAssert(eb.luid != DF_NOLUID && aMsg("DocFile id is DF_NOLUID!"));
  1651. olMemTo(EH_pdf,
  1652. *ppdfDocFile = new CExposedDocFile(this, pdf, df, eb.luid,
  1653. _pilbBase, pdfnName, _pmsBase, _pdfb));
  1654. return S_OK;
  1655. EH_pdf:
  1656. pdf->Release();
  1657. EH_Err:
  1658. return sc;
  1659. }
  1660. //+--------------------------------------------------------------
  1661. //
  1662. // Member: CExposedDocFile::RenameEntry, public
  1663. //
  1664. // Synopsis: Renames an element of a DocFile
  1665. //
  1666. // Arguments: [pdfnName] - Current name
  1667. // [pdfnNewName] - New name
  1668. //
  1669. // Returns: Appropriate status code
  1670. //
  1671. //---------------------------------------------------------------
  1672. SCODE CExposedDocFile::RenameEntry(CDfName const *pdfnName,
  1673. CDfName const *pdfnNewName)
  1674. {
  1675. SCODE sc;
  1676. olChk(CheckReverted());
  1677. if (!P_WRITE(_df))
  1678. sc = STG_E_ACCESSDENIED;
  1679. else
  1680. {
  1681. sc = _pdf->RenameEntry(pdfnName, pdfnNewName);
  1682. if (SUCCEEDED(sc))
  1683. {
  1684. _cilChildren.RenameChild(pdfnName, pdfnNewName);
  1685. SetDirty();
  1686. }
  1687. }
  1688. // Fall through
  1689. EH_Err:
  1690. return sc;
  1691. }
  1692. //+--------------------------------------------------------------
  1693. //
  1694. // Member: CExposedDocFile::RevertFromAbove, public virtual from
  1695. // PRevertable
  1696. //
  1697. // Synopsis: Parent has asked for reversion
  1698. //
  1699. //---------------------------------------------------------------
  1700. void CExposedDocFile::RevertFromAbove(void)
  1701. {
  1702. olDebugOut((DEB_ITRACE, "In CExposedDocFile::RevertFromAbove:%p()\n", this));
  1703. _df |= DF_REVERTED;
  1704. _cilChildren.DeleteByName(NULL);
  1705. _pdf->Release();
  1706. #if DBG == 1
  1707. _pdf = NULL;
  1708. #endif
  1709. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::RevertFromAbove\n"));
  1710. }
  1711. //+--------------------------------------------------------------
  1712. //
  1713. // Member: CExposedDocFile::IsAtOrAbove, public
  1714. //
  1715. // Synopsis: Determines whether the given docfile is an ancestor
  1716. // of this docfile
  1717. //
  1718. // Arguments: [pdf] - Docfile to check
  1719. //
  1720. // Returns: TRUE or FALSE
  1721. //
  1722. //---------------------------------------------------------------
  1723. BOOL CExposedDocFile::IsAtOrAbove(CExposedDocFile *pdf)
  1724. {
  1725. CExposedDocFile *pdfPar = this;
  1726. olAssert(SUCCEEDED(CheckReverted()));
  1727. do
  1728. {
  1729. if (pdfPar == pdf)
  1730. break;
  1731. }
  1732. while (pdfPar = pdfPar->_pdfParent);
  1733. return pdfPar == pdf;
  1734. }
  1735. #ifdef NEWPROPS
  1736. //+---------------------------------------------------------------------------
  1737. //
  1738. // Member: CExposedDocFile::GetStorage, public IPrivateStorage
  1739. //
  1740. // Synopsis: Returns the IStorage for this object.
  1741. //
  1742. // Notes: This member is called by CPropertyStorage.
  1743. //
  1744. //----------------------------------------------------------------------------
  1745. STDMETHODIMP_(IStorage *)
  1746. CExposedDocFile::GetStorage(VOID)
  1747. {
  1748. return this;
  1749. }
  1750. //+--------------------------------------------------------------
  1751. //
  1752. // Member: CExposedDocFile::FlushBufferedData
  1753. // : public, virtual : PRevertable
  1754. //
  1755. // Synopsis: Flush buffered data in any child streams.
  1756. //
  1757. //---------------------------------------------------------------
  1758. SCODE CExposedDocFile::FlushBufferedData(void)
  1759. {
  1760. SCODE sc;
  1761. olDebugOut((DEB_ITRACE,
  1762. "In CExposedDocFile::FlushBufferedData:%p()\n", this));
  1763. sc = _cilChildren.FlushBufferedData();
  1764. olDebugOut((DEB_ITRACE, "Out CExposedDocFile::FlushBufferedData\n"));
  1765. return sc;
  1766. }
  1767. #endif // #ifdef NEWPROPS