1321 lines
37 KiB
C++
1321 lines
37 KiB
C++
//+-------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1992.
|
|
//
|
|
// File: sstream.cxx
|
|
//
|
|
// Contents: Stream operations for Mstream project
|
|
//
|
|
// Classes: None. (defined in sstream.hxx)
|
|
//
|
|
// History: 18-Jul-91 PhilipLa Created.
|
|
// 24-Apr-92 AlexT Small object support
|
|
//
|
|
//--------------------------------------------------------------------
|
|
|
|
#include "msfhead.cxx"
|
|
|
|
#pragma hdrstop
|
|
|
|
#include <dirfunc.hxx>
|
|
#include <sstream.hxx>
|
|
#include <dl.hxx>
|
|
#include <tstream.hxx>
|
|
#include <time.h>
|
|
#include <mread.hxx>
|
|
|
|
#define DEB_STREAM (DEB_ITRACE | 0x00020000)
|
|
|
|
|
|
|
|
|
|
//+--------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::CDirectStream, public
|
|
//
|
|
// Synopsis: Empty object constructor
|
|
//
|
|
// Arguments: [dl] - LUID
|
|
//
|
|
// History: 25-Aug-92 DrewB Created
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_CDirectStream)
|
|
#endif
|
|
|
|
CDirectStream::CDirectStream(DFLUID dl)
|
|
: PSStream(dl)
|
|
{
|
|
_pdlHolder = NULL;
|
|
_cReferences = 0;
|
|
#ifdef SECURE_STREAMS
|
|
_ulSize = 0;
|
|
_ulHighWater = 0;
|
|
#endif
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Method: CDirectStream::~CDirectStream, public
|
|
//
|
|
// Synopsis: CDirectStream destructor
|
|
//
|
|
// History: 18-Jul-91 PhilipLa Created.
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CDirectStream::~CDirectStream()
|
|
{
|
|
msfAssert(_cReferences == 0);
|
|
#ifdef SECURE_STREAMS
|
|
if (_ulSize > _ulHighWater)
|
|
{
|
|
ClearSects(_ulHighWater, _ulSize);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
//+--------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::InitSystem, public
|
|
//
|
|
// Synopsis: Initializes special system streams like the ministream
|
|
//
|
|
// Arguments: [pms] - Multistream
|
|
// [sid] - SID
|
|
// [cbSize] - size
|
|
//
|
|
// History: 25-Aug-92 DrewB Created
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_InitSystem)
|
|
#endif
|
|
|
|
void CDirectStream::InitSystem(CMStream *pms,
|
|
SID sid,
|
|
ULONG cbSize)
|
|
{
|
|
_stmh.Init(pms, sid);
|
|
_ulSize = _ulOldSize = cbSize;
|
|
|
|
_stmc.Init(pms, sid, this);
|
|
|
|
#ifdef SECURE_STREAMS
|
|
#ifndef SECURE_BUFFER
|
|
//If SECURE_BUFFER is defined, we've already zeroed this in the
|
|
// multistream.
|
|
memset(s_bufSecure, SECURECHAR, MINISTREAMSIZE);
|
|
#endif
|
|
_ulHighWater = cbSize;
|
|
#endif
|
|
|
|
AddRef();
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Method: CDirectStream::Init, public
|
|
//
|
|
// Synopsis: CDirectStream constructor
|
|
//
|
|
// Arguments: [pstgh] - Parent
|
|
// [pdfn] - Name of entry
|
|
// [fCreate] - Create or get
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// History: 18-Jul-91 PhilipLa Created.
|
|
// 02-Jan-92 PhilipLa Converted to use handle.
|
|
// 25-Aug-92 DrewB Converted to use StgHandle
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_Init)
|
|
#endif
|
|
|
|
SCODE CDirectStream::Init(
|
|
CStgHandle *pstgh,
|
|
CDfName const *pdfn,
|
|
BOOL const fCreate)
|
|
{
|
|
SCODE sc;
|
|
|
|
if (fCreate)
|
|
sc = pstgh->CreateEntry(pdfn, STGTY_STREAM, &_stmh);
|
|
else
|
|
sc = pstgh->GetEntry(pdfn, STGTY_STREAM, &_stmh);
|
|
|
|
if (SUCCEEDED(sc))
|
|
{
|
|
sc = _stmh.GetSize(&_ulSize);
|
|
_ulOldSize = _ulSize;
|
|
#ifdef SECURE_STREAMS
|
|
_ulHighWater = (fCreate) ? 0 : _ulSize;
|
|
#endif
|
|
|
|
#if DBG == 1
|
|
if (SUCCEEDED(sc))
|
|
{
|
|
//Make sure that the stream is sane.
|
|
SCODE sc2;
|
|
CFat *pfat;
|
|
ULONG cbSector;
|
|
ULONG cSect;
|
|
ULONG cSectReal;
|
|
SECT sectStart;
|
|
|
|
if (_ulSize < MINISTREAMSIZE)
|
|
{
|
|
cbSector = MINISECTORSIZE;
|
|
pfat = _stmh.GetMS()->GetMiniFat();
|
|
}
|
|
else
|
|
{
|
|
cbSector = _stmh.GetMS()->GetSectorSize();
|
|
pfat = _stmh.GetMS()->GetFat();
|
|
}
|
|
|
|
cSect = (_ulSize + cbSector - 1) / cbSector;
|
|
sc2 = _stmh.GetMS()->GetDir()->GetStart(
|
|
_stmh.GetSid(),
|
|
§Start);
|
|
if (SUCCEEDED(sc2))
|
|
{
|
|
sc2 = pfat->GetLength(sectStart, &cSectReal);
|
|
if (SUCCEEDED(sc2))
|
|
{
|
|
msfAssert((cSect == cSectReal) &&
|
|
aMsg("Chain length incorrect at init."));
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (SUCCEEDED(sc))
|
|
AddRef();
|
|
_stmc.Init(_stmh.GetMS(), _stmh.GetSid(), this);
|
|
}
|
|
return sc;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::ReadAt, public
|
|
//
|
|
// Synposis: Reads binary data from a linear single stream
|
|
//
|
|
// Arguments: [ulOffset] -- Position to be read from
|
|
//
|
|
// [pBuffer] -- Pointer to the area into which the data
|
|
// will be read.
|
|
// [ulCount] -- Indicates the number of bytes to be read
|
|
// [pulRetval] -- Area into which return value will be stored
|
|
//
|
|
// Returns: Error Code of parent MStream call
|
|
//
|
|
// Algorithm: Calculate start and end sectors and offsets, then
|
|
// pass call up to parent MStream.
|
|
//
|
|
// History: 18-Jul-91 PhilipLa Created.
|
|
// 16-Aug-91 PhilipLa Converted to use multi-sect read
|
|
// 23-Aug-91 PhilipLa Brought into compliance with protocol
|
|
// 11-Sep-91 PhilipLa Moved most functionality up
|
|
// to MStream level.
|
|
// 24-Apr-92 AlexT Move everything to MStream::MRead
|
|
// 09-Jun-92 PhilipLa Added fUnconverted support
|
|
//
|
|
// Notes: [pBuffer] may be unsafe memory
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_ReadAt) // Dirst_Read_TEXT
|
|
#endif
|
|
|
|
|
|
SCODE CDirectStream::ReadAt(
|
|
ULONG ulOffset,
|
|
VOID HUGEP *pBuffer,
|
|
ULONG ulCount,
|
|
ULONG STACKBASED *pulRetval)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"In CDirectStream::ReadAt(%lu,%p,%lu)\n",
|
|
ulOffset,pBuffer,ulCount));
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
CMStream *pms = _stmh.GetMS();
|
|
|
|
*pulRetval = 0;
|
|
|
|
// Check for offset beyond stream size and zero count
|
|
|
|
if ((ulOffset >= _ulSize) || (0 == ulCount))
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
if (ulOffset + ulCount > _ulSize)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"Truncating Read: ulOffset = %lu, ulCount = %lu, _ulSize = %lu\n",
|
|
ulOffset,ulCount,_ulSize));
|
|
ulCount = _ulSize - ulOffset;
|
|
}
|
|
|
|
#ifdef DELAYCONVERT
|
|
if (pms->IsUnconverted())
|
|
{
|
|
ULARGE_INTEGER ulTmp;
|
|
ULISet32(ulTmp, ulOffset);
|
|
return DfGetScode(pms->GetILB()->ReadAt(ulTmp, pBuffer, ulCount,
|
|
pulRetval));
|
|
}
|
|
#else
|
|
msfAssert(!pms->IsUnconverted());
|
|
#endif
|
|
|
|
// Stream is stored in ministream if size < MINISTREAMSIZE
|
|
// and this is not a scratch stream.
|
|
|
|
#ifdef SECURE_STREAMS
|
|
if (ulOffset + ulCount > _ulHighWater)
|
|
{
|
|
if (ulOffset > _ulHighWater)
|
|
{
|
|
//Zero the whole buffer and return.
|
|
memset(pBuffer, SECURECHAR, ulCount);
|
|
*pulRetval = ulCount;
|
|
return S_OK;
|
|
}
|
|
|
|
//Need to read into part of the buffer, then zero fill the
|
|
// rest.
|
|
if (FAILED(sc = ReadAt(ulOffset,
|
|
pBuffer,
|
|
_ulHighWater - ulOffset,
|
|
pulRetval)) ||
|
|
(*pulRetval != _ulHighWater - ulOffset))
|
|
{
|
|
return sc;
|
|
}
|
|
memset((BYTE *)pBuffer + *pulRetval,
|
|
SECURECHAR,
|
|
ulCount - (_ulHighWater - ulOffset));
|
|
|
|
*pulRetval = ulCount;
|
|
return S_OK;
|
|
}
|
|
#endif
|
|
|
|
|
|
SID sid = _stmh.GetSid();
|
|
CFat *pfat = pms->GetFat();
|
|
USHORT cbSector = pms->GetSectorSize();
|
|
USHORT uShift = pms->GetSectorShift();
|
|
USHORT uMask = pms->GetSectorMask();
|
|
|
|
|
|
|
|
if ((_ulSize < MINISTREAMSIZE) &&
|
|
(!pms->IsScratch()) &&
|
|
(sid != SIDMINISTREAM))
|
|
{
|
|
msfAssert(sid <= MAXREGSID);
|
|
|
|
// This stream is stored in the ministream
|
|
|
|
cbSector = MINISECTORSIZE;
|
|
uShift = MINISECTORSHIFT;
|
|
uMask = cbSector - 1;
|
|
pfat = pms->GetMiniFat();
|
|
}
|
|
|
|
SECT start = (SECT)(ulOffset >> uShift);
|
|
OFFSET oStart = (OFFSET)(ulOffset & uMask);
|
|
|
|
SECT end = (SECT)((ulOffset + ulCount - 1) >> uShift);
|
|
OFFSET oEnd = (OFFSET)((ulOffset + ulCount - 1) & uMask);
|
|
|
|
ULONG total = 0;
|
|
|
|
ULONG cSect = end - start + 1;
|
|
|
|
USHORT offset;
|
|
offset = oStart;
|
|
|
|
while (TRUE)
|
|
{
|
|
ULONG cSeg;
|
|
SSegment segtab[CSEG + 1];
|
|
|
|
msfChk(_stmc.Contig(start,
|
|
FALSE,
|
|
(SSegment STACKBASED *) segtab,
|
|
cSect,
|
|
&cSeg));
|
|
msfAssert(cSeg <= CSEG);
|
|
|
|
USHORT oend = cbSector - 1;
|
|
for (USHORT iseg = 0; iseg < cSeg;)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"Segment: (%lu,%lu)\n",segtab[iseg].sectStart,segtab[iseg].cSect));
|
|
SECT sectStart = segtab[iseg].sectStart;
|
|
ULONG i = segtab[iseg].cSect;
|
|
if (i > cSect)
|
|
i = cSect;
|
|
|
|
cSect -= i;
|
|
start += i;
|
|
|
|
iseg++;
|
|
if (cSect == 0)
|
|
oend = oEnd;
|
|
|
|
ULONG ulSize = ((i - 1) << uShift) - offset + oend + 1;
|
|
|
|
ULONG bytecount;
|
|
SCODE sc;
|
|
|
|
if (pms->GetMiniFat() == pfat)
|
|
{
|
|
sc = pms->GetMiniStream()->CDirectStream::ReadAt(
|
|
(sectStart << uShift) + offset,
|
|
pBuffer, ulSize,
|
|
(ULONG STACKBASED *)&bytecount);
|
|
}
|
|
else
|
|
{
|
|
ULARGE_INTEGER ulOffset;
|
|
ULISet32(ulOffset, ConvertSectOffset(sectStart,offset,uShift));
|
|
sc = DfGetScode(pms->GetILB()->ReadAt(ulOffset,
|
|
(BYTE *)pBuffer, ulSize,
|
|
&bytecount));
|
|
}
|
|
|
|
total += bytecount;
|
|
if ((0 == cSect) || (FAILED(sc)))
|
|
{
|
|
*pulRetval = total;
|
|
msfDebugOut((DEB_ITRACE,
|
|
"Leaving CDirectStream::ReadAt()=>%lu, ret is %lu\n",
|
|
sc,*pulRetval));
|
|
return sc;
|
|
}
|
|
|
|
pBuffer = (BYTE HUGEP *)pBuffer + bytecount;
|
|
offset = 0;
|
|
}
|
|
}
|
|
|
|
msfDebugOut((DEB_ERROR,"In CDirectStream::ReadAt - reached end of function\n"));
|
|
Err:
|
|
return sc;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::Write, public
|
|
//
|
|
// Synposis: Writes binary data from a linear single stream
|
|
//
|
|
// Effects: Modifies _ulSeekPos. May cause modification in parent
|
|
// MStream.
|
|
//
|
|
// Arguments: [pBuffer] -- Pointer to the area from which the data
|
|
// will be written.
|
|
// [ulCount] -- Indicates the number of bytes to be written
|
|
// [pulRetval] -- Pointer to area in which number of bytes
|
|
// will be returned
|
|
//
|
|
// Returns: Error code of MStream call.
|
|
//
|
|
// Algorithm: Calculate sector and offset for beginning and end of
|
|
// write, then pass call up to MStream.
|
|
//
|
|
//
|
|
// History: 18-Jul-91 PhilipLa Created.
|
|
// 16-Aug-91 PhilipLa Converted to use multi-sect write
|
|
// 23-Aug-91 PhilipLa Brought into compliance with protocol
|
|
// 11-Sep-91 PhilipLa Moved most functionality up
|
|
// to MStream level.
|
|
//
|
|
// Notes: [pBuffer] may be unsafe memory
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_WriteAt) // Dirst_Write_TEXT
|
|
#endif
|
|
|
|
SCODE CDirectStream::WriteAt(
|
|
ULONG ulOffset,
|
|
VOID const HUGEP *pBuffer,
|
|
ULONG ulCount,
|
|
ULONG STACKBASED *pulRetval)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"In CDirectStream::WriteAt(%lu,%p,%lu)\n",ulOffset,pBuffer,ulCount));
|
|
|
|
*pulRetval = 0;
|
|
|
|
if (0 == ulCount)
|
|
return S_OK;
|
|
|
|
SCODE sc;
|
|
|
|
CMStream *pms;
|
|
pms = _stmh.GetMS();
|
|
msfAssert(pms != NULL);
|
|
|
|
if (ulOffset + ulCount > _ulSize)
|
|
{
|
|
if (_ulSize > MINISTREAMSIZE)
|
|
{
|
|
}
|
|
else
|
|
{
|
|
msfChk(SetSize(ulOffset + ulCount));
|
|
}
|
|
}
|
|
|
|
// This should be an inline call to MWrite
|
|
|
|
#ifdef SECURE_STREAMS
|
|
if (ulOffset > _ulHighWater)
|
|
{
|
|
ClearSects(_ulHighWater, ulOffset);
|
|
}
|
|
#endif
|
|
|
|
msfChk(pms->MWrite(
|
|
_stmh.GetSid(),
|
|
(_ulSize < MINISTREAMSIZE),
|
|
ulOffset,
|
|
pBuffer,
|
|
ulCount,
|
|
&_stmc,
|
|
pulRetval));
|
|
|
|
msfDebugOut((DEB_ITRACE,"Leaving CDirectStream::WriteAt()==>%lu, ret is %lu\n",sc,*pulRetval));
|
|
|
|
Err:
|
|
#ifdef SECURE_STREAMS
|
|
if ((*pulRetval > 0 ) &&
|
|
(ulOffset + *pulRetval > _ulHighWater))
|
|
{
|
|
_ulHighWater = ulOffset + *pulRetval;
|
|
}
|
|
#endif
|
|
|
|
if (*pulRetval > 0 &&
|
|
ulOffset + *pulRetval > _ulSize)
|
|
{
|
|
SCODE scSet;
|
|
|
|
_ulSize = ulOffset + *pulRetval;
|
|
|
|
scSet = pms->GetDir()->SetSize(_stmh.GetSid(), _ulSize);
|
|
if (SUCCEEDED(sc) && FAILED(scSet))
|
|
{
|
|
sc = scSet;
|
|
}
|
|
}
|
|
#ifdef SECURE_STREAMS
|
|
msfAssert(_ulHighWater <= _ulSize);
|
|
#endif
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::SetSize, public
|
|
//
|
|
// Synposis: Set the size of a linear stream
|
|
//
|
|
// Effects: Modifies _ulSize. May cause change in parent MStream.
|
|
//
|
|
// Arguments: [ulNewSize] -- New size for stream
|
|
//
|
|
// Returns: Error code returned by MStream call.
|
|
//
|
|
// Algorithm: Pass call up to parent.
|
|
//
|
|
// History: 29-Jul-91 PhilipLa Created.
|
|
// 14-May-92 AlexT Add small object support
|
|
//
|
|
// Notes: When changing the size of a stream, we need to be concerned
|
|
// with the cases where each stream is either zero length,
|
|
// stored in the ministream, or stored in a regular stream. The following
|
|
// grid shows the actions that we must perform in each case:
|
|
//
|
|
// New Sector Count (Cn)
|
|
//
|
|
// 0 S L
|
|
// O ------------------------------------------------
|
|
// l | same size | allocate Cn | allocate Cn
|
|
// d 0 | (fast out) | small sectors | large sectors
|
|
// ------------------------------------------------
|
|
// S | small | Co > Cn: | cbCopy = cbOld
|
|
// e S | setchain(Cn) | small | large allocate Cn
|
|
// c | | setchain(Cn)| copy bytes
|
|
// t | | Cn > Co: | small setchain(0)
|
|
// o | | extend small | copy data
|
|
// r ------------------------------------------------
|
|
// L | large | cbCopy = cbNew| Co > Cn:
|
|
// C | setchain(Cn) | small | large setchain(Cn)
|
|
// o | | allocate Cn | Cn > Co:
|
|
// u | | copy bytes | extend large
|
|
// n | | large |
|
|
// t | | setchain(0) |
|
|
// | | copy data |
|
|
// (Co) ------------------------------------------------
|
|
//
|
|
// where S indicates small sectors, L indicates large sectors, and Cx
|
|
// represents count of sectors. For example, the middle box represents
|
|
// doing a setsize on a stream which is currently stored in a small
|
|
// stream in Co small sectors and which will end up in a large stream
|
|
// with Cn sectors.
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_SetSize) // Dirst_SetSize_TEXT
|
|
#endif
|
|
|
|
SCODE CDirectStream::SetSize(ULONG cbNewSize)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"In CDirectStream::SetSize(%lu)\n",cbNewSize));
|
|
|
|
SCODE sc = S_OK;
|
|
BYTE *pBuf = NULL;
|
|
SID sid = _stmh.GetSid();
|
|
CMStream *pms = _stmh.GetMS();
|
|
msfAssert(sid <= MAXREGSID);
|
|
CDirectory *pdir = pms->GetDir();
|
|
|
|
if (_ulSize == cbNewSize)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
#ifdef SECURE_STREAMS
|
|
msfAssert(_ulHighWater <= _ulSize);
|
|
#endif
|
|
|
|
USHORT cbpsOld = pms->GetSectorSize();
|
|
// Count of Bytes Per Sector
|
|
USHORT cbpsNew = cbpsOld;
|
|
CFat *pfatOld = pms->GetFat();
|
|
CFat *pfatNew = pfatOld;
|
|
|
|
if ((!pms->IsScratch()) && (SIDMINISTREAM != sid))
|
|
{
|
|
// This is not a scratch DocFile, nor is this stream the ministream;
|
|
// check if this stream is and/or will be stored in the ministream.
|
|
|
|
if (cbNewSize < MINISTREAMSIZE)
|
|
{
|
|
cbpsNew = MINISECTORSIZE;
|
|
pfatNew = pms->GetMiniFat();
|
|
}
|
|
|
|
if (_ulSize < MINISTREAMSIZE)
|
|
{
|
|
cbpsOld = MINISECTORSIZE;
|
|
pfatOld = pms->GetMiniFat();
|
|
}
|
|
}
|
|
|
|
ULONG csectOld = (ULONG)(_ulSize + cbpsOld - 1) / (ULONG) cbpsOld;
|
|
ULONG csectNew = (ULONG)(cbNewSize + cbpsNew - 1) / (ULONG) cbpsNew;
|
|
|
|
msfAssert(sid <= MAXREGSID);
|
|
SECT sectstart, sectOldStart;
|
|
msfChk(pdir->GetStart(sid, §start));
|
|
|
|
//Save start sector so we can free it later.
|
|
sectOldStart = sectstart;
|
|
|
|
msfDebugOut((DEB_ITRACE,"pdbOld size is %lu\n\tSid is %lu\n\tStart is %lu\n",
|
|
_ulSize,sid,sectstart));
|
|
msfDebugOut((DEB_ITRACE,"CMStream::SetSize() needs %lu %u byte sectors\n",
|
|
csectNew, cbpsNew));
|
|
msfDebugOut((DEB_ITRACE,"SetSize() currently has %lu %u byte sectors\n",
|
|
csectOld, cbpsOld));
|
|
|
|
USHORT cbCopy;
|
|
cbCopy = 0;
|
|
if (cbpsOld != cbpsNew)
|
|
{
|
|
// Sector sizes are different, so we'll copy the data
|
|
msfAssert((cbNewSize > _ulSize ? _ulSize : cbNewSize) < 0x10000);
|
|
cbCopy =(USHORT)(cbNewSize > _ulSize ? _ulSize : cbNewSize);
|
|
}
|
|
|
|
|
|
if (cbCopy > 0)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"Copying between fat and minifat\n"));
|
|
GetSafeBuffer(cbCopy, cbCopy, &pBuf, &cbCopy);
|
|
msfAssert((pBuf != NULL) && aMsg("Couldn't get scratch buffer"));
|
|
|
|
ULONG ulRetVal;
|
|
sc = ReadAt(0, pBuf, cbCopy, (ULONG STACKBASED *)&ulRetVal);
|
|
#ifdef SECURE_STREAMS
|
|
//Part of the buffer may have gunk in it, so clear it out.
|
|
if (_ulHighWater < cbCopy)
|
|
{
|
|
memset(pBuf + _ulHighWater, SECURECHAR, cbCopy - _ulHighWater);
|
|
}
|
|
#endif
|
|
|
|
if ((FAILED(sc)) ||
|
|
((ulRetVal != cbCopy) ? (sc = STG_E_UNKNOWN) : 0))
|
|
{
|
|
msfErr(Err, sc);
|
|
}
|
|
#ifdef SECURE_STREAMS
|
|
ClearSects(_ulHighWater, _ulSize);
|
|
_ulHighWater = cbNewSize;
|
|
#endif
|
|
|
|
//The cache is no longer valid, so empty it.
|
|
_stmc.Empty();
|
|
|
|
msfChk(_stmc.Allocate(pfatNew, csectNew, §start));
|
|
}
|
|
else
|
|
{
|
|
SECT dummy;
|
|
|
|
if ((csectOld > csectNew))
|
|
{
|
|
#ifdef SECURE_STREAMS
|
|
ClearSects(_ulHighWater, _ulSize);
|
|
_ulHighWater = cbNewSize;
|
|
#endif
|
|
if (0 == csectNew)
|
|
{
|
|
//Note: We need to set the start sect in the directory
|
|
// first in case the SetChainLength call fails part way
|
|
// through, which would leave this directory entry pointing
|
|
// to a FREESECT.
|
|
SECT sectOldStart = sectstart;
|
|
msfChk(pdir->SetStart(sid, ENDOFCHAIN));
|
|
sectstart = ENDOFCHAIN;
|
|
msfChk(pfatOld->SetChainLength(sectOldStart, 0));
|
|
}
|
|
else
|
|
{
|
|
msfChk(pfatOld->SetChainLength(sectstart, csectNew));
|
|
}
|
|
|
|
//If this turns out to be a common case, we can
|
|
// sometimes keep the cache valid here.
|
|
_stmc.Empty();
|
|
}
|
|
else if (0 == csectOld)
|
|
{
|
|
msfChk(_stmc.Allocate(pfatNew, csectNew, §start));
|
|
}
|
|
else if (csectNew > csectOld)
|
|
{
|
|
ULONG start = csectNew - 1;
|
|
msfChk(_stmc.GetESect(start, &dummy));
|
|
}
|
|
}
|
|
|
|
|
|
// Resize the ministream, if necessary
|
|
if (((MINISECTORSIZE == cbpsOld) && (csectOld > 0)) ||
|
|
((MINISECTORSIZE == cbpsNew) && (csectNew > 0)))
|
|
{
|
|
msfChk(pms->SetMiniSize());
|
|
}
|
|
|
|
msfChk(pms->SetSize());
|
|
|
|
//If we fail on either of these operations and cbCopy != 0,
|
|
// we will have data loss. Ick.
|
|
|
|
// Optimization - we only set the start sector if it has changed.
|
|
|
|
if (sectstart != sectOldStart)
|
|
{
|
|
msfChk(pdir->SetStart(sid, sectstart));
|
|
}
|
|
|
|
//If we fail here, we're screwed.
|
|
msfChk(pdir->SetSize(sid, cbNewSize));
|
|
|
|
_ulSize = cbNewSize;
|
|
#ifdef SECURE_STREAMS
|
|
if (_ulHighWater > _ulSize)
|
|
{
|
|
_ulHighWater = _ulSize;
|
|
}
|
|
#endif
|
|
|
|
if (cbCopy > 0)
|
|
{
|
|
// now copy the data
|
|
ULONG ulRetVal;
|
|
|
|
msfAssert(cbCopy <= _ulSize);
|
|
msfChk(WriteAt(0, pBuf, cbCopy, (ULONG STACKBASED *)&ulRetVal));
|
|
|
|
if (ulRetVal != cbCopy)
|
|
{
|
|
msfErr(Err, STG_E_UNKNOWN);
|
|
}
|
|
|
|
msfChk(pfatOld->SetChainLength(sectOldStart, 0));
|
|
msfChk(pms->SetMiniSize());
|
|
msfChk(pms->SetSize());
|
|
}
|
|
|
|
#ifdef SECURE
|
|
if (((csectNew > csectOld) || (cbCopy > 0)) &&
|
|
((cbNewSize & (cbpsNew - 1)) != 0))
|
|
{
|
|
SECT sectLast;
|
|
msfChk(_stmc.GetSect(csectNew - 1, §Last));
|
|
|
|
msfVerify(SUCCEEDED(pms->SecureSect(
|
|
sectLast,
|
|
cbNewSize,
|
|
(cbNewSize < MINISTREAMSIZE) && (sid != SIDMINISTREAM))));
|
|
}
|
|
#endif //SECURE
|
|
#ifdef SECURE_STREAMS
|
|
msfAssert(_ulHighWater <= _ulSize);
|
|
#endif
|
|
Err:
|
|
#ifdef SECURE_STREAMS
|
|
if (_ulHighWater > _ulSize)
|
|
{
|
|
_ulHighWater = _ulSize;
|
|
}
|
|
#endif
|
|
// Optimization - avoid calling FreeBuffer (which will end up calling
|
|
// out to CompObj.DLL) when pBuf is NULL (common case).
|
|
if (pBuf != NULL)
|
|
FreeBuffer(pBuf);
|
|
|
|
if (FAILED(sc))
|
|
_stmc.Empty();
|
|
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Method: CDirectStream::BeginCommitFromChild, public
|
|
//
|
|
// Synopsis: Begin a commit from a child stream
|
|
//
|
|
// Arguments: [ulSize] -- Size of child stream
|
|
// [pDelta] -- Pointer to delta list
|
|
// [pstChild] - Child
|
|
//
|
|
// Returns: S_OK if call completed successfully.
|
|
//
|
|
// Algorithm: *Finish This*
|
|
//
|
|
// History: 22-Jan-92 PhilipLa Created.
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_BeginCommitFromChild) // Dirst_Commit_TEXT
|
|
#endif
|
|
|
|
SCODE CDirectStream::BeginCommitFromChild(
|
|
ULONG ulSize,
|
|
CDeltaList *pDelta,
|
|
CTransactedStream *pstChild)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"In CDirectStream::BeginCommitFromChild:%p("
|
|
"%lu, %p, %p)\n", this, ulSize, pDelta, pstChild));
|
|
|
|
// msfDebugOut((DEB_ITRACE,"MultiStrean is %p\nStream name is: %ws\n",_stmh.GetMS(),((_stmh.GetMS()))->GetName(_stmh.GetSid())));
|
|
|
|
SCODE sc = S_OK;
|
|
ULONG temp;
|
|
BYTE *pb = NULL;
|
|
|
|
_pdlHolder = P_TO_BP(CBasedDeltaListPtr, pDelta);
|
|
_ulOldSize = _ulSize;
|
|
|
|
// Copy-on-write will back out these changes if we fail
|
|
|
|
#ifdef USE_NOSCRATCH
|
|
//For no-scratch mode, we commit very differently than we do regularly,
|
|
// unless this commit is somehow involving the minifat.
|
|
|
|
if ((pDelta->IsNoScratch()) &&
|
|
(ulSize >= MINISTREAMSIZE) &&
|
|
((_ulSize >= MINISTREAMSIZE) || (_ulSize == 0)) &&
|
|
(!pDelta->IsEmpty()))
|
|
{
|
|
USHORT cbSector = pDelta->GetDataSectorSize();
|
|
SECT sectEnd = (ulSize + cbSector - 1) / cbSector;
|
|
SECT sectLast = ENDOFCHAIN;
|
|
|
|
//We update our size and directory first so the stream cache
|
|
// code knows which fat to get to.
|
|
_ulSize = ulSize;
|
|
msfChk(_stmh.GetMS()->GetDir()->SetSize(_stmh.GetSid(), ulSize));
|
|
|
|
if (_ulOldSize > _ulSize)
|
|
{
|
|
//We need to truncate the chain.
|
|
SECT sectStart;
|
|
_stmh.GetMS()->GetDir()->GetStart(_stmh.GetSid(), §Start);
|
|
_stmh.GetMS()->GetFat()->SetChainLength(sectStart, sectEnd);
|
|
_stmc.Empty();
|
|
}
|
|
|
|
for (ULONG i = 0; i < sectEnd; i++)
|
|
{
|
|
SECT sectDirty;
|
|
SECT sectOld;
|
|
SECT sectNext;
|
|
|
|
//For each sector, chain the old sector with the
|
|
//existing new one.
|
|
msfChk(pDelta->GetMap(i, DL_READ, §Dirty));
|
|
if (sectDirty != ENDOFCHAIN)
|
|
{
|
|
if (i > 0)
|
|
{
|
|
CFat *pfat = _stmh.GetMS()->GetFat();
|
|
|
|
msfChk(_stmc.GetSect(i - 1, §Last));
|
|
msfAssert(sectLast != ENDOFCHAIN);
|
|
|
|
msfChk(pfat->GetNext(sectLast, §Old));
|
|
msfChk(pfat->SetNext(sectLast, sectDirty));
|
|
if (sectOld != ENDOFCHAIN)
|
|
{
|
|
msfChk(pfat->GetNext(sectOld, §Next));
|
|
msfChk(pfat->SetNext(sectOld, FREESECT));
|
|
}
|
|
else
|
|
{
|
|
sectNext = ENDOFCHAIN;
|
|
}
|
|
msfChk(pfat->SetNext(sectDirty, sectNext));
|
|
msfChk(_stmc.EmptyRegion(i, i));
|
|
}
|
|
else
|
|
{
|
|
CFat *pfat = _stmh.GetMS()->GetFat();
|
|
CDirectory *pdir = _stmh.GetMS()->GetDir();
|
|
|
|
sectNext = ENDOFCHAIN;
|
|
|
|
msfChk(pdir->GetStart(_stmh.GetSid(), §Last));
|
|
if (sectLast != ENDOFCHAIN)
|
|
{
|
|
msfChk(pfat->GetNext(sectLast, §Next));
|
|
msfChk(pfat->SetNext(sectLast, FREESECT));
|
|
}
|
|
msfChk(pfat->SetNext(sectDirty, sectNext));
|
|
msfChk(pdir->SetStart(_stmh.GetSid(), sectDirty));
|
|
msfChk(_stmc.EmptyRegion(i, i));
|
|
}
|
|
}
|
|
}
|
|
#ifdef SECURE_STREAMS
|
|
_ulHighWater = _ulSize;
|
|
#endif
|
|
|
|
#if DBG == 1
|
|
//Make sure that the stream is sane.
|
|
ULONG ulDirSize;
|
|
ULONG cSect;
|
|
ULONG cSectReal;
|
|
SECT sectStart;
|
|
|
|
_stmh.GetMS()->GetDir()->GetSize(_stmh.GetSid(), &ulDirSize);
|
|
cSect = (ulDirSize + cbSector - 1) / cbSector;
|
|
_stmh.GetMS()->GetDir()->GetStart(_stmh.GetSid(), §Start);
|
|
_stmh.GetMS()->GetFat()->GetLength(sectStart, &cSectReal);
|
|
msfAssert((cSect == cSectReal) &&
|
|
aMsg("Chain length incorrect after commit."));
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
#endif //USE_NOSCRATCH
|
|
|
|
// Note: It's critical that we do this SetSize first (since we
|
|
// use our scratch buffer below and SetSize can potentially also
|
|
// use the scratch buffer.
|
|
|
|
msfChk(SetSize(ulSize));
|
|
|
|
msfAssert(pDelta != NULL);
|
|
|
|
if (!pDelta->IsEmpty())
|
|
{
|
|
USHORT cbSector = pDelta->GetDataSectorSize();
|
|
USHORT uSectorShift = pDelta->GetDataSectorShift();
|
|
|
|
USHORT cbActualSize;
|
|
USHORT cbMaxSects = 15;
|
|
|
|
GetSafeBuffer(cbSector, cbSector * cbMaxSects, &pb, &cbActualSize);
|
|
|
|
BYTE *pbcurrent = pb;
|
|
|
|
msfAssert((pb != NULL) && aMsg("Couldn't get scratch buffer"));
|
|
cbMaxSects = cbActualSize / cbSector;
|
|
|
|
OFFSET oEnd;
|
|
oEnd = (OFFSET)((_ulSize - 1) % cbSector) + 1;
|
|
|
|
SECT sectEnd;
|
|
sectEnd = ((_ulSize + cbSector - 1) / cbSector);
|
|
|
|
ULONG ulOffset;
|
|
ILockBytes *pilbDirty;
|
|
|
|
SECT sectBeginRead = ENDOFCHAIN; //the first sector to read
|
|
SECT sectBeginWrite = ENDOFCHAIN; //the offset of the first place to write
|
|
SECT sectDirty = ENDOFCHAIN; //the sector at this offset
|
|
|
|
USHORT uReadCount = 0;
|
|
|
|
ULARGE_INTEGER ulTmp;
|
|
pilbDirty = pDelta->GetDataILB();
|
|
|
|
|
|
for (ulOffset = 0; ulOffset < sectEnd; ulOffset++)
|
|
{
|
|
|
|
if (sectDirty == ENDOFCHAIN)
|
|
{
|
|
sectBeginWrite = ulOffset;
|
|
}
|
|
|
|
msfChk(pDelta->GetMap(ulOffset, DL_READ, §Dirty));
|
|
|
|
|
|
// Read will happen when we have determined there is something to read
|
|
// and either
|
|
// we have reached an ENDOFCHAIN
|
|
// we have reached a non-adjacent sector
|
|
// or we have reached the maximun amount that can be read
|
|
if ( (uReadCount) &&
|
|
( (sectDirty == ENDOFCHAIN) ||
|
|
(sectDirty != sectBeginRead + uReadCount) ||
|
|
(ulOffset - sectBeginWrite == cbMaxSects) ))
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"Reading %u sectors from sect %lu\n",
|
|
uReadCount,
|
|
sectBeginRead));
|
|
|
|
ULISet32(ulTmp, ConvertSectOffset(sectBeginRead, 0,
|
|
uSectorShift));
|
|
msfHChk(pilbDirty->ReadAt(
|
|
ulTmp,
|
|
pbcurrent,
|
|
cbSector * uReadCount,
|
|
(ULONG STACKBASED *)&temp));
|
|
|
|
pbcurrent += cbSector * uReadCount;
|
|
sectBeginRead = sectDirty;
|
|
|
|
//reset the ReadCount now that we are done reading
|
|
if (sectDirty == ENDOFCHAIN)
|
|
uReadCount = 0;
|
|
else
|
|
uReadCount = 1;
|
|
|
|
}
|
|
|
|
// increment the read count
|
|
else if (sectDirty != ENDOFCHAIN)
|
|
{
|
|
if (sectBeginRead == ENDOFCHAIN)
|
|
{
|
|
sectBeginRead = sectDirty;
|
|
uReadCount=1;
|
|
}
|
|
else if (sectDirty == sectBeginRead + uReadCount)
|
|
{
|
|
uReadCount++;
|
|
}
|
|
}
|
|
|
|
// Write occurs when we have something to write
|
|
// and we have reached an ENDOFCHAIN
|
|
// or the write buffer is full
|
|
if (((sectDirty == ENDOFCHAIN) &&
|
|
(sectBeginWrite != ulOffset) ) ||
|
|
(ulOffset - sectBeginWrite == cbMaxSects) )
|
|
{
|
|
|
|
msfDebugOut((DEB_ITRACE,"Writing %u sectors from sect %lu\n",
|
|
ulOffset - sectBeginWrite,
|
|
sectBeginWrite));
|
|
|
|
msfChk(WriteAt(sectBeginWrite * cbSector,
|
|
pb,
|
|
cbSector * (ulOffset - sectBeginWrite),
|
|
(ULONG STACKBASED *)&temp));
|
|
pbcurrent = pb;
|
|
sectBeginWrite = ulOffset;
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
// After loop, do last read and write
|
|
if (uReadCount)
|
|
{
|
|
ULISet32(ulTmp, ConvertSectOffset(sectBeginRead, 0,
|
|
uSectorShift));
|
|
|
|
msfDebugOut((DEB_ITRACE,"END::Reading %u sectors from sect %lu\n",
|
|
uReadCount,
|
|
sectBeginRead));
|
|
|
|
msfHChk(pilbDirty->ReadAt(
|
|
ulTmp,
|
|
pbcurrent,
|
|
cbSector * uReadCount,
|
|
(ULONG STACKBASED *)&temp));
|
|
|
|
msfDebugOut((DEB_ITRACE,"END::Writing %u sectors from sect %lu\n",
|
|
ulOffset - sectBeginWrite,
|
|
sectBeginWrite));
|
|
|
|
msfChk(WriteAt(sectBeginWrite * cbSector,
|
|
pb,
|
|
// all sectors except the last one
|
|
(cbSector * ((ulOffset-1) - sectBeginWrite)
|
|
// last sector fragment
|
|
+ oEnd ),
|
|
(ULONG STACKBASED *)&temp));
|
|
}
|
|
|
|
|
|
}
|
|
#ifdef USE_NOSCRATCH
|
|
}
|
|
#endif
|
|
msfDebugOut((DEB_ITRACE,"Out CDirectStream::BeginCommitFromChild()\n"));
|
|
|
|
Err:
|
|
#ifdef SECURE_STREAMS
|
|
msfAssert(_ulHighWater <= _ulSize);
|
|
#endif
|
|
FreeBuffer(pb);
|
|
return sc;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Method: CDirectStream::EndCommitFromChild
|
|
//
|
|
// Synopsis: End a commit sequence from a child stream
|
|
//
|
|
// Arguments: [df] -- Indicates whether to commit or abort
|
|
// [pstChild] - Child
|
|
//
|
|
// Returns: S_OK if call completed successfully.
|
|
//
|
|
// Algorithm: *Finish This*
|
|
//
|
|
// History: 22-Jan-92 PhilipLa Created.
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_EndCommitFromChild)
|
|
#endif
|
|
|
|
void CDirectStream::EndCommitFromChild(DFLAGS df,
|
|
CTransactedStream *pstChild)
|
|
{
|
|
msfDebugOut((DEB_ITRACE,"In CDirectStream::EndCommitFromChild:%p("
|
|
"%lu, %p)\n", this, df, pstChild));
|
|
if (!P_COMMIT(df))
|
|
{
|
|
_ulSize = _ulOldSize;
|
|
#ifdef SECURE_STREAMS
|
|
_ulHighWater = _ulSize;
|
|
#endif
|
|
|
|
//Stream cache is no longer valid, so nuke it.
|
|
_stmc.Empty();
|
|
}
|
|
|
|
_pdlHolder = NULL;
|
|
msfDebugOut((DEB_ITRACE,"Out CDirectStream::EndCommitFromChild()\n"));
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::Release, public
|
|
//
|
|
// Synopsis: Decrements the ref count and frees if necessary
|
|
//
|
|
// History: 08-May-92 DrewB Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_Release) // Dirst_Release_TEXT
|
|
#endif
|
|
|
|
void CDirectStream::Release(VOID)
|
|
{
|
|
LONG lRet;
|
|
|
|
msfDebugOut((DEB_ITRACE,"In CDirectStream::Release()\n"));
|
|
msfAssert(_cReferences > 0);
|
|
|
|
lRet = AtomicDec(&_cReferences);
|
|
if (lRet == 0)
|
|
delete this;
|
|
msfDebugOut((DEB_ITRACE,"Out CDirectStream::Release()\n"));
|
|
}
|
|
|
|
//+--------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::AddRef, public
|
|
//
|
|
// Synopsis: Increments the ref count
|
|
//
|
|
// History: 08-May-92 DrewB Created
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_AddRef) //
|
|
#endif
|
|
|
|
void CDirectStream::AddRef(void)
|
|
{
|
|
msfDebugOut((DEB_ITRACE, "In CDirectStream::AddRef()\n"));
|
|
AtomicInc(&_cReferences);
|
|
msfDebugOut((DEB_ITRACE, "Out CDirectStream::AddRef, %lu\n",
|
|
_cReferences));
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::GetSize, public
|
|
//
|
|
// Synopsis: Gets the size of the stream
|
|
//
|
|
// Arguments: [pulSize] - Size return
|
|
//
|
|
// Modifies: [pulSize]
|
|
//
|
|
// History: 08-May-92 DrewB Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_GetSize)
|
|
#endif
|
|
|
|
void CDirectStream::GetSize(ULONG *pulSize)
|
|
{
|
|
*pulSize = _ulSize;
|
|
}
|
|
|
|
|
|
#ifdef CODESEGMENTS
|
|
#pragma code_seg(SEG_CDirectStream_GetDeltaList) //
|
|
#endif
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::GetDeltaList, public
|
|
//
|
|
// Synopsis: Returns NULL, since a direct stream can never have
|
|
// a delta list.
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: NULL
|
|
//
|
|
// History: 30-Jul-93 PhilipLa Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
CDeltaList * CDirectStream::GetDeltaList(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
#ifdef SECURE_STREAMS
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CDirectStream::ClearSects, private
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: void
|
|
//
|
|
// History: 10-Oct-95 PhilipLa Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void CDirectStream::ClearSects(ULONG ulStartOffset, ULONG ulEndOffset)
|
|
{
|
|
ULONG cbBytesToWrite = ulEndOffset - ulStartOffset;
|
|
ULONG ulOffset = ulStartOffset;
|
|
msfAssert(ulEndOffset >= ulStartOffset);
|
|
//Start offset must be less than high water mark, or the WriteAt
|
|
// will recurse.
|
|
msfAssert(ulStartOffset <= _ulHighWater);
|
|
|
|
while (cbBytesToWrite > 0)
|
|
{
|
|
ULONG cbWritten;
|
|
if (FAILED(WriteAt(ulOffset,
|
|
s_bufSecure,
|
|
min(MINISTREAMSIZE, cbBytesToWrite),
|
|
&cbWritten)))
|
|
{
|
|
break;
|
|
}
|
|
ulOffset += cbWritten;
|
|
cbBytesToWrite -= cbWritten;
|
|
}
|
|
|
|
return;
|
|
}
|
|
#endif
|