//+--------------------------------------------------------------------------- // // Microsoft Windows // Copyright (C) Microsoft Corporation, 1992 - 1994. // // File: simpstg2.cxx // // Contents: SimpStorageOpen class implementation // // Classes: CSimpStorageOpen, CSafeBYTEArray // // Functions: // // Notes: No error labels, tried to use destructors for cleanup // // History: 04-May-96 HenryLee Created // //---------------------------------------------------------------------------- #include "simphead.cxx" #pragma hdrstop #include //+--------------------------------------------------------------------------- // // Class: CSafeBYTEArray // // Purpose: automatically allocate & destroy an array of BYTEs // // Interface: // // History: 04-Jun-96 HenryLee Created // // Notes: destructor automatically cleans up // //---------------------------------------------------------------------------- class CSafeBYTEArray { public: inline CSafeBYTEArray (ULONG cBYTE) {_p = new BYTE[cBYTE]; }; inline ~CSafeBYTEArray () { if (_p) delete [] _p; }; inline operator BYTE* () { return _p; }; private: BYTE *_p; // allowed to be NULL }; //+--------------------------------------------------------------------------- // // Member: CSimpStorageOpen::Init, public // // Synopsis: Init function // // Arguments: [psdh] -- Pointer to hints structure // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // 14-Oct-97 HenryLee recoginize CNSS file format // //---------------------------------------------------------------------------- SCODE CSimpStorageOpen::Init(WCHAR const * pwcsName, DWORD grfMode, SSimpDocfileHints *psdh) { SCODE sc = S_OK; simpDebugOut((DEB_ITRACE, "In CSimpStorageOpen::Init:%p(%ws)\n", this, pwcsName)); #ifdef UNICODE TCHAR const *atcPath = pwcsName; #else TCHAR atcPath[_MAX_PATH+1]; UINT uCodePage = AreFileApisANSI() ? CP_ACP : CP_OEMCP; if (!WideCharToMultiByte( uCodePage, 0, pwcsName, -1, atcPath, _MAX_PATH + 1, NULL, NULL)) { return STG_E_INVALIDNAME; } #endif DWORD dwMode; switch (grfMode & (STGM_READ | STGM_WRITE | STGM_READWRITE)) { case STGM_READWRITE : dwMode = GENERIC_READ | GENERIC_WRITE; break; case STGM_READ : dwMode = GENERIC_READ; break; case STGM_WRITE : dwMode = GENERIC_WRITE; break; default: Win4Assert (!"Invalid mode"); return STG_E_INVALIDPARAMETER; } _hFile = CreateFileT(atcPath, dwMode, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (_hFile == INVALID_HANDLE_VALUE) { return Win32ErrorToScode(GetLastError()); } _grfMode = grfMode; _sectMax = 0; _fDirty = FALSE; _clsid = IID_NULL; _grfStateBits = 0; simpAssert (pwcsName != NULL); // CreateFileT would fail above if it was NULL sc = StringCbCopy(_awcsName, sizeof(_awcsName), pwcsName); if (FAILED(sc)) return sc; ULONG cbRead; BOOL f= ReadFile(_hFile, _hdr.GetData(), HEADERSIZE, &cbRead, NULL); if (!f) { return Win32ErrorToScode(GetLastError()); } if (cbRead != HEADERSIZE) { return STG_E_READFAULT; } if (!SUCCEEDED(sc = ValidateHeader(_hdr))) { return sc; } ULONG ulEndOfFile = GetFileSize(_hFile, NULL); if (ulEndOfFile == 0xFFFFFFFF && GetLastError() != NO_ERROR) return Win32ErrorToScode(GetLastError()); const ULONG ulFatStart=_hdr.GetFatStart()*SECTORSIZE + HEADERSIZE; const ULONG ulDifStart=_hdr.GetDifStart()*SECTORSIZE + HEADERSIZE; const ULONG ulFatLength = _hdr.GetFatLength()*SECTORSIZE; const ULONG ulDifLength = _hdr.GetDifLength()*SECTORSIZE; const ULONG ulDirLength = ulEndOfFile - ulFatStart - ulFatLength; const ULONG cBytes = ulDirLength + ulFatLength + ulDifLength; if (ulFatLength == 0 || ulDirLength == 0) return STG_E_DOCFILECORRUPT; DWORD dwErr; CSafeBYTEArray pByte (cBytes); if (pByte == NULL) return STG_E_INSUFFICIENTMEMORY; if ((dwErr = SetFilePointer (_hFile, (ulDifLength == 0 ? ulFatStart : ulDifStart), NULL, FILE_BEGIN)) == 0xFFFFFFFF) { return Win32ErrorToScode(GetLastError()); } // Read the FAT, DIFAT, and Directory into one big buffer // if (!(f=ReadFile(_hFile, pByte, cBytes, &cbRead, NULL))) { return Win32ErrorToScode(GetLastError()); } if (cbRead != cBytes) { return STG_E_READFAULT; } if (!SUCCEEDED(sc = ValidateDirectory( pByte+ulDifLength+ulFatLength, ulDirLength))) { return sc; } if (!SUCCEEDED(sc = ValidateFat ((SECT*) (pByte+ulDifLength), ulFatLength))) { return sc; } if (ulDifLength != 0 && !SUCCEEDED(sc = ValidateDIFat ((SECT *) (pByte+0), ulDifLength, _hdr.GetFatSect(CSECTFAT-1)))) { return sc; } simpDebugOut((DEB_ITRACE, "Out CSimpStorage::Init\n")); return sc; } //+--------------------------------------------------------------------------- // // Member: CSimpStorageOpen::ValidateHeader, public // // Synopsis: verifies header is in simple mode format // // Arguments: [hdr] -- reference to a docfile header // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // 14-Oct-97 HenryLee recoginize CNSS file format // //---------------------------------------------------------------------------- SCODE CSimpStorageOpen::ValidateHeader (CMSFHeader &hdr) { SCODE sc = S_OK; if (!SUCCEEDED(sc = hdr.Validate())) { return sc; } const SECT sectDifStart = hdr.GetDifStart(); const SECT sectFatStart = hdr.GetFatStart(); const SECT sectDirStart = hdr.GetDirStart(); // in simple mode, DifStart < FatStart < DirStart // if(hdr.GetMiniFatStart() != ENDOFCHAIN || hdr.GetMiniFatLength() != 0 || (sectDifStart != ENDOFCHAIN && sectDifStart >= sectFatStart)) { return STG_E_OLDFORMAT; } // in simple mode, DifStart+DifLength = FatStart // FatStart+FatLength = DirStart // // if (sectDifStart != ENDOFCHAIN && (sectDifStart + hdr.GetDifLength() != sectFatStart)) { return STG_E_OLDFORMAT; } if (sectDirStart != 0 && (sectFatStart + hdr.GetFatLength() != sectDirStart)) { return STG_E_OLDFORMAT; } // make sure the FAT is contiguous within the header // for (INT i=1; i < CSECTFAT; i++) { if (hdr.GetFatSect(i) == FREESECT) break; if (hdr.GetFatSect(i-1)+1 != hdr.GetFatSect(i)) return STG_E_OLDFORMAT; } return sc; } //+--------------------------------------------------------------------------- // // Member: CSimpStorageOpen::ValidateDirectory, public // // Synopsis: verifies stream entries are correct // // Arguments: // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // 14-Oct-97 HenryLee recoginize CNSS file format // //---------------------------------------------------------------------------- SCODE CSimpStorageOpen::ValidateDirectory (BYTE *pByte, ULONG ulDirLength) { SCODE sc = S_OK; CDfNameList *pdflRoot = _pdfl; SECT sectStartLowest = ENDOFCHAIN; ULONG ulSize = 0; ULONG cbStorages = 0; CDirEntry *pde = (CDirEntry *) pByte; // Read the directory entries until the end of buffer CDfNameList *pdflPrev = NULL; for (ULONG i=0; i < ulDirLength/sizeof(CDirEntry); i++) { if (!pde[i].IsFree()) { if (pde[i].GetFlags() != STGTY_ROOT && pde[i].GetFlags() != STGTY_STREAM && pde[i].GetFlags() != STGTY_STORAGE) return STG_E_OLDFORMAT; if (STORAGELIKE(pde[i].GetFlags())) { cbStorages++; // first entry must be a storage if (pdflPrev != NULL || cbStorages > 1) return STG_E_OLDFORMAT; } if (pde[i].GetRightSib() == (SID) i || pde[i].GetLeftSib() == (SID) i) return STG_E_DOCFILECORRUPT; CDfNameList *pdfl = new CDfNameList; if (pdfl != NULL) { pdfl->SetDfName(pde[i].GetName()); pdfl->SetStart(pde[i].GetStart()); if (sectStartLowest > pdfl->GetStart()) sectStartLowest = pdfl->GetStart(); #ifdef LARGE_STREAMS pdfl->SetSize((ULONG)pde[i].GetSize(FALSE)); #else pdfl->SetSize((ULONG)pde[i].GetSize()); #endif pdfl->Insert (&_pdfl, pdflPrev, NULL); //insert at end pdflPrev = pdfl; } else return STG_E_INSUFFICIENTMEMORY; } } pdflRoot = _pdfl; if (pdflRoot == 0 || pdflRoot->GetStart() != ENDOFCHAIN || pdflRoot->GetSize() != 0) { return STG_E_OLDFORMAT; } else pdflRoot = pdflRoot->GetNext(); // make sure streams are one after another // for (CDfNameList *pdfl = pdflRoot; pdfl != NULL; pdfl = pdfl->GetNext()) { // start should be after another stream's end // In the CNSS case, the pdflRoot points to 1st stream in file // In the docfile case, entry with 0 start sector is 1st stream in file if (pdfl->GetStart() != sectStartLowest) // skip 1st stream { CDfNameList *pdfl2 = NULL; for (pdfl2 = pdflRoot; pdfl2 != NULL; pdfl2=pdfl2->GetNext()) { if (pdfl->GetStart() == (pdfl2->GetStart() + ( pdfl2->GetSize()+SECTORSIZE-1)/SECTORSIZE)) break; } if (pdfl2 == NULL) // did not find a match return STG_E_OLDFORMAT; } } return sc; } //+--------------------------------------------------------------------------- // // Member: CSimpStorageOpen::ValidateFat, public // // Synopsis: verifies that stream sectors are contiguous // // Arguments: [pSect] array of Fat sectors // [ulFatLength] length of the Fat // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // //---------------------------------------------------------------------------- SCODE CSimpStorageOpen::ValidateFat (SECT *pSect, ULONG ulFatLength) { SCODE sc = S_OK; simpAssert (_pdfl != NULL && pSect != NULL); for (CDfNameList *pdfl = _pdfl->GetNext(); pdfl; pdfl = pdfl->GetNext()) { SECT sectStart = pdfl->GetStart(); ULONG ulSize = pdfl->GetSize(); SECT *psect = &pSect[sectStart]; SECT sectCount = sectStart+1; for (ULONG i = sectStart; i < sectStart + (ulSize+SECTORSIZE-1)/SECTORSIZE; i++) { if (*psect != sectCount && *psect != ENDOFCHAIN) return STG_E_OLDFORMAT; psect++; // check for sector numbers sectCount++; // increasing in order by 1 } if ((ULONG)(psect - pSect) > ulFatLength / sizeof(SECT)) { return STG_E_OLDFORMAT; } } return sc; } //+--------------------------------------------------------------------------- // // Member: CSimpStorageOpen::ValidateDIFat, public // // Synopsis: verifies that FAT sectors are contiguous // // Arguments: [pSect] array of DIFat sectors // [ulDIFatLength] length of the DIFat // [sectStart] last Fat sector in header // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // //---------------------------------------------------------------------------- SCODE CSimpStorageOpen::ValidateDIFat (SECT *pSect, ULONG ulDIFatLength, SECT sectStart) { SCODE sc = S_OK; simpAssert (pSect != NULL); simpAssert (sectStart != ENDOFCHAIN); SECT *psect = pSect; SECT sectCount = sectStart + 1; SECT iLastSect = SECTORSIZE / sizeof(SECT); for (ULONG i = 0; i < ulDIFatLength/sizeof(SECT); i++) { // skip last sector entry if (*psect != FREESECT && ((i+1) % iLastSect) != 0) { if (*psect != sectCount) return STG_E_OLDFORMAT; sectCount++; // check for sector numbers increasing by 1 } psect++; } return sc; } //+-------------------------------------------------------------- // // Member: CSimpStorageOpen::Release, public // // Synopsis: Releases resources for a CSimpStorageOpen // override CSimpStorage::Release because of delete this // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // //--------------------------------------------------------------- STDMETHODIMP_(ULONG) CSimpStorageOpen::Release(void) { simpDebugOut((DEB_TRACE, "In CSimpStorageOpen::Release()\n")); simpAssert(_cReferences > 0); LONG lRet = AtomicDec(&_cReferences); if (lRet == 0) { if (_fDirty) Commit(STGC_DEFAULT); CloseHandle(_hFile); // streams are not reverted delete this; } simpDebugOut((DEB_TRACE, "Out CSimpStorageOpen::Release()\n")); return (ULONG) lRet; } //+-------------------------------------------------------------- // // Member: CSimpStorageOpen::Stat, public // // Synopsis: Fills in a buffer of information about this object // // Arguments: [pstatstg] - Buffer // // Returns: Appropriate status code // // Modifies: [pstatstg] // // History: 04-May-96 HenryLee Created // //--------------------------------------------------------------- STDMETHODIMP CSimpStorageOpen::Stat(STATSTGW *pstatstg, DWORD grfStatFlag) { SCODE sc = S_OK; simpDebugOut((DEB_TRACE, "In CSimpStorageOpen::Stat(%p)\n", pstatstg)); SIMP_VALIDATE(Stat(pstatstg, grfStatFlag)); if (GetFileTime(_hFile, &pstatstg->ctime, &pstatstg->atime, &pstatstg->mtime) == FALSE) { return Win32ErrorToScode(GetLastError()); } if ((grfStatFlag & STATFLAG_NONAME) == 0) { DWORD cbName; cbName = (lstrlenW(_awcsName)+1)*sizeof(WCHAR); if ((pstatstg->pwcsName = (WCHAR*) CoTaskMemAlloc( cbName)) == 0) { return STG_E_INSUFFICIENTMEMORY; } StringCbCopy(pstatstg->pwcsName, cbName, _awcsName); } pstatstg->grfMode = _grfMode; pstatstg->clsid = _clsid; pstatstg->grfStateBits = _grfStateBits; pstatstg->type = STGTY_STORAGE; ULISet32(pstatstg->cbSize, 0); pstatstg->grfLocksSupported = 0; pstatstg->STATSTG_dwStgFmt = 0; simpDebugOut((DEB_TRACE, "Out CSimpStorageOpen::Stat\n")); return sc; } //+-------------------------------------------------------------- // // Member: CSimpStorageOpen::OpenStream, public // // Synopsis: Opens an existing stream // // Arguments: [pwcsName] - Name // [reserved1] // [grfMode] - Permissions // [reserved2] // [ppstm] - Stream return // // Returns: Appropriate status code // // Modifies: [ppstm] // // History: 04-May-96 HenryLee Created // //--------------------------------------------------------------- STDMETHODIMP CSimpStorageOpen::OpenStream(WCHAR const *pwcsName, void *reserved1, DWORD grfMode, DWORD reserved2, IStream **ppstm) { SCODE sc = S_OK; simpAssert (_pdfl != NULL); CDfNameList *pdflLoop = _pdfl->GetNext(); CDfName dfn; simpDebugOut((DEB_TRACE, "In CSimpStorageOpen:OpenStream(" "%ws, %p, %lX, %lu, %p)\n", pwcsName, reserved1, grfMode, reserved2, ppstm)); SIMP_VALIDATE(OpenStream(pwcsName, reserved1, grfMode, reserved2, ppstm)); if (_pdflCurrent != NULL) return STG_E_INVALIDFUNCTION; if (grfMode != (STGM_READWRITE | STGM_SHARE_EXCLUSIVE) && grfMode != (STGM_READ | STGM_SHARE_EXCLUSIVE)) return STG_E_INVALIDFLAG; if (_grfMode == (STGM_READ | STGM_SHARE_EXCLUSIVE) && grfMode == (STGM_READWRITE | STGM_SHARE_EXCLUSIVE)) return STG_E_ACCESSDENIED; dfn.Set(pwcsName); while (pdflLoop != NULL) { INT iCmp = CDirectory::NameCompare(&dfn, pdflLoop->GetName()); if (iCmp == 0) { //Found a stream with this name CSimpStreamOpen *pstm = new CSimpStreamOpen (); if (pstm != NULL) { _pdflCurrent = pdflLoop; if (!SUCCEEDED(sc = pstm->Init (this, _hFile, (_pdflCurrent->GetStart()+1)*SECTORSIZE, grfMode, _pdflCurrent))) { delete pstm; pstm = NULL; _pdflCurrent = NULL; } *ppstm = pstm; break; } else return STG_E_INSUFFICIENTMEMORY; } pdflLoop = pdflLoop->GetNext(); } if (pdflLoop == NULL) { sc = STG_E_FILENOTFOUND; } simpDebugOut((DEB_TRACE, "Out CSimpStorageOpen::OpenStream => %p\n", *ppstm)); return sc; } //+-------------------------------------------------------------- // // Member: CSimpStorageOpen::CreateStream, public // // Synopsis: stub // // History: 04-May-96 HenryLee Created // //--------------------------------------------------------------- STDMETHODIMP CSimpStorageOpen::CreateStream(WCHAR const *pwcsName, DWORD grfMode, DWORD reserved1, DWORD reserved2, IStream **ppstm) { simpDebugOut((DEB_TRACE, "Stb CSimpStorageOpen::CreateStream\n")); return STG_E_INVALIDFUNCTION; } //+-------------------------------------------------------------- // // Member: CSimpStorageOpen::EnumElements, public // // Synopsis: Starts an iterator // // Arguments: [reserved1] // [reserved2] // [reserved3] // [ppenm] - Enumerator return // // Returns: Appropriate status code // // Modifies: [ppenm] // // History: 04-May-96 HenryLee Created // //--------------------------------------------------------------- STDMETHODIMP CSimpStorageOpen::EnumElements(DWORD reserved1, void *reserved2, DWORD reserved3, IEnumSTATSTG **ppenm) { simpDebugOut((DEB_TRACE, "In CSimpStorageOpen::EnumElements\n")); SCODE sc = S_OK; SIMP_VALIDATE(EnumElements(reserved1, reserved2, reserved3, ppenm)); if ((*ppenm = new CSimpEnumSTATSTG (_pdfl, _pdfl)) == NULL) sc = STG_E_INSUFFICIENTMEMORY; simpDebugOut((DEB_TRACE, "Out CSimpStorageOpen::EnumElements => %x\n", sc)); return sc; } //+-------------------------------------------------------------- // // Member: CSimpStorageOpen::SetClass, public // // Synopsis: Sets storage class // // Arguments: [clsid] - class id // // Returns: Appropriate status code // // History: 04-May-96 HenryLee Created // //--------------------------------------------------------------- STDMETHODIMP CSimpStorageOpen::SetClass(REFCLSID rclsid) { simpDebugOut((DEB_TRACE, "Stb CSimpStorageOpen::SetClass\n")); return STG_E_INVALIDFUNCTION; }