//+-------------------------------------------------------------- // // Microsoft Windows // Copyright (C) Microsoft Corporation, 1992 - 1992. // // File: rpubdf.cxx // // Contents: CRootPubDocFile implementation // // History: 26-Aug-92 DrewB Created // //--------------------------------------------------------------- #include #pragma hdrstop #include #include #include #include // Priority mode lock permissions #define PRIORITY_PERMS DF_READ //+-------------------------------------------------------------- // // Member: CRootPubDocFile::CRootPubDocFile, public // // Synopsis: Ctor - Initializes empty object // // History: 30-Mar-92 DrewB Created // 05-Sep-5 MikeHill Init _timeModifyAtCommit. // // //--------------------------------------------------------------- CRootPubDocFile::CRootPubDocFile(IMalloc * const pMalloc) : _pMalloc(pMalloc), CPubDocFile(NULL, NULL, 0, ROOT_LUID, NULL, NULL, 0, NULL) { olDebugOut((DEB_ITRACE, "In CRootPubDocFile::CRootPubDocFile()\n")); _ulPriLock = 0; // Default to an invalid value. _timeModifyAtCommit.dwLowDateTime = _timeModifyAtCommit.dwHighDateTime = (DWORD) -1L; _sig = CROOTPUBDOCFILE_SIG; olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::CRootPubDocFile\n")); } //+-------------------------------------------------------------- // // Member: CRootPubDocFile::InitInd, private // // Synopsis: Initializes independent root // // Arguments: [plstBase] - Base // [snbExclude] - Limited instantiation exclusions // [dwStartFlags] - Startup flags // [df] - Transactioning flags // // Returns: Appropriate status code // // History: 11-Jun-92 DrewB Created // //--------------------------------------------------------------- SCODE CRootPubDocFile::InitInd(ILockBytes *plstBase, SNBW snbExclude, DWORD const dwStartFlags, DFLAGS const df) { CFileStream *pfstCopy; ILockBytes *plkbCopy; ULONG ulLock = 0; CDocFile *pdfFrom, *pdfTo; SCODE sc; CMStream *pms; olDebugOut((DEB_ITRACE, "In CRootPubDocFile::InitInd()\n")); if ((sc = DllGetCommitSig(plstBase, &_sigMSF)) == STG_E_INVALIDHEADER || sc == STG_E_UNKNOWN) { _sigMSF = DF_INVALIDSIGNATURE; } else if (FAILED(sc)) { olErr(EH_Err,sc); } olMem(pfstCopy = new (_pMalloc) CFileStream(_pMalloc)); olChkTo(EH_pfstCopy, pfstCopy->InitGlobal( RSF_CREATE | RSF_DELETEONRELEASE | RSF_SNAPSHOT | (dwStartFlags & RSF_ENCRYPTED), DF_READWRITE)); olChkTo(EH_pfstCopy, pfstCopy->InitSnapShot()); if (!P_PRIORITY(df) && (_pdfb->GetOrigLockFlags() & LOCK_ONLYONCE)) olChkTo(EH_pfstCopyInit, WaitForAccess(plstBase, DF_READ, &ulLock)); if (snbExclude) { plkbCopy = pfstCopy; olChkTo(EH_GetAccess, DllMultiStreamFromStream(_pMalloc, &pms, &plstBase, dwStartFlags, df)); olMemTo(EH_pmsFrom, pdfFrom = new (_pMalloc) CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb))); pdfFrom->AddRef(); olChkTo(EH_pdfFrom, DllMultiStreamFromStream(_pMalloc, &pms, &plkbCopy, RSF_CREATE, 0)); olMemTo(EH_pmsTo, pdfTo = new (_pMalloc) CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb))); pdfTo->AddRef(); olChkTo(EH_pdfTo, pdfFrom->CopyTo(pdfTo, CDF_EXACT, snbExclude)); olChkTo(EH_pdfTo, pms->Flush(0)); pdfFrom->Release(); pdfTo->Release(); } else if ((dwStartFlags & RSF_TRUNCATE) == 0) { olChkTo(EH_GetAccess, CopyLStreamToLStream(plstBase, pfstCopy)); } if (!P_PRIORITY(df) && ulLock != 0) ReleaseAccess(plstBase, DF_READ, ulLock); _pdfb->SetBase(pfstCopy); _pdfb->SetOriginal(plstBase); olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::InitInd\n")); return S_OK; EH_pdfTo: pdfTo->Release(); goto EH_pdfFrom; EH_pmsTo: DllReleaseMultiStream(pms); EH_pdfFrom: pdfFrom->Release(); goto EH_GetAccess; EH_pmsFrom: DllReleaseMultiStream(pms); EH_GetAccess: if (!P_PRIORITY(df) && ulLock != 0) ReleaseAccess(plstBase, DF_READ, ulLock); EH_pfstCopyInit: EH_pfstCopy: olVerSucc(pfstCopy->Release()); EH_Err: return sc; } //+-------------------------------------------------------------- // // Member: CRootPubDocFile::InitNotInd, private // // Synopsis: Dependent root initialization // // Arguments: [plstBase] - Base // [snbExclude] - Limited instantiation exclusions // [dwStartFlags] - Startup flags // // Returns: Appropriate status code // // History: 11-Jun-92 DrewB Created // //--------------------------------------------------------------- SCODE CRootPubDocFile::InitNotInd(ILockBytes *plstBase, SNBW snbExclude, DWORD const dwStartFlags, DFLAGS const df) { CDocFile *pdf; SCODE sc; CMStream *pms; olDebugOut((DEB_ITRACE, "In CRootPubDocFile::InitNotInd()\n")); if (snbExclude) { olChk(DllMultiStreamFromStream(_pMalloc, &pms, &plstBase, dwStartFlags, df)); olMemTo(EH_pms, pdf = new(_pMalloc) CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb))); pdf->AddRef(); olChkTo(EH_pdf, PDocFile::ExcludeEntries(pdf, snbExclude)); olChkTo(EH_pdf, pms->Flush(0)); pdf->Release(); } _pdfb->SetBase(plstBase); plstBase->AddRef(); _pdfb->SetOriginal(plstBase); olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::InitNotInd\n")); return S_OK; EH_pdf: //pdf->Release() will also release the multistream. pdf->Release(); return sc; EH_pms: DllReleaseMultiStream(pms); EH_Err: return sc; } //+-------------------------------------------------------------- // // Member: CRootPubDocFile::InitRoot, public // // Synopsis: Constructor // // Arguments: [plstBase] - Base LStream // [dwStartFlags] - How to start things // [df] - Transactioning flags // [snbExclude] - Parital instantiation list // [ppdfb] - Basis pointer return // [pulOpenLock] - Open lock index return // // Returns: Appropriate status code // // Modifies: [ppdfb] // [pulOpenLock] // // History: 09-Dec-91 DrewB Created // 09-Jun-92 PhilipLa Added conversion support // 05-Sep-95 MikeHill Initialize _timeModifyAtCommit. // Removed duplicate call to pdfWrapped->CopyTimesFrom // //--------------------------------------------------------------- SCODE CRootPubDocFile::InitRoot(ILockBytes *plstBase, DWORD dwStartFlags, DFLAGS const df, SNBW snbExclude, CDFBasis **ppdfb, ULONG *pulOpenLock, CGlobalContext *pgc) { CWrappedDocFile *pdfWrapped; CDocFile *pdfBase; CFileStream *pfstScratch; CMStream *pmsScratch; SCODE sc, scConv = S_OK; STATSTG statstg; olDebugOut((DEB_ITRACE, "In CRootPubDocFile::InitRoot(" "%p, %lX, %lX, %p, %p)\n", plstBase, dwStartFlags, df, snbExclude, ppdfb)); // Exclusion only works with a plain open olAssert(snbExclude == NULL || (dwStartFlags & (RSF_CREATEFLAGS | RSF_CONVERT)) == 0); // ILockBytes::Stat calls are very expensive; we avoid one here // if possible HRESULT hr; IFileLockBytes *pfl; if (SUCCEEDED(plstBase->QueryInterface(IID_IFileLockBytes, (void**) &pfl))) { // This is our private ILockBytes implementation. hr = pfl->GetLocksSupported(&statstg.grfLocksSupported); if (pfl->IsEncryptedFile()) dwStartFlags |= RSF_ENCRYPTED; pfl->Release(); } else hr = plstBase->Stat(&statstg, STATFLAG_NONAME); olHChk(hr); *pulOpenLock = 0; if (statstg.grfLocksSupported & LOCK_ONLYONCE) olChk(GetOpen(plstBase, df, TRUE, pulOpenLock)); if (P_PRIORITY(df) && (statstg.grfLocksSupported & LOCK_ONLYONCE)) olChkTo(EH_GetOpen, GetAccess(plstBase, PRIORITY_PERMS, &_ulPriLock)); olMemTo(EH_GetPriority, *ppdfb = new (_pMalloc) CDFBasis(_pMalloc, df, statstg.grfLocksSupported, pgc)); _pdfb = P_TO_BP(CBasedDFBasisPtr, *ppdfb); if (P_INDEPENDENT(df)) olChkTo(EH_GetPriority, InitInd(plstBase, snbExclude, dwStartFlags, df)); else olChkTo(EH_GetPriority, InitNotInd(plstBase, snbExclude, dwStartFlags, df)); olMemTo(EH_SubInit, pfstScratch = new (_pMalloc) CFileStream(_pMalloc)); olChkTo(EH_pfstScratchInit, pfstScratch->InitGlobal( RSF_CREATE | RSF_DELETEONRELEASE | RSF_SCRATCH | (dwStartFlags & RSF_ENCRYPTED), DF_READWRITE)); _pdfb->SetDirty(pfstScratch); CMStream *pms; scConv = DllMultiStreamFromStream(_pMalloc, &pms, _pdfb->GetPBase(), dwStartFlags | ((!P_INDEPENDENT(df) && P_TRANSACTED(df)) ? RSF_DELAY : 0), df); _pmsBase = P_TO_BP(CBasedMStreamPtr, pms); if (scConv == STG_E_INVALIDHEADER) scConv = STG_E_FILEALREADYEXISTS; olChkTo(EH_pfstScratchInit, scConv); if (P_NOSNAPSHOT(df)) { if ((sc = DllGetCommitSig(plstBase, &_sigMSF)) == STG_E_INVALIDHEADER || sc == STG_E_UNKNOWN) { _sigMSF = DF_INVALIDSIGNATURE; } else if (FAILED(sc)) { olErr(EH_pmsBase,sc); } } olMemTo(EH_pmsBase, pdfBase = new (_pMalloc) CDocFile(pms, SIDROOT, ROOT_LUID, BP_TO_P(CDFBasis *, _pdfb))); pdfBase->AddRef(); if (P_TRANSACTED(df)) { _cTransactedDepth = 1; CDfName dfnNull; // auto-initialized to 0 WCHAR wcZero = 0; dfnNull.Set(2, (BYTE*)&wcZero); // 3/11/93 - Demand scratch when opening/creating transacted olChkTo(EH_pdfBaseInit, _pdfb->GetDirty()->InitScratch()); olMemTo(EH_pdfBaseInit, pdfWrapped = new(_pMalloc) CWrappedDocFile(&dfnNull, pdfBase->GetLuid(), df, BP_TO_P(CDFBasis *, _pdfb), this)); olChkTo(EH_pdfWrapped, pdfWrapped->Init(pdfBase)); AddXSMember(NULL, pdfWrapped, pdfWrapped->GetLuid()); _pdf = P_TO_BP(CBasedDocFilePtr, (PDocFile *)pdfWrapped); } else _pdf = P_TO_BP(CBasedDocFilePtr, (PDocFile *)pdfBase); // For no-scratch transacted files, also save the Docfile's current modify // time. This will be used on the Release (in vdtor). if( P_NOSCRATCH( df )) { if( FAILED( _pmsBase->GetTime( SIDROOT, WT_MODIFICATION, &_timeModifyAtCommit ))) { // Do not return an error, but record an error flag so that // vdtor will not try to use it. _timeModifyAtCommit.dwLowDateTime = _timeModifyAtCommit.dwHighDateTime = (DWORD) -1; } } olChkTo(EH_pfstScratchInit, DllGetScratchMultiStream(&pmsScratch, (df & DF_NOSCRATCH), (ILockBytes **)_pdfb->GetPDirty(), pms)); _pdfb->SetScratch(pmsScratch); if (df & DF_NOSCRATCH) { _pdfb->SetBaseMultiStream(pms); olChkTo(EH_pfstScratchInit, pmsScratch->InitScratch(pms, TRUE)); _pmsBase->SetScratchMS(pmsScratch); } else { _pdfb->SetBaseMultiStream(NULL); } _df = df; // _pdfb->mxs is constructed automatically olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::InitRoot\n")); return scConv; EH_pdfWrapped: delete pdfWrapped; EH_pdfBaseInit: pdfBase->Release(); goto EH_pfstScratchInit; EH_pmsBase: DllReleaseMultiStream(BP_TO_P(CMStream *, _pmsBase)); EH_pfstScratchInit: olVerSucc(pfstScratch->Release()); _pdfb->SetDirty(NULL); EH_SubInit: olVerSucc(_pdfb->GetBase()->Release()); _pdfb->SetBase(NULL); EH_GetPriority: if (_ulPriLock > 0) { olAssert(P_PRIORITY(df) && (statstg.grfLocksSupported & LOCK_ONLYONCE)); ReleaseAccess(plstBase, PRIORITY_PERMS, _ulPriLock); _ulPriLock = 0; } EH_GetOpen: if (*pulOpenLock != 0) { olAssert(statstg.grfLocksSupported & LOCK_ONLYONCE); ReleaseOpen(plstBase, df, *pulOpenLock); *pulOpenLock = 0; } EH_Err: return sc; } //+-------------------------------------------------------------- // // Member: CRootPubDocFile::~CRootPubDocFile, public // // Synopsis: dtor // // History: 09-Dec-91 DrewB Created // 05-Sep-95 MikeHill Revert time using _timeModifyAtCommit. // //--------------------------------------------------------------- void CRootPubDocFile::vdtor(void) { olDebugOut((DEB_ITRACE, "In CRootPubDocFile::~CRootPubDocFile\n")); olAssert(_cReferences == 0); // If this is a no-scratch transacted file, revert the Modify timestamp // on the Docfile to that of the last commit. if( P_NOSCRATCH( _df ) && ( _timeModifyAtCommit.dwLowDateTime != -1L || _timeModifyAtCommit.dwHighDateTime != -1L ) // Don't use an invalid timestamp. ) { TIME_T timeModify; BOOL fSetTime = TRUE; // set the modify time only if it's changed if(SUCCEEDED(_pmsBase->GetTime( SIDROOT, WT_MODIFICATION, &timeModify))) { if (timeModify.dwLowDateTime == _timeModifyAtCommit.dwLowDateTime && timeModify.dwHighDateTime == _timeModifyAtCommit.dwHighDateTime) fSetTime = FALSE; } // We call SetFileLockBytesTime, rather than SetTime, so that // the underlying Docfile's timestamp is changed, but the Storage's // timestamp in the Directory is unchanged. If we changed the // Directory, we would have to flush the Multi-Stream. // An error here is ignored. if (fSetTime) _pmsBase->SetFileLockBytesTime( WT_MODIFICATION, _timeModifyAtCommit ); } // We can't rely on CPubDocFile::~CPubDocFile to do this since // we're using a virtual destructor _sig = CROOTPUBDOCFILE_SIGDEL; if (SUCCEEDED(CheckReverted())) { ChangeXs(DF_NOLUID, XSO_RELEASE); _cilChildren.DeleteByName(NULL); if (_ulPriLock > 0) { // Priority instantiation can't be independent olAssert(!P_INDEPENDENT(_df)); ReleaseAccess(_pdfb->GetBase(), PRIORITY_PERMS, _ulPriLock); } if (_pdf) _pdf->Release(); if (_pdfb) _pdfb->vRelease(); } olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::~CRootPubDocFile\n")); delete this; } //+--------------------------------------------------------------------------- // // Member: CRootPubDocFile::ReleaseLocks, public // // Synopsis: Release any locks using the given ILockBytes // // Arguments: [plkb] -- ILockBytes to use for release // // Returns: void // // History: 24-Jan-95 PhilipLa Created // // Notes: This is a cleanup function used to resolve the many // conflicts we get trying to release locks using an // ILockBytes in a basis that's already been released. // //---------------------------------------------------------------------------- void CRootPubDocFile::ReleaseLocks(ILockBytes *plkb) { olDebugOut((DEB_ITRACE, "In CRootPubDocFile::ReleaseLocks:%p()\n", this)); if (_ulPriLock > 0) { // Priority instantiation can't be independent olAssert(!P_INDEPENDENT(_df)); ReleaseAccess(plkb, PRIORITY_PERMS, _ulPriLock); _ulPriLock = 0; } olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::ReleaseLocks\n")); } //+-------------------------------------------------------------- // // Member: CRootPubDocFile::Stat, public // // Synopsis: Fills in a stat buffer from the base LStream // // Arguments: [pstatstg] - Stat buffer // [grfStatFlag] - Stat flags // // Returns: Appropriate status code // // Modifies: [pstatstg] // // History: 25-Mar-92 DrewB Created // //--------------------------------------------------------------- SCODE CRootPubDocFile::Stat(STATSTGW *pstatstg, DWORD grfStatFlag) { SCODE sc; olDebugOut((DEB_ITRACE, "In CRootPubDocFile::Stat(%p, %lu)\n", pstatstg, grfStatFlag)); olChk(CheckReverted()); olHChk(_pdfb->GetOriginal()->Stat((STATSTG *)pstatstg, grfStatFlag)); pstatstg->grfMode = DFlagsToMode(_df); olChkTo(EH_pwcsName, _pdf->GetClass(&pstatstg->clsid)); olChkTo(EH_pwcsName, _pdf->GetStateBits(&pstatstg->grfStateBits)); olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::Stat\n")); return S_OK; EH_pwcsName: if (pstatstg->pwcsName) { TaskMemFree(pstatstg->pwcsName); pstatstg->pwcsName = NULL; } EH_Err: return sc; } //+--------------------------------------------------------------------------- // // Member: CRootPubDocFile::SwitchToFile, public // // Synopsis: Switches the underlying file in the base ILockBytes // // Arguments: [ptcsFile] - Filename // [plkb] - The ILockBytes to operate on // [pulOpenLock] - On entry, the current open lock // On exit, the new open lock // // Returns: Appropriate status code // // Modifies: [pulOpenLock] // // History: 08-Jan-93 DrewB Created // //---------------------------------------------------------------------------- SCODE CRootPubDocFile::SwitchToFile(OLECHAR const *ptcsFile, ILockBytes *plkb, ULONG *pulOpenLock) { IFileLockBytes *pfl; SCODE sc; BYTE *pbBuffer; ULONG cbBuffer; olDebugOut((DEB_ITRACE, "In CRootPubDocFile::SwitchToFile:%p(" "%s, %p, %p)\n", this, ptcsFile, plkb, pulOpenLock)); // If you're transacted, nothing can be dirty in the base // If you're not dirty, there's no point in flushing // This is also necessary to allow SwitchToFile with a read-only source if (!P_TRANSACTED(_df) && IsDirty()) { // Make sure pending changes are flushed olChk(_pmsBase->Flush(0)); // Make sure ILockBytes contents are on disk olHChk(plkb->Flush()); } #ifdef LARGE_DOCFILE ULONGLONG ulCommitSize; #else ULONG ulCommitSize; #endif olChk(GetCommitSize(&ulCommitSize)); // Check for FileLockBytes olHChkTo(EH_NotFile, plkb->QueryInterface(IID_IFileLockBytes, (void **)&pfl)); // Release old locks if (*pulOpenLock) ReleaseOpen(plkb, _df, *pulOpenLock); // Ask ILockBytes to switch GetSafeBuffer(CB_SMALLBUFFER, CB_LARGEBUFFER, &pbBuffer, &cbBuffer); olAssert(pbBuffer != NULL); sc = DfGetScode(pfl->SwitchToFile( ptcsFile, ulCommitSize, cbBuffer, pbBuffer)); pfl->Release(); FreeBuffer(pbBuffer); //Record the fact that we have enough space for overwrite commit. _wFlags = _wFlags | PF_PREPARED; // Attempt to get new locks // If SwitchToFile failed, the ILockBytes is the same so this will // restore our open locks released above // If SwitchToFile succeeded, the ILockBytes is working on the new file // so this will get locks for that if (*pulOpenLock) { ULONG ulLock; // Don't propagate failures here since there's nothing // that can be done if (SUCCEEDED(GetOpen(plkb, _df, FALSE, &ulLock))) *pulOpenLock = ulLock; } olDebugOut((DEB_ITRACE, "Out CRootPubDocFile::SwitchToFile\n")); EH_Err: return sc; EH_NotFile: return(STG_E_NOTFILEBASEDSTORAGE); } //+-------------------------------------------------------------- // // Member: CRootPubDocFile::Commit, public // // Synopsis: Commits transacted changes // // Arguments: [dwFlags] - DFC_* // // Returns: Appropriate status code // // History: 29-Aug-95 MikeHill Created // 26-Apr-99 RogerCh Changed to use GetSystemTimeAsFileTime // //--------------------------------------------------------------- void CRootPubDocFile::CommitTimestamps(DWORD const dwFlags) { // For no-scratch transacted files, also save the Docfile's modify // time. This will be used to restore the file's current time on a Release. if( P_NOSCRATCH( _df ) && P_TRANSACTED( _df )) { GetSystemTimeAsFileTime(&_timeModifyAtCommit); } }