Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

729 lines
22 KiB

  1. //+--------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation, 1992 - 1992.
  5. //
  6. // File: rpubdf.cxx
  7. //
  8. // Contents: CRootPubDocFile implementation
  9. //
  10. // History: 26-Aug-92 DrewB Created
  11. //
  12. //---------------------------------------------------------------
  13. #include <dfhead.cxx>
  14. #pragma hdrstop
  15. #include <header.hxx>
  16. #include <rpubdf.hxx>
  17. #include <lock.hxx>
  18. #include <filelkb.hxx>
  19. // Priority mode lock permissions
  20. #define PRIORITY_PERMS DF_READ
  21. //+--------------------------------------------------------------
  22. //
  23. // Member: CRootPubDocFile::CRootPubDocFile, public
  24. //
  25. // Synopsis: Ctor - Initializes empty object
  26. //
  27. // History: 30-Mar-92 DrewB Created
  28. // 05-Sep-5 MikeHill Init _timeModifyAtCommit.
  29. //
  30. //
  31. //---------------------------------------------------------------
  32. CRootPubDocFile::CRootPubDocFile(IMalloc * const pMalloc) :
  33. _pMalloc(pMalloc),
  34. CPubDocFile(NULL, NULL, 0, ROOT_LUID, NULL, NULL, 0, NULL)
  35. {
  36. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::CRootPubDocFile()\n"));
  37. _ulPriLock = 0;
  38. // Default to an invalid value.
  39. _timeModifyAtCommit.dwLowDateTime = _timeModifyAtCommit.dwHighDateTime = (DWORD) -1L;
  40. _sig = CROOTPUBDOCFILE_SIG;
  41. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::CRootPubDocFile\n"));
  42. }
  43. //+--------------------------------------------------------------
  44. //
  45. // Member: CRootPubDocFile::InitInd, private
  46. //
  47. // Synopsis: Initializes independent root
  48. //
  49. // Arguments: [plstBase] - Base
  50. // [snbExclude] - Limited instantiation exclusions
  51. // [dwStartFlags] - Startup flags
  52. // [df] - Transactioning flags
  53. //
  54. // Returns: Appropriate status code
  55. //
  56. // History: 11-Jun-92 DrewB Created
  57. //
  58. //---------------------------------------------------------------
  59. SCODE CRootPubDocFile::InitInd(ILockBytes *plstBase,
  60. SNBW snbExclude,
  61. DWORD const dwStartFlags,
  62. DFLAGS const df)
  63. {
  64. CFileStream *pfstCopy;
  65. ILockBytes *plkbCopy;
  66. ULONG ulLock = 0;
  67. CDocFile *pdfFrom, *pdfTo;
  68. SCODE sc;
  69. CMStream *pms;
  70. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::InitInd()\n"));
  71. if ((sc = DllGetCommitSig(plstBase, &_sigMSF)) == STG_E_INVALIDHEADER ||
  72. sc == STG_E_UNKNOWN)
  73. {
  74. _sigMSF = DF_INVALIDSIGNATURE;
  75. }
  76. else if (FAILED(sc))
  77. {
  78. olErr(EH_Err,sc);
  79. }
  80. olMem(pfstCopy = new (_pMalloc) CFileStream(_pMalloc));
  81. olChkTo(EH_pfstCopy, pfstCopy->InitGlobal(
  82. RSF_CREATE | RSF_DELETEONRELEASE | RSF_SNAPSHOT |
  83. (dwStartFlags & RSF_ENCRYPTED),
  84. DF_READWRITE));
  85. olChkTo(EH_pfstCopy, pfstCopy->InitSnapShot());
  86. if (!P_PRIORITY(df) && (_pdfb->GetOrigLockFlags() & LOCK_ONLYONCE))
  87. olChkTo(EH_pfstCopyInit, WaitForAccess(plstBase, DF_READ, &ulLock));
  88. if (snbExclude)
  89. {
  90. plkbCopy = pfstCopy;
  91. olChkTo(EH_GetAccess, DllMultiStreamFromStream(_pMalloc,
  92. &pms, &plstBase,
  93. dwStartFlags,
  94. df));
  95. olMemTo(EH_pmsFrom, pdfFrom = new (_pMalloc)
  96. CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb)));
  97. pdfFrom->AddRef();
  98. olChkTo(EH_pdfFrom, DllMultiStreamFromStream(_pMalloc,
  99. &pms, &plkbCopy,
  100. RSF_CREATE,
  101. 0));
  102. olMemTo(EH_pmsTo, pdfTo = new (_pMalloc)
  103. CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb)));
  104. pdfTo->AddRef();
  105. olChkTo(EH_pdfTo, pdfFrom->CopyTo(pdfTo, CDF_EXACT, snbExclude));
  106. olChkTo(EH_pdfTo, pms->Flush(0));
  107. pdfFrom->Release();
  108. pdfTo->Release();
  109. }
  110. else if ((dwStartFlags & RSF_TRUNCATE) == 0)
  111. {
  112. olChkTo(EH_GetAccess, CopyLStreamToLStream(plstBase, pfstCopy));
  113. }
  114. if (!P_PRIORITY(df) && ulLock != 0)
  115. ReleaseAccess(plstBase, DF_READ, ulLock);
  116. _pdfb->SetBase(pfstCopy);
  117. _pdfb->SetOriginal(plstBase);
  118. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::InitInd\n"));
  119. return S_OK;
  120. EH_pdfTo:
  121. pdfTo->Release();
  122. goto EH_pdfFrom;
  123. EH_pmsTo:
  124. DllReleaseMultiStream(pms);
  125. EH_pdfFrom:
  126. pdfFrom->Release();
  127. goto EH_GetAccess;
  128. EH_pmsFrom:
  129. DllReleaseMultiStream(pms);
  130. EH_GetAccess:
  131. if (!P_PRIORITY(df) && ulLock != 0)
  132. ReleaseAccess(plstBase, DF_READ, ulLock);
  133. EH_pfstCopyInit:
  134. EH_pfstCopy:
  135. olVerSucc(pfstCopy->Release());
  136. EH_Err:
  137. return sc;
  138. }
  139. //+--------------------------------------------------------------
  140. //
  141. // Member: CRootPubDocFile::InitNotInd, private
  142. //
  143. // Synopsis: Dependent root initialization
  144. //
  145. // Arguments: [plstBase] - Base
  146. // [snbExclude] - Limited instantiation exclusions
  147. // [dwStartFlags] - Startup flags
  148. //
  149. // Returns: Appropriate status code
  150. //
  151. // History: 11-Jun-92 DrewB Created
  152. //
  153. //---------------------------------------------------------------
  154. SCODE CRootPubDocFile::InitNotInd(ILockBytes *plstBase,
  155. SNBW snbExclude,
  156. DWORD const dwStartFlags,
  157. DFLAGS const df)
  158. {
  159. CDocFile *pdf;
  160. SCODE sc;
  161. CMStream *pms;
  162. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::InitNotInd()\n"));
  163. if (snbExclude)
  164. {
  165. olChk(DllMultiStreamFromStream(_pMalloc,
  166. &pms, &plstBase, dwStartFlags,
  167. df));
  168. olMemTo(EH_pms, pdf = new(_pMalloc)
  169. CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb)));
  170. pdf->AddRef();
  171. olChkTo(EH_pdf, PDocFile::ExcludeEntries(pdf, snbExclude));
  172. olChkTo(EH_pdf, pms->Flush(0));
  173. pdf->Release();
  174. }
  175. _pdfb->SetBase(plstBase);
  176. plstBase->AddRef();
  177. _pdfb->SetOriginal(plstBase);
  178. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::InitNotInd\n"));
  179. return S_OK;
  180. EH_pdf:
  181. //pdf->Release() will also release the multistream.
  182. pdf->Release();
  183. return sc;
  184. EH_pms:
  185. DllReleaseMultiStream(pms);
  186. EH_Err:
  187. return sc;
  188. }
  189. //+--------------------------------------------------------------
  190. //
  191. // Member: CRootPubDocFile::InitRoot, public
  192. //
  193. // Synopsis: Constructor
  194. //
  195. // Arguments: [plstBase] - Base LStream
  196. // [dwStartFlags] - How to start things
  197. // [df] - Transactioning flags
  198. // [snbExclude] - Parital instantiation list
  199. // [ppdfb] - Basis pointer return
  200. // [pulOpenLock] - Open lock index return
  201. //
  202. // Returns: Appropriate status code
  203. //
  204. // Modifies: [ppdfb]
  205. // [pulOpenLock]
  206. //
  207. // History: 09-Dec-91 DrewB Created
  208. // 09-Jun-92 PhilipLa Added conversion support
  209. // 05-Sep-95 MikeHill Initialize _timeModifyAtCommit.
  210. // Removed duplicate call to pdfWrapped->CopyTimesFrom
  211. //
  212. //---------------------------------------------------------------
  213. SCODE CRootPubDocFile::InitRoot(ILockBytes *plstBase,
  214. DWORD dwStartFlags,
  215. DFLAGS const df,
  216. SNBW snbExclude,
  217. CDFBasis **ppdfb,
  218. ULONG *pulOpenLock,
  219. CGlobalContext *pgc)
  220. {
  221. CWrappedDocFile *pdfWrapped;
  222. CDocFile *pdfBase;
  223. CFileStream *pfstScratch;
  224. CMStream *pmsScratch;
  225. SCODE sc, scConv = S_OK;
  226. STATSTG statstg;
  227. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::InitRoot("
  228. "%p, %lX, %lX, %p, %p)\n",
  229. plstBase, dwStartFlags, df, snbExclude, ppdfb));
  230. // Exclusion only works with a plain open
  231. olAssert(snbExclude == NULL ||
  232. (dwStartFlags & (RSF_CREATEFLAGS | RSF_CONVERT)) == 0);
  233. // ILockBytes::Stat calls are very expensive; we avoid one here
  234. // if possible
  235. HRESULT hr;
  236. IFileLockBytes *pfl;
  237. if (SUCCEEDED(plstBase->QueryInterface(IID_IFileLockBytes,
  238. (void**) &pfl)))
  239. {
  240. // This is our private ILockBytes implementation.
  241. hr = pfl->GetLocksSupported(&statstg.grfLocksSupported);
  242. if (pfl->IsEncryptedFile())
  243. dwStartFlags |= RSF_ENCRYPTED;
  244. pfl->Release();
  245. }
  246. else
  247. hr = plstBase->Stat(&statstg, STATFLAG_NONAME);
  248. olHChk(hr);
  249. *pulOpenLock = 0;
  250. if (statstg.grfLocksSupported & LOCK_ONLYONCE)
  251. olChk(GetOpen(plstBase, df, TRUE, pulOpenLock));
  252. if (P_PRIORITY(df) && (statstg.grfLocksSupported & LOCK_ONLYONCE))
  253. olChkTo(EH_GetOpen, GetAccess(plstBase, PRIORITY_PERMS, &_ulPriLock));
  254. olMemTo(EH_GetPriority, *ppdfb = new (_pMalloc) CDFBasis(_pMalloc, df,
  255. statstg.grfLocksSupported, pgc));
  256. _pdfb = P_TO_BP(CBasedDFBasisPtr, *ppdfb);
  257. if (P_INDEPENDENT(df))
  258. olChkTo(EH_GetPriority, InitInd(plstBase, snbExclude, dwStartFlags,
  259. df));
  260. else
  261. olChkTo(EH_GetPriority,
  262. InitNotInd(plstBase, snbExclude, dwStartFlags, df));
  263. olMemTo(EH_SubInit, pfstScratch = new (_pMalloc) CFileStream(_pMalloc));
  264. olChkTo(EH_pfstScratchInit, pfstScratch->InitGlobal(
  265. RSF_CREATE | RSF_DELETEONRELEASE | RSF_SCRATCH |
  266. (dwStartFlags & RSF_ENCRYPTED),
  267. DF_READWRITE));
  268. _pdfb->SetDirty(pfstScratch);
  269. CMStream *pms;
  270. scConv = DllMultiStreamFromStream(_pMalloc,
  271. &pms, _pdfb->GetPBase(),
  272. dwStartFlags |
  273. ((!P_INDEPENDENT(df) &&
  274. P_TRANSACTED(df)) ? RSF_DELAY : 0),
  275. df);
  276. _pmsBase = P_TO_BP(CBasedMStreamPtr, pms);
  277. if (scConv == STG_E_INVALIDHEADER)
  278. scConv = STG_E_FILEALREADYEXISTS;
  279. olChkTo(EH_pfstScratchInit, scConv);
  280. if (P_NOSNAPSHOT(df))
  281. {
  282. if ((sc = DllGetCommitSig(plstBase, &_sigMSF)) == STG_E_INVALIDHEADER ||
  283. sc == STG_E_UNKNOWN)
  284. {
  285. _sigMSF = DF_INVALIDSIGNATURE;
  286. }
  287. else if (FAILED(sc))
  288. {
  289. olErr(EH_pmsBase,sc);
  290. }
  291. }
  292. olMemTo(EH_pmsBase, pdfBase = new (_pMalloc)
  293. CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb)));
  294. pdfBase->AddRef();
  295. if (P_TRANSACTED(df))
  296. {
  297. _cTransactedDepth = 1;
  298. CDfName dfnNull; // auto-initialized to 0
  299. WCHAR wcZero = 0;
  300. dfnNull.Set(2, (BYTE*)&wcZero);
  301. // 3/11/93 - Demand scratch when opening/creating transacted
  302. olChkTo(EH_pdfBaseInit, _pdfb->GetDirty()->InitScratch());
  303. olMemTo(EH_pdfBaseInit, pdfWrapped = new(_pMalloc)
  304. CWrappedDocFile(&dfnNull, pdfBase->GetLuid(), df,
  305. BP_TO_P(CDFBasis *, _pdfb), this));
  306. olChkTo(EH_pdfWrapped,
  307. pdfWrapped->Init(pdfBase));
  308. AddXSMember(NULL, pdfWrapped, pdfWrapped->GetLuid());
  309. _pdf = P_TO_BP(CBasedDocFilePtr, (PDocFile *)pdfWrapped);
  310. }
  311. else
  312. _pdf = P_TO_BP(CBasedDocFilePtr, (PDocFile *)pdfBase);
  313. // For no-scratch transacted files, also save the Docfile's current modify
  314. // time. This will be used on the Release (in vdtor).
  315. if( P_NOSCRATCH( df ))
  316. {
  317. if( FAILED( _pmsBase->GetTime( SIDROOT, WT_MODIFICATION, &_timeModifyAtCommit )))
  318. {
  319. // Do not return an error, but record an error flag so that
  320. // vdtor will not try to use it.
  321. _timeModifyAtCommit.dwLowDateTime = _timeModifyAtCommit.dwHighDateTime = (DWORD) -1;
  322. }
  323. }
  324. olChkTo(EH_pfstScratchInit,
  325. DllGetScratchMultiStream(&pmsScratch,
  326. (df & DF_NOSCRATCH),
  327. (ILockBytes **)_pdfb->GetPDirty(),
  328. pms));
  329. _pdfb->SetScratch(pmsScratch);
  330. if (df & DF_NOSCRATCH)
  331. {
  332. _pdfb->SetBaseMultiStream(pms);
  333. olChkTo(EH_pfstScratchInit, pmsScratch->InitScratch(pms, TRUE));
  334. _pmsBase->SetScratchMS(pmsScratch);
  335. }
  336. else
  337. {
  338. _pdfb->SetBaseMultiStream(NULL);
  339. }
  340. _df = df;
  341. // _pdfb->mxs is constructed automatically
  342. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::InitRoot\n"));
  343. return scConv;
  344. EH_pdfWrapped:
  345. delete pdfWrapped;
  346. EH_pdfBaseInit:
  347. pdfBase->Release();
  348. goto EH_pfstScratchInit;
  349. EH_pmsBase:
  350. DllReleaseMultiStream(BP_TO_P(CMStream *, _pmsBase));
  351. EH_pfstScratchInit:
  352. olVerSucc(pfstScratch->Release());
  353. _pdfb->SetDirty(NULL);
  354. EH_SubInit:
  355. olVerSucc(_pdfb->GetBase()->Release());
  356. _pdfb->SetBase(NULL);
  357. EH_GetPriority:
  358. if (_ulPriLock > 0)
  359. {
  360. olAssert(P_PRIORITY(df) &&
  361. (statstg.grfLocksSupported & LOCK_ONLYONCE));
  362. ReleaseAccess(plstBase, PRIORITY_PERMS, _ulPriLock);
  363. _ulPriLock = 0;
  364. }
  365. EH_GetOpen:
  366. if (*pulOpenLock != 0)
  367. {
  368. olAssert(statstg.grfLocksSupported & LOCK_ONLYONCE);
  369. ReleaseOpen(plstBase, df, *pulOpenLock);
  370. *pulOpenLock = 0;
  371. }
  372. EH_Err:
  373. return sc;
  374. }
  375. //+--------------------------------------------------------------
  376. //
  377. // Member: CRootPubDocFile::~CRootPubDocFile, public
  378. //
  379. // Synopsis: dtor
  380. //
  381. // History: 09-Dec-91 DrewB Created
  382. // 05-Sep-95 MikeHill Revert time using _timeModifyAtCommit.
  383. //
  384. //---------------------------------------------------------------
  385. void CRootPubDocFile::vdtor(void)
  386. {
  387. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::~CRootPubDocFile\n"));
  388. olAssert(_cReferences == 0);
  389. // If this is a no-scratch transacted file, revert the Modify timestamp
  390. // on the Docfile to that of the last commit.
  391. if( P_NOSCRATCH( _df )
  392. &&
  393. ( _timeModifyAtCommit.dwLowDateTime != -1L ||
  394. _timeModifyAtCommit.dwHighDateTime != -1L ) // Don't use an invalid timestamp.
  395. )
  396. {
  397. TIME_T timeModify;
  398. BOOL fSetTime = TRUE; // set the modify time only if it's changed
  399. if(SUCCEEDED(_pmsBase->GetTime( SIDROOT, WT_MODIFICATION, &timeModify)))
  400. {
  401. if (timeModify.dwLowDateTime == _timeModifyAtCommit.dwLowDateTime &&
  402. timeModify.dwHighDateTime == _timeModifyAtCommit.dwHighDateTime)
  403. fSetTime = FALSE;
  404. }
  405. // We call SetFileLockBytesTime, rather than SetTime, so that
  406. // the underlying Docfile's timestamp is changed, but the Storage's
  407. // timestamp in the Directory is unchanged. If we changed the
  408. // Directory, we would have to flush the Multi-Stream.
  409. // An error here is ignored.
  410. if (fSetTime)
  411. _pmsBase->SetFileLockBytesTime( WT_MODIFICATION, _timeModifyAtCommit );
  412. }
  413. // We can't rely on CPubDocFile::~CPubDocFile to do this since
  414. // we're using a virtual destructor
  415. _sig = CROOTPUBDOCFILE_SIGDEL;
  416. if (SUCCEEDED(CheckReverted()))
  417. {
  418. ChangeXs(DF_NOLUID, XSO_RELEASE);
  419. _cilChildren.DeleteByName(NULL);
  420. if (_ulPriLock > 0)
  421. {
  422. // Priority instantiation can't be independent
  423. olAssert(!P_INDEPENDENT(_df));
  424. ReleaseAccess(_pdfb->GetBase(), PRIORITY_PERMS, _ulPriLock);
  425. }
  426. if (_pdf)
  427. _pdf->Release();
  428. if (_pdfb)
  429. _pdfb->vRelease();
  430. }
  431. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::~CRootPubDocFile\n"));
  432. delete this;
  433. }
  434. //+---------------------------------------------------------------------------
  435. //
  436. // Member: CRootPubDocFile::ReleaseLocks, public
  437. //
  438. // Synopsis: Release any locks using the given ILockBytes
  439. //
  440. // Arguments: [plkb] -- ILockBytes to use for release
  441. //
  442. // Returns: void
  443. //
  444. // History: 24-Jan-95 PhilipLa Created
  445. //
  446. // Notes: This is a cleanup function used to resolve the many
  447. // conflicts we get trying to release locks using an
  448. // ILockBytes in a basis that's already been released.
  449. //
  450. //----------------------------------------------------------------------------
  451. void CRootPubDocFile::ReleaseLocks(ILockBytes *plkb)
  452. {
  453. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::ReleaseLocks:%p()\n", this));
  454. if (_ulPriLock > 0)
  455. {
  456. // Priority instantiation can't be independent
  457. olAssert(!P_INDEPENDENT(_df));
  458. ReleaseAccess(plkb, PRIORITY_PERMS, _ulPriLock);
  459. _ulPriLock = 0;
  460. }
  461. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::ReleaseLocks\n"));
  462. }
  463. //+--------------------------------------------------------------
  464. //
  465. // Member: CRootPubDocFile::Stat, public
  466. //
  467. // Synopsis: Fills in a stat buffer from the base LStream
  468. //
  469. // Arguments: [pstatstg] - Stat buffer
  470. // [grfStatFlag] - Stat flags
  471. //
  472. // Returns: Appropriate status code
  473. //
  474. // Modifies: [pstatstg]
  475. //
  476. // History: 25-Mar-92 DrewB Created
  477. //
  478. //---------------------------------------------------------------
  479. SCODE CRootPubDocFile::Stat(STATSTGW *pstatstg, DWORD grfStatFlag)
  480. {
  481. SCODE sc;
  482. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::Stat(%p, %lu)\n",
  483. pstatstg, grfStatFlag));
  484. olChk(CheckReverted());
  485. olHChk(_pdfb->GetOriginal()->Stat((STATSTG *)pstatstg, grfStatFlag));
  486. pstatstg->grfMode = DFlagsToMode(_df);
  487. olChkTo(EH_pwcsName, _pdf->GetClass(&pstatstg->clsid));
  488. olChkTo(EH_pwcsName, _pdf->GetStateBits(&pstatstg->grfStateBits));
  489. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::Stat\n"));
  490. return S_OK;
  491. EH_pwcsName:
  492. if (pstatstg->pwcsName)
  493. {
  494. TaskMemFree(pstatstg->pwcsName);
  495. pstatstg->pwcsName = NULL;
  496. }
  497. EH_Err:
  498. return sc;
  499. }
  500. //+---------------------------------------------------------------------------
  501. //
  502. // Member: CRootPubDocFile::SwitchToFile, public
  503. //
  504. // Synopsis: Switches the underlying file in the base ILockBytes
  505. //
  506. // Arguments: [ptcsFile] - Filename
  507. // [plkb] - The ILockBytes to operate on
  508. // [pulOpenLock] - On entry, the current open lock
  509. // On exit, the new open lock
  510. //
  511. // Returns: Appropriate status code
  512. //
  513. // Modifies: [pulOpenLock]
  514. //
  515. // History: 08-Jan-93 DrewB Created
  516. //
  517. //----------------------------------------------------------------------------
  518. SCODE CRootPubDocFile::SwitchToFile(OLECHAR const *ptcsFile,
  519. ILockBytes *plkb,
  520. ULONG *pulOpenLock)
  521. {
  522. IFileLockBytes *pfl;
  523. SCODE sc;
  524. BYTE *pbBuffer;
  525. ULONG cbBuffer;
  526. olDebugOut((DEB_ITRACE, "In CRootPubDocFile::SwitchToFile:%p("
  527. "%s, %p, %p)\n", this, ptcsFile, plkb, pulOpenLock));
  528. // If you're transacted, nothing can be dirty in the base
  529. // If you're not dirty, there's no point in flushing
  530. // This is also necessary to allow SwitchToFile with a read-only source
  531. if (!P_TRANSACTED(_df) && IsDirty())
  532. {
  533. // Make sure pending changes are flushed
  534. olChk(_pmsBase->Flush(0));
  535. // Make sure ILockBytes contents are on disk
  536. olHChk(plkb->Flush());
  537. }
  538. #ifdef LARGE_DOCFILE
  539. ULONGLONG ulCommitSize;
  540. #else
  541. ULONG ulCommitSize;
  542. #endif
  543. olChk(GetCommitSize(&ulCommitSize));
  544. // Check for FileLockBytes
  545. olHChkTo(EH_NotFile, plkb->QueryInterface(IID_IFileLockBytes,
  546. (void **)&pfl));
  547. // Release old locks
  548. if (*pulOpenLock)
  549. ReleaseOpen(plkb, _df, *pulOpenLock);
  550. // Ask ILockBytes to switch
  551. GetSafeBuffer(CB_SMALLBUFFER, CB_LARGEBUFFER, &pbBuffer, &cbBuffer);
  552. olAssert(pbBuffer != NULL);
  553. sc = DfGetScode(pfl->SwitchToFile(
  554. ptcsFile,
  555. ulCommitSize,
  556. cbBuffer,
  557. pbBuffer));
  558. pfl->Release();
  559. FreeBuffer(pbBuffer);
  560. //Record the fact that we have enough space for overwrite commit.
  561. _wFlags = _wFlags | PF_PREPARED;
  562. // Attempt to get new locks
  563. // If SwitchToFile failed, the ILockBytes is the same so this will
  564. // restore our open locks released above
  565. // If SwitchToFile succeeded, the ILockBytes is working on the new file
  566. // so this will get locks for that
  567. if (*pulOpenLock)
  568. {
  569. ULONG ulLock;
  570. // Don't propagate failures here since there's nothing
  571. // that can be done
  572. if (SUCCEEDED(GetOpen(plkb, _df, FALSE, &ulLock)))
  573. *pulOpenLock = ulLock;
  574. }
  575. olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::SwitchToFile\n"));
  576. EH_Err:
  577. return sc;
  578. EH_NotFile:
  579. return(STG_E_NOTFILEBASEDSTORAGE);
  580. }
  581. //+--------------------------------------------------------------
  582. //
  583. // Member: CRootPubDocFile::Commit, public
  584. //
  585. // Synopsis: Commits transacted changes
  586. //
  587. // Arguments: [dwFlags] - DFC_*
  588. //
  589. // Returns: Appropriate status code
  590. //
  591. // History: 29-Aug-95 MikeHill Created
  592. // 26-Apr-99 RogerCh Changed to use GetSystemTimeAsFileTime
  593. //
  594. //---------------------------------------------------------------
  595. void CRootPubDocFile::CommitTimestamps(DWORD const dwFlags)
  596. {
  597. // For no-scratch transacted files, also save the Docfile's modify
  598. // time. This will be used to restore the file's current time on a Release.
  599. if( P_NOSCRATCH( _df ) && P_TRANSACTED( _df ))
  600. {
  601. GetSystemTimeAsFileTime(&_timeModifyAtCommit);
  602. }
  603. }