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.
395 lines
8.9 KiB
395 lines
8.9 KiB
//+-------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1996.
|
|
//
|
|
// File: ilbmem.cx
|
|
//
|
|
// Contents: ILockBytes memory implementation
|
|
//
|
|
// Classes: CMapBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#include "headers.cxx"
|
|
|
|
#include <memory.h>
|
|
#include "ilb.hxx"
|
|
|
|
#if DBG == 1
|
|
#ifndef _UNIX // on unix we use static lib, so the symbol is defined
|
|
DECLARE_INFOLEVEL(ol, DEB_ERROR);
|
|
#endif
|
|
#endif
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::CMapBytes, public
|
|
//
|
|
// Synopsis: constructor
|
|
//
|
|
// Effects: initialize member variables
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
// Notes: Returns a fully initialized CMapBytes (ref count == 1)
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CMapBytes::CMapBytes(void)
|
|
{
|
|
_ulSize = 0;
|
|
_pv = 0;
|
|
|
|
_ulRef = 1;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::QueryInterface, public
|
|
//
|
|
// Arguments: [riid] - interface id
|
|
// [ppvObj] - place holder for interface
|
|
//
|
|
// Returns: Always fails
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
// Notes: Not used in tests
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
|
|
{
|
|
*ppvObj = NULL;
|
|
return ResultFromScode(STG_E_INVALIDFUNCTION);
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::AddRef, public
|
|
//
|
|
// Synopsis: add reference
|
|
//
|
|
// Returns: post reference count
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP_(ULONG) CMapBytes::AddRef(void)
|
|
{
|
|
_ulRef++;
|
|
return(_ulRef);
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::Release, public
|
|
//
|
|
// Synopsis: release reference
|
|
//
|
|
// Effects: deletes object when reference count reaches zero
|
|
//
|
|
// Returns: post reference count
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP_(ULONG) CMapBytes::Release(void)
|
|
{
|
|
_ulRef--;
|
|
|
|
if (_ulRef > 0)
|
|
return(_ulRef);
|
|
|
|
free(_pv);
|
|
|
|
delete this;
|
|
|
|
return(0);
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::ReadAt
|
|
//
|
|
// Synopsis: Reads bytes from memory
|
|
//
|
|
// Arguments: [ulOffset] - byte offset
|
|
// [pv] - input buffer
|
|
// [cb] - count of bytes to read
|
|
// [pcbRead] - count of bytes read
|
|
//
|
|
// Returns: SCODE
|
|
//
|
|
// Modifies: pv, pcbRead
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::ReadAt(ULARGE_INTEGER uliOffset,
|
|
VOID HUGEP *pv,
|
|
ULONG cb,
|
|
ULONG *pcbRead)
|
|
{
|
|
olAssert(ULIGetHigh(uliOffset) == 0);
|
|
|
|
ULONG ulOffset = ULIGetLow(uliOffset);
|
|
if (ulOffset >= _ulSize)
|
|
{
|
|
// truncate read
|
|
cb = 0;
|
|
}
|
|
else if (cb > (_ulSize - ulOffset))
|
|
{
|
|
// truncate range that exceeds size
|
|
cb = _ulSize - ulOffset;
|
|
}
|
|
|
|
memcpy(pv, (void*)(((BYTE*)_pv) + ulOffset), (size_t) cb);
|
|
*pcbRead = cb;
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::WriteAt, public
|
|
//
|
|
// Synopsis: Writes bytes to memory
|
|
//
|
|
// Effects: May change memory size
|
|
//
|
|
// Arguments: [uliOffset] - byte offset
|
|
// [pv] - output buffer
|
|
// [cb] - count of bytes to write
|
|
// [pcbWritten] - count of bytes written
|
|
//
|
|
// Returns: SCODE
|
|
//
|
|
// Modifies: pcbWritten
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
// Notes: This implementation doesn't write partial buffers.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::WriteAt(ULARGE_INTEGER uliOffset,
|
|
VOID const HUGEP *pv,
|
|
ULONG cb,
|
|
ULONG FAR *pcbWritten)
|
|
{
|
|
olAssert(ULIGetHigh(uliOffset) == 0);
|
|
|
|
ULONG ulOffset = ULIGetLow(uliOffset);
|
|
HRESULT hr;
|
|
|
|
if (ulOffset + cb > _ulSize)
|
|
{
|
|
// increase memory buffer to accomodate write
|
|
|
|
ULARGE_INTEGER uliSize;
|
|
|
|
ULISetHigh(uliSize, 0);
|
|
ULISetLow(uliSize, ulOffset + cb);
|
|
hr = SetSize(uliSize);
|
|
|
|
if (FAILED(DfGetScode(hr)))
|
|
{
|
|
// don't bother writing partial buffers
|
|
|
|
*pcbWritten = 0;
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
memcpy((void *)(((BYTE*)_pv) + ulOffset), pv, (size_t) cb);
|
|
*pcbWritten = cb;
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::Flush, public
|
|
//
|
|
// Synopsis: flushes memory - not appropriate for this implementation
|
|
//
|
|
// Effects: none
|
|
//
|
|
// Returns: SUCCESS_SUCCESS
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::Flush(void)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::GetSize, public
|
|
//
|
|
// Synopsis: gets memory buffer size
|
|
//
|
|
// Arguments: [pcb] - size place holder
|
|
//
|
|
// Returns: SUCCESS_SUCCESS
|
|
//
|
|
// Modifies: pcb
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::GetSize(ULARGE_INTEGER FAR *pcb)
|
|
{
|
|
ULISetHigh(*pcb, 0);
|
|
ULISetLow(*pcb, _ulSize);
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::SetSize, public
|
|
//
|
|
// Synopsis: sets memory buffer size
|
|
//
|
|
// Effects: may change buffer size
|
|
//
|
|
// Arguments: [ulicb] - new memory size
|
|
//
|
|
// Returns: SCODE
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
// Algorithm: realloc the buffer
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::SetSize(ULARGE_INTEGER ulicb)
|
|
{
|
|
olAssert(ULIGetHigh(ulicb) == 0);
|
|
|
|
ULONG cb = ULIGetLow(ulicb);
|
|
|
|
if (cb == _ulSize)
|
|
return NOERROR;
|
|
|
|
void *pv = realloc(_pv, (size_t) cb);
|
|
|
|
if ((cb > 0) && (pv == NULL))
|
|
{
|
|
// Unable to allocate memory
|
|
// Leave current memory and size alone
|
|
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
_pv = pv;
|
|
_ulSize = cb;
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::LockRegion, public
|
|
//
|
|
// Synopsis: not supported (intentionally)
|
|
//
|
|
// Effects: asserts if called
|
|
//
|
|
// Arguments: [libOffset] - lock range offset
|
|
// [cb] - lock range size
|
|
// [dwLockType] - lock type
|
|
//
|
|
// Returns: STG_E_INVALIDFUNCTION
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::LockRegion(ULARGE_INTEGER libOffset,
|
|
ULARGE_INTEGER cb,
|
|
DWORD dwLockType)
|
|
{
|
|
olAssert(0 && "Can't lock CMapBytes");
|
|
return ResultFromScode(STG_E_INVALIDFUNCTION);
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::UnLockRegion, public
|
|
//
|
|
// Synopsis: not supported (intentionally)
|
|
//
|
|
// Effects: asserts if called
|
|
//
|
|
// Arguments: [libOffset] - lock range offset
|
|
// [cb] - lock range size
|
|
// [dwLockType] - lock type
|
|
//
|
|
// Returns: STG_E_INVALIDFUNCTION
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::UnlockRegion(ULARGE_INTEGER libOffset,
|
|
ULARGE_INTEGER cb,
|
|
DWORD dwLockType)
|
|
{
|
|
olAssert(0 && "Can't unlock CMapBytes");
|
|
return ResultFromScode(STG_E_INVALIDFUNCTION);
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CMapBytes::Stat, public
|
|
//
|
|
// Synopsis: Provide instance information
|
|
//
|
|
// Arguments: [pstatstg] - status buffer
|
|
// [grfStatFlag] - status flags
|
|
//
|
|
// Returns: SCODE
|
|
//
|
|
// Modifies: pstatstg
|
|
//
|
|
// Derivation: ILockBytes
|
|
//
|
|
// Notes: No time stamps
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CMapBytes::Stat(STATSTG FAR *pstatstg, DWORD grfStatFlag)
|
|
{
|
|
memset(pstatstg, 0, sizeof(STATSTG));
|
|
|
|
if ((grfStatFlag & STATFLAG_NONAME) == 0)
|
|
{
|
|
static char const abName[] = "Memory";
|
|
|
|
HRESULT hr;
|
|
|
|
if (FAILED(DfGetScode(hr = drtMemAlloc(sizeof(abName),
|
|
(void **) &pstatstg->pwcsName))))
|
|
return hr;
|
|
|
|
memcpy(pstatstg->pwcsName, abName, sizeof(abName));
|
|
}
|
|
|
|
pstatstg->type = STGTY_LOCKBYTES;
|
|
|
|
ULISetHigh(pstatstg->cbSize, 0);
|
|
ULISetLow(pstatstg->cbSize, _ulSize);
|
|
|
|
pstatstg->grfMode = STGM_READWRITE | STGM_DIRECT | STGM_SHARE_EXCLUSIVE;
|
|
|
|
return NOERROR;
|
|
}
|