2020-09-30 17:12:29 +02:00

774 lines
19 KiB
C++

//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1996.
//
// File: laylkb.cxx
//
// Contents: File ILockBytes implementation for layout storage
//
// Classes:
//
// Functions:
//
// History: 19-Feb-96 SusiA Created
//
//----------------------------------------------------------------------------
#include "layouthd.cxx"
#pragma hdrstop
#include "laylkb.hxx"
#include <valid.h>
class CSafeCriticalSection
{
public:
inline CSafeCriticalSection(CRITICAL_SECTION *pcs);
inline ~CSafeCriticalSection();
private:
CRITICAL_SECTION *_pcs;
};
inline CSafeCriticalSection::CSafeCriticalSection(CRITICAL_SECTION *pcs)
{
_pcs = pcs;
EnterCriticalSection(_pcs);
}
inline CSafeCriticalSection::~CSafeCriticalSection()
{
LeaveCriticalSection(_pcs);
#if DBG == 1
_pcs = NULL;
#endif
}
#define TAKE_CS CSafeCriticalSection scs(&_cs);
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::CLayoutLockBytes, public
//
// Synopsis: Default constructor
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
CLayoutLockBytes::CLayoutLockBytes(void)
{
_cReferences = 1;
_h = INVALID_HANDLE_VALUE;
_hScript = INVALID_HANDLE_VALUE;
_fLogging = FALSE;
_cbSectorShift = 0;
_atcScriptName[0] = TEXT('\0');
_awcName[0] = L'\0';
InitializeCriticalSection(&_cs);
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::~CLayoutLockBytes, public
//
// Synopsis: Destructor
//
// Returns: Appropriate status code
//
// History: 20-Feb-96 SusiA Created
//
//----------------------------------------------------------------------------
CLayoutLockBytes::~CLayoutLockBytes()
{
if (_h != INVALID_HANDLE_VALUE)
{
CloseHandle(_h);
_h = INVALID_HANDLE_VALUE;
}
if (_hScript != INVALID_HANDLE_VALUE)
{
CloseHandle(_hScript);
_hScript = INVALID_HANDLE_VALUE;
}
DeleteCriticalSection(&_cs);
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::Init, public
//
// Synopsis: Initialization function
//
// Arguments:
//
// Returns: Appropriate status code
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
SCODE CLayoutLockBytes::Init(OLECHAR const *pwcsName,
DWORD grfMode)
{
SCODE sc = S_OK;
BYTE abHeader[sizeof(CMSFHeaderData)];
ULONG cbRead;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::Init:%p()\n", this));
if (pwcsName == NULL)
return STG_E_INVALIDNAME;
_grfMode = grfMode;
#ifndef UNICODE
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;
}
_h = CreateFileA(atcPath,
GENERIC_READ | GENERIC_WRITE, //Read-write
0, // No sharing
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS,
NULL);
#else
_h = CreateFile(pwcsName,
GENERIC_READ | GENERIC_WRITE, //Read-write
0, // No sharing
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS,
NULL);
#endif
if (_h == INVALID_HANDLE_VALUE)
{
layErr(Err, Win32ErrorToScode(GetLastError()));
}
lstrcpyW(_awcName, pwcsName);
//Get the sector size
boolChk(ReadFile(_h, abHeader, sizeof(CMSFHeaderData), &cbRead, NULL));
if (cbRead != sizeof(CMSFHeaderData))
{
return STG_E_READFAULT;
}
_cbSectorShift = ((CMSFHeaderData *)abHeader)->_uSectorShift;
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::Init\n"));
Err:
return ResultFromScode(sc);
}
//+--------------------------------------------------------------
//
// Member: CLayoutLockBytes::QueryInterface, public
//
// Synopsis: Returns an object for the requested interface
//
// Arguments: [iid] - Interface ID
// [ppvObj] - Object return
//
// Returns: Appropriate status code
//
// Modifies: [ppvObj]
//
// History: 26-Mar-92 DrewB Created
//
//---------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::QueryInterface(REFIID iid, void **ppvObj)
{
SCODE sc;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::QueryInterface(?, %p)\n",
ppvObj));
layChk(ValidateOutPtrBuffer(ppvObj));
*ppvObj = NULL;
sc = S_OK;
if (IsEqualIID(iid, IID_IUnknown))
{
*ppvObj = (IUnknown *)this;
CLayoutLockBytes::AddRef();
}
else if (IsEqualIID(iid, IID_ILockBytes))
{
*ppvObj = (ILockBytes *)this;
CLayoutLockBytes::AddRef();
}
else
{
sc = E_NOINTERFACE;
}
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::QueryInterface => %p\n",
ppvObj));
Err:
return ResultFromScode(sc);
}
//+--------------------------------------------------------------
//
// Member: CLayoutLockBytes::AddRef, public
//
// Synopsis: Increments the ref count
//
// Returns: Appropriate status code
//
// History: 16-Mar-92 DrewB Created
//
//---------------------------------------------------------------
STDMETHODIMP_(ULONG) CLayoutLockBytes::AddRef(void)
{
ULONG ulRet;
layDebugOut((DEB_TRACE, "In CLayoutLockBytes::AddRef()\n"));
InterlockedIncrement(&_cReferences);
ulRet = _cReferences;
layDebugOut((DEB_TRACE, "Out CLayoutLockBytes::AddRef\n"));
return ulRet;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::Release, public
//
// History: 20-Feb-96 SusiA Created
//
//----------------------------------------------------------------------------
STDMETHODIMP_(ULONG) CLayoutLockBytes::Release(void)
{
LONG lRet;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::Release:%p()\n", this));
lRet = InterlockedDecrement(&_cReferences);
if (lRet == 0)
{
delete this;
}
else if (lRet < 0)
{
lRet = 0;
}
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::Release\n"));
return lRet;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::ReadAt, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::ReadAt(ULARGE_INTEGER ulOffset,
VOID HUGEP *pv,
ULONG cb,
ULONG *pcbRead)
{
CSafeCriticalSection scs(&_cs);
SCODE sc = S_OK;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::ReadAt:%p()\n", this));
ULONG ulLow = ulOffset.LowPart;
LONG lHigh = (LONG)ulOffset.HighPart;
if ((_fLogging)&&(ulOffset.QuadPart >= sizeof(CMSFHeaderData)))
{
if (_hScript == INVALID_HANDLE_VALUE)
{
return STG_E_INVALIDHANDLE;
}
ULONG ulFirstSector = (ULONG) ((ulOffset.QuadPart -
sizeof(CMSFHeaderData))
>> _cbSectorShift);
ULONG ulLastSector = (ULONG) ((ulOffset.QuadPart + (cb - 1) -
sizeof(CMSFHeaderData))
>> _cbSectorShift);
ULONG ulSect;
ULONG cbScriptWritten;
for (ulSect = ulFirstSector; ulSect <= ulLastSector; ulSect++)
{
layAssert(_hScript !=INVALID_HANDLE_VALUE);
boolChk(WriteFile(_hScript,
(VOID *)&ulSect,
sizeof(ULONG),
&cbScriptWritten,
NULL));
if (cbScriptWritten != sizeof(ULONG))
{
return STG_E_WRITEFAULT;
}
}
}
negChk(SetFilePointer(_h,
ulLow,
&lHigh,
FILE_BEGIN));
boolChk(ReadFile(_h, pv, cb, pcbRead, NULL));
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::ReadAt\n"));
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::WriteAt, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::WriteAt(ULARGE_INTEGER ulOffset,
VOID const HUGEP *pv,
ULONG cb,
ULONG *pcbWritten)
{
CSafeCriticalSection scs(&_cs);
SCODE sc = S_OK;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::WriteAt:%p()\n", this));
ULONG ulLow = ulOffset.LowPart;
LONG lHigh = (LONG)ulOffset.HighPart;
if ((_fLogging)&&(ulOffset.QuadPart >= sizeof(CMSFHeaderData)))
{
if (_hScript == INVALID_HANDLE_VALUE)
{
return STG_E_INVALIDHANDLE;
}
ULONG ulFirstSector = (ULONG) ((ulOffset.QuadPart -
sizeof(CMSFHeaderData))
>> _cbSectorShift);
ULONG ulLastSector = (ULONG) ((ulOffset.QuadPart + (cb - 1) -
sizeof(CMSFHeaderData))
>> _cbSectorShift);
ULONG ulSect;
ULONG cbScriptWritten;
for (ulSect = ulFirstSector; ulSect <= ulLastSector; ulSect++)
{
boolChk(WriteFile(_hScript,
(VOID *)&ulSect,
sizeof(ULONG),
&cbScriptWritten,
NULL));
if (cbScriptWritten != sizeof(ULONG))
{
return STG_E_WRITEFAULT;
}
}
}
negChk(SetFilePointer(_h,
ulLow,
&lHigh,
FILE_BEGIN));
boolChk(WriteFile(_h, pv, cb, pcbWritten, NULL));
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::WriteAt\n"));
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::Flush, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::Flush(void)
{
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::Flush:%p()\n", this));
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::Flush\n"));
return S_OK;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::SetSize, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::SetSize(ULARGE_INTEGER cb)
{
CSafeCriticalSection scs(&_cs);
SCODE sc = S_OK;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::SetSize:%p()\n", this));
LONG lHigh = (LONG)cb.HighPart;
ULONG ulLow = cb.LowPart;
negChk(SetFilePointer(_h, ulLow, &lHigh, FILE_BEGIN));
boolChk(SetEndOfFile(_h));
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::SetSize\n"));
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::LockRegion, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::LockRegion(ULARGE_INTEGER libOffset,
ULARGE_INTEGER cb,
DWORD dwLockType)
{
CSafeCriticalSection scs(&_cs);
SCODE sc = S_OK;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::LockRegion:%p()\n", this));
if (dwLockType != LOCK_EXCLUSIVE && dwLockType != LOCK_ONLYONCE)
{
return STG_E_INVALIDFUNCTION;
}
boolChk(LockFile(_h,
libOffset.LowPart,
libOffset.HighPart,
cb.LowPart,
cb.HighPart));
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::LockRegion\n"));
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::UnlockRegion, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::UnlockRegion(ULARGE_INTEGER libOffset,
ULARGE_INTEGER cb,
DWORD dwLockType)
{
CSafeCriticalSection scs(&_cs);
SCODE sc = S_OK;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::UnlockRegion:%p()\n", this));
if (dwLockType != LOCK_EXCLUSIVE && dwLockType != LOCK_ONLYONCE)
{
return STG_E_INVALIDFUNCTION;
}
boolChk(UnlockFile(_h,
libOffset.LowPart,
libOffset.HighPart,
cb.LowPart,
cb.HighPart));
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::UnlockRegion\n"));
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::Stat, public
//
// Synopsis:
//
// Arguments:
//
// Returns: Appropriate status code
//
// Modifies:
//
// History: 20-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
STDMETHODIMP CLayoutLockBytes::Stat(STATSTG *pstatstg, DWORD grfStatFlag)
{
CSafeCriticalSection scs(&_cs);
SCODE sc;
layDebugOut((DEB_ITRACE, "In CLayoutLockBytes::Stat:%p()\n", this));
negChk(pstatstg->cbSize.LowPart =
GetFileSize(_h, &pstatstg->cbSize.HighPart));
boolChk(GetFileTime(_h, &pstatstg->ctime, &pstatstg->atime,
&pstatstg->mtime));
pstatstg->grfLocksSupported = LOCK_EXCLUSIVE | LOCK_ONLYONCE;
pstatstg->type = STGTY_LOCKBYTES;
pstatstg->grfMode = _grfMode;
pstatstg->pwcsName = NULL;
if ((grfStatFlag & STATFLAG_NONAME) == 0)
{
pstatstg->pwcsName = (OLECHAR *)CoTaskMemAlloc(
(lstrlenW(_awcName) + 1) * sizeof(OLECHAR));
if (pstatstg->pwcsName == NULL)
return STG_E_INSUFFICIENTMEMORY;
lstrcpyW(pstatstg->pwcsName, _awcName);
}
sc = S_OK;
layDebugOut((DEB_ITRACE, "Out CLayoutLockBytes::Stat\n"));
return NOERROR;
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::StartLogging, public
//
// Returns: Appropriate status code
//
// Modifies: _fLogging
//
// History: 24-Feb-96 SusiA Created
//
// Notes:
//
//----------------------------------------------------------------------------
SCODE CLayoutLockBytes::StartLogging(void)
{
CSafeCriticalSection scs(&_cs);
OLECHAR acTempPathName[MAX_PATH + 1];
SCODE sc = S_OK;
if (_fLogging)
{
return STG_E_INUSE; //logging already started!
}
if (_atcScriptName[0] != TEXT('\0'))
{
LONG dwDistanceToMoveHigh = 0;
//Script has already been started. Need to reopen it and seek
// to the end.
#ifndef UNICODE
_hScript = CreateFileA
#else
_hScript = CreateFile
#endif
(_atcScriptName,
GENERIC_WRITE, //Read-write
0, // No sharing
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS,
NULL);
if (_hScript == INVALID_HANDLE_VALUE)
{
return LAST_STG_SCODE;
}
negChk(SetFilePointer(_hScript, 0, &dwDistanceToMoveHigh, FILE_END));
}
else
{
TCHAR atcPath[MAX_PATH + 1];
//Generate the script name, then create it.
boolChk(GetTempPath(MAX_PATH, atcPath));
boolChk(GetTempFileName(atcPath, TEXT("SCR"), 0, _atcScriptName));
//GetTempFileName actually creates the file, so we open with
// OPEN_EXISTING
#ifndef UNICODE
_hScript = CreateFileA
#else
_hScript = CreateFile
#endif
(_atcScriptName,
GENERIC_READ | GENERIC_WRITE, //Read-write
0, // No sharing
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS,
NULL);
if (_hScript == INVALID_HANDLE_VALUE)
{
return LAST_STG_SCODE;
}
}
_fLogging = TRUE;
Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Member: CLayoutLockBytes::StopLogging, public
//
// Returns: Appropriate status code
//
// Modifies: _fLogging
//
// History: 24-Feb-96 SusiA Created
//
// Notes:
//
//-----------------------------------------------------------------------------
SCODE CLayoutLockBytes::StopLogging(void)
{
SCODE sc = S_OK;
CSafeCriticalSection scs(&_cs);
if (!_fLogging)
{
layAssert(FALSE && "StopLogging called when logging not yet started.");
return E_FAIL;
}
boolChk(CloseHandle(_hScript));
_hScript = INVALID_HANDLE_VALUE;
_fLogging = FALSE;
Err:
return sc;
}