NT4/private/ole32/stg/exp/marshl.cxx
2020-09-30 17:12:29 +02:00

1112 lines
32 KiB
C++

//+--------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1992.
//
// File: marshl.cxx
//
// Contents: Marshal/Unmarshal implementation
//
// History: 04-May-92 DrewB Created
//
//---------------------------------------------------------------
#include <exphead.cxx>
#pragma hdrstop
#include <expdf.hxx>
#include <expst.hxx>
#include <pbstream.hxx>
#include <marshl.hxx>
#include <logfile.hxx>
// Standard marshal data is an IID plus a DWORD
#define CBSTDMARSHALSIZE (sizeof(IID)+sizeof(DWORD))
#ifndef DCOM
STDAPI CoUnmarshalInterfaceEx(IStream *pStm,
REFIID riid,
void **ppv,
BOOL fNormalDoesRelease);
#endif
inline SCODE VerifyIid(REFIID iid, REFIID iidObj)
{
if ((IsEqualIID(iid, IID_IUnknown) || (IsEqualIID(iid, iidObj))))
{
return S_OK;
}
if (IsEqualIID(iidObj, IID_ILockBytes))
{
if (IsEqualIID(iid, IID_IFillLockBytes))
{
return S_OK;
}
}
if (IsEqualIID(iidObj, IID_IStorage))
{
if (IsEqualIID(iid, IID_IPropertySetStorage))
{
return S_OK;
}
}
return STG_E_INVALIDPARAMETER;
}
//+--------------------------------------------------------------
//
// Function: DfUnMarshalInterface, public
//
// Synopsis: Unmarshals marshaled data
//
// Arguments: [pstStm] - Stream to read data from
// [iid] - Interface to unmarshal
// [fFirst] - First time unmarshalling
// [ppvObj] - Interface return
//
// Returns: Appropriate status code
//
// Modifies: [ppvObj]
//
// History: 04-May-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_DfUnMarshalInterface) // Marshal_TEXT
#endif
STDAPI DfUnMarshalInterface(IStream *pstStm,
REFIID iid,
BOOL fFirst,
void **ppvObj)
{
SCODE sc;
ULONG cbRead;
IID iidSt;
DWORD mshlflags;
SafeIUnknown punk;
olLog(("--------::In DfUnMarshalInterface(%p, iid, %d, %p). "
"Context == %lX\n", pstStm, fFirst, ppvObj,
(ULONG)GetCurrentContextId()));
olDebugOut((DEB_TRACE, "In DfUnMarshalInterface("
"%p, ?, %d, %p)\n", pstStm, fFirst, ppvObj));
olChk(ValidateOutPtrBuffer(ppvObj));
*ppvObj = NULL;
olChk(ValidateInterface(pstStm, IID_IStream));
olChk(ValidateIid(iid));
if (!fFirst)
olErr(EH_Err, STG_E_INVALIDPARAMETER);
olHChk(pstStm->Read(&iidSt, sizeof(iidSt), &cbRead));
if (cbRead != sizeof(iidSt))
olErr(EH_Err, STG_E_READFAULT);
olHChk(pstStm->Read(&mshlflags, sizeof(mshlflags), &cbRead));
if (cbRead != sizeof(mshlflags))
olErr(EH_Err, STG_E_READFAULT);
olChk(VerifyIid(iid, iidSt));
#if !defined(MULTIHEAP)
olChk(DfSyncSharedMemory());
DfInitSharedMemBase();
#endif
if (IsEqualIID(iidSt, IID_ILockBytes))
sc = CFileStream::Unmarshal(pstStm, (void **)&punk, mshlflags);
else if (IsEqualIID(iidSt, IID_IStream))
sc = CExposedStream::Unmarshal(pstStm, (void **)&punk, mshlflags);
else if (IsEqualIID(iidSt, IID_IStorage))
sc = CExposedDocFile::Unmarshal(pstStm, (void **)&punk, mshlflags);
else
sc = E_NOINTERFACE;
if (SUCCEEDED(sc))
{
if (!IsEqualIID(iid, iidSt))
{
sc = punk->QueryInterface(iid, ppvObj);
}
else
{
TRANSFER_INTERFACE(punk, IUnknown, ppvObj);
#if DBG
void *pvCheck;
olAssert( S_OK == ((IUnknown*)*ppvObj)->QueryInterface(iidSt, &pvCheck));
olAssert( pvCheck == *ppvObj );
((IUnknown*)pvCheck)->Release();
#endif
}
}
olDebugOut((DEB_TRACE, "Out DfUnMarshalInterface => %p\n",
*ppvObj));
EH_Err:
olLog(("--------::Out DfUnMarshalInterface(). "
"*ppvObj == %p, ret == %lX\n", *ppvObj, sc));
return ResultFromScode(sc);
}
//+---------------------------------------------------------------------------
//
// Function: GetCoMarshalSize, private
//
// Synopsis: Gets the marshal size for an interface marshalled using
// CoMarshalInterface
//
// Arguments: [riid] - Interface id
// [punk] - Interface pointer
// [pv] - Context info
// [dwDestContext] - Destination context
// [pvDestContext] - Destination context
// [mshlflags] - Marshal flags
// [pcb] - Size return
//
// Returns: Appropriate status code
//
// Modifies: [pcb]
//
// Algorithm: CoMarshalInterface is guaranteed to add no more than
// MARSHALINTERFACE_MIN bytes of overhead to a marshal
// Also, the standard marshaller takes no more than that
// So if the given object supports IMarshal, the return
// is IMarshal::GetMarshalSizeMax+MARSHALINTERFACE_MIN,
// otherwise it is just MARSHALINTERFACE_MIN
//
// History: 03-Aug-93 DrewB Created
//
// Notes: On 32-bit platforms, we can use CoGetMarshalSizeMax
//
//----------------------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_GetCoMarshalSize)
#endif
#ifndef WIN32
static SCODE GetCoMarshalSize(REFIID riid,
IUnknown *punk,
void *pv,
DWORD dwDestContext,
void *pvDestContext,
DWORD mshlflags,
DWORD *pcb)
{
IMarshal *pmsh;
SCODE sc;
DWORD cb;
olDebugOut((DEB_ITRACE, "In GetCoMarshalSize("
"riid, %p, %p, %lu, %p, %lu, %p)\n", pv, punk, dwDestContext,
pvDestContext, mshlflags, pcb));
sc = DfGetScode(punk->QueryInterface(IID_IMarshal, (void **)&pmsh));
if (sc == E_NOINTERFACE)
{
*pcb = MARSHALINTERFACE_MIN;
sc = S_OK;
}
else if (SUCCEEDED(sc))
{
sc = DfGetScode(pmsh->GetMarshalSizeMax(riid, pv, dwDestContext,
pvDestContext, mshlflags,
&cb));
if (SUCCEEDED(sc))
*pcb = MARSHALINTERFACE_MIN+cb;
pmsh->Release();
}
olDebugOut((DEB_ITRACE, "Out GetCoMarshalSize => %lu, 0x%lX\n",
*pcb, sc));
return sc;
}
#else
#define GetCoMarshalSize(riid, punk, pv, dwDestContext, pvDestContext,\
mshlflags, pcb) \
GetScode(CoGetMarshalSizeMax(pcb, riid, punk, dwDestContext, \
pvDestContext, mshlflags))
#endif
//+--------------------------------------------------------------
//
// Function: GetStdMarshalSize, public
//
// Synopsis: Returns the size needed for a standard marshal buffer
//
// Arguments: [iid] - Requested marshal IID
// [iidObj] - IID of object being marshalled
// [dwDestContext] - Destination context
// [pvDestContext] - Unreferenced
// [mshlflags] - Marshal flags
// [pcbSize] - Size return
// [cbSize] - Object private size
// [ppc] - Context to marshal or NULL
// [fMarshalOriginal] - Marshal original in context
//
// Returns: Appropriate status code
//
// Modifies: [pcbSize]
//
// History: 04-May-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_GetStdMarshalSize)
#endif
SCODE GetStdMarshalSize(REFIID iid,
REFIID iidObj,
DWORD dwDestContext,
LPVOID pvDestContext,
DWORD mshlflags,
DWORD *pcbSize,
DWORD cbSize,
#ifdef ASYNC
CAsyncConnection *pcpoint,
BOOL fMarshalILBs,
#endif
CPerContext *ppc,
BOOL const fMarshalOriginal)
{
DWORD cbLBSize;
SCODE sc;
olDebugOut((DEB_ITRACE, "In GetStdMarshalSize("
"iid, iidObj, %lu, %p, %lu, %p, %lu, %p, %d)\n",
dwDestContext, pvDestContext, mshlflags, pcbSize, cbSize, ppc,
fMarshalOriginal));
olChk(ValidateOutBuffer(pcbSize, sizeof(DWORD)));
*pcbSize = 0;
olChk(ValidateIid(iid));
olChk(VerifyIid(iid, iidObj));
if (((dwDestContext != MSHCTX_LOCAL) && (dwDestContext != MSHCTX_INPROC))
|| pvDestContext != NULL)
olErr(EH_Err, STG_E_INVALIDFLAG);
*pcbSize = CBSTDMARSHALSIZE+cbSize;
#ifdef MULTIHEAP
*pcbSize += sizeof(ULONG)+sizeof(ContextId)+sizeof(CPerContext*);
#endif
#ifdef POINTER_IDENTITY
*pcbSize += sizeof(CMarshalList*);
#endif
#ifdef ASYNC
if ((ppc) && fMarshalILBs)
#else
if (ppc)
#endif
{
*pcbSize += sizeof(CGlobalContext *);
olChk(GetCoMarshalSize(IID_ILockBytes,
(ILockBytes *)ppc->GetBase(),
NULL, dwDestContext, pvDestContext,
mshlflags, &cbLBSize));
*pcbSize += cbLBSize;
olChk(GetCoMarshalSize(IID_ILockBytes,
(ILockBytes *)ppc->GetDirty(),
NULL, dwDestContext, pvDestContext,
mshlflags, &cbLBSize));
*pcbSize += cbLBSize;
if (fMarshalOriginal)
{
olChk(GetCoMarshalSize(IID_ILockBytes,
(ILockBytes *)ppc->GetOriginal(),
NULL, dwDestContext, pvDestContext,
mshlflags, &cbLBSize));
*pcbSize += cbLBSize;
}
}
#ifdef ASYNC
//BOOL determines whether we have a connection to marshal or not
*pcbSize += sizeof(BOOL);
if ((pcpoint) && (pcpoint->GetMarshalPoint() != NULL))
{
ULONG cbConnectSize;
//Async flags
*pcbSize += sizeof(DWORD);
olChk(GetCoMarshalSize(IID_IDocfileAsyncConnectionPoint,
pcpoint->GetMarshalPoint(),
NULL, dwDestContext, pvDestContext,
mshlflags, &cbConnectSize));
*pcbSize += cbConnectSize;
}
#endif
olDebugOut((DEB_ITRACE, "Out GetStdMarshalSize\n"));
EH_Err:
return sc;
}
//+--------------------------------------------------------------
//
// Member: StartMarshal, public
//
// Synopsis: Writes standard marshal header
//
// Arguments: [pstStm] - Stream to write marshal data into
// [iid] - Interface to marshal
// [iidObj] - Object being marshalled
// [mshlflags] - Marshal flags
//
// Returns: Appropriate status code
//
// History: 04-May-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_StartMarshal)
#endif
SCODE StartMarshal(IStream *pstStm,
REFIID iid,
REFIID iidObj,
DWORD mshlflags)
{
SCODE sc;
ULONG cbWritten;
olDebugOut((DEB_ITRACE, "In StartMarshal(%p, iid, iidObj, %lu)\n",
pstStm, mshlflags));
olChk(ValidateInterface(pstStm, IID_IStream));
olChk(ValidateIid(iid));
olChk(VerifyIid(iid, iidObj));
olHChk(pstStm->Write((void *)&iidObj, sizeof(iidObj), &cbWritten));
if (cbWritten != sizeof(iidObj))
olErr(EH_Err, STG_E_WRITEFAULT);
olHChk(pstStm->Write((void *)&mshlflags, sizeof(mshlflags), &cbWritten));
if (cbWritten != sizeof(mshlflags))
olErr(EH_Err, STG_E_WRITEFAULT);
olDebugOut((DEB_ITRACE, "Out StartMarshal\n"));
EH_Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Function: SkipStdMarshal, public
//
// Synopsis: Skips over the standard marshal data
//
// Arguments: [pstm] - Marshal stream
// [piid] - IID return
// [pmshlflags] - Return marshal flags
//
// Returns: Appropriate status code
//
// Modifies: [piid]
// [pmshlflags]
//
// History: 20-Nov-92 DrewB Created
//
//----------------------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_SkipStdMarshal)
#endif
#ifdef WIN32
SCODE SkipStdMarshal(IStream *pstm, IID *piid, DWORD *pmshlflags)
{
SCODE sc;
ULONG cbRead;
olDebugOut((DEB_ITRACE, "In SkipStdMarshal(%p, %p, %p)\n", pstm,
piid, pmshlflags));
olHChk(pstm->Read(piid, sizeof(IID), &cbRead));
if (cbRead != sizeof(IID))
olErr(EH_Err, STG_E_READFAULT);
olHChk(pstm->Read(pmshlflags, sizeof(DWORD), &cbRead));
if (cbRead != sizeof(DWORD))
olErr(EH_Err, STG_E_READFAULT);
olDebugOut((DEB_ITRACE, "Out SkipStdMarshal => %lX\n", sc));
EH_Err:
return sc;
}
#endif
//+--------------------------------------------------------------
//
// Function: MarshalPointer, public
//
// Synopsis: Marshals a pointer
//
// Arguments: [pstm] - Marshal stream
// [pv] - Pointer
//
// Returns: Appropriate status code
//
// History: 20-Aug-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_MarshalPointer)
#endif
SCODE MarshalPointer(IStream *pstm, void *pv)
{
SCODE sc;
ULONG cbWritten;
olDebugOut((DEB_ITRACE, "In MarshalPointer(%p, %p)\n", pstm, pv));
#ifdef USEBASED
pv = (void *)((ULONG)pv-(ULONG)DFBASEPTR);
#endif
sc = DfGetScode(pstm->Write(&pv, sizeof(pv), &cbWritten));
if (SUCCEEDED(sc) && cbWritten != sizeof(pv))
sc = STG_E_WRITEFAULT;
olDebugOut((DEB_ITRACE, "Out MarshalPointer\n"));
return sc;
}
//+--------------------------------------------------------------
//
// Function: MarshalContext, public
//
// Synopsis: Marshals a context
//
// Arguments: [pstm] - Marshal stream
// [ppc] - Context
// [dwDestContext] - Destination context
// [pvDestContext] - Unreferenced
// [mshlflags] - Marshal flags
// [fMarshalOriginal] - Marshal original or not
//
// Returns: Appropriate status code
//
// History: 20-Aug-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_MarshalContext)
#endif
SCODE MarshalContext(IStream *pstm,
CPerContext *ppc,
DWORD dwDestContext,
LPVOID pvDestContext,
DWORD mshlflags,
#ifdef ASYNC
BOOL const fMarshalILBs,
#endif
BOOL const fMarshalOriginal)
{
SCODE sc;
olDebugOut((DEB_ITRACE, "In MarshalContext(%p, %p, %lu, %p, %lu, %d)\n",
pstm, ppc, dwDestContext, pvDestContext, mshlflags,
fMarshalOriginal));
olChk(MarshalPointer(pstm, ppc->GetGlobal()));
#ifdef ASYNC
if (fMarshalILBs)
#endif
{
olHChk(CoMarshalInterface(pstm, IID_ILockBytes, ppc->GetBase(),
dwDestContext, pvDestContext, mshlflags));
olHChk(CoMarshalInterface(pstm, IID_ILockBytes,
(ILockBytes *)ppc->GetDirty(),
dwDestContext, pvDestContext, mshlflags));
if (fMarshalOriginal)
olHChk(CoMarshalInterface(pstm, IID_ILockBytes, ppc->GetOriginal(),
dwDestContext, pvDestContext, mshlflags));
}
#ifdef WIN32
if (mshlflags != MSHLFLAGS_TABLEWEAK)
ppc->GetGlobal()->AddRef();
#endif
olDebugOut((DEB_ITRACE, "Out MarshalContext\n"));
EH_Err:
return sc;
}
//+--------------------------------------------------------------
//
// Function: UnmarshalPointer, public
//
// Synopsis: Unmarshals a pointer
//
// Arguments: [pstm] - Marshal stream
// [ppv] - Pointer return
//
// Returns: Appropriate status code
//
// Modifies: [ppv]
//
// History: 20-Aug-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_UnmarshalPointer)
#endif
SCODE UnmarshalPointer(IStream *pstm,
void **ppv)
{
SCODE sc;
ULONG cbRead;
olDebugOut((DEB_ITRACE, "In UnmarshalPointer(%p, %p)\n", pstm, ppv));
sc = DfGetScode(pstm->Read(ppv, sizeof(*ppv), &cbRead));
if (SUCCEEDED(sc) && cbRead != sizeof(*ppv))
sc = STG_E_READFAULT;
#ifdef USEBASED
*ppv = (void *)((ULONG)*ppv+(ULONG)DFBASEPTR);
#endif
olDebugOut((DEB_ITRACE, "Out UnmarshalPointer => %p\n", *ppv));
return sc;
}
//+--------------------------------------------------------------
//
// Function: UnmarshalContext, public
//
// Synopsis: Unmarshals a context
//
// Arguments: [pstm] - Marshal stream
// [pppc] - Context return
// [fUnmarshalOriginal] - Marshalled original exists or not
// [fIsRoot] - Root unmarshal or not
//
// Returns: Appropriate status code
//
// Modifies: [pppc]
//
// History: 20-Aug-92 DrewB Created
//
//---------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_UnmarshalContext)
#endif
SCODE UnmarshalContext(IStream *pstm,
CGlobalContext *pgc,
CPerContext **pppc,
DWORD mshlflags,
#ifdef ASYNC
BOOL const fUnmarshalILBs,
#endif
BOOL const fUnmarshalOriginal,
#ifdef MULTIHEAP
ContextId cntxid,
#endif
BOOL const fIsRoot)
{
BOOL fNewContext;
ILockBytes *plkbBase = NULL;
CFileStream *pfstDirty = NULL;
ILockBytes *plkbOriginal = NULL;
SCODE sc, sc2;
CPerContext *ppc;
ULONG ulOpenLock = 0;
olDebugOut((DEB_ITRACE, "In UnmarshalContext(%p, %p, %lu, %d, %d)\n",
pstm, pppc, mshlflags, fUnmarshalOriginal, fIsRoot));
ppc = pgc->Find(GetCurrentContextId());
fNewContext = (ppc == NULL);
#ifdef MULTIHEAP
// when marshaling to the same process, use the same heap
// when marshaling to a different process, check the context list
// if there is a matching percontext, use that heap
if (GetCurrentContextId() != cntxid && ppc != NULL)
{
ppc->SetThreadAllocatorState(NULL); // set new base
// Whenever we unmarshal into a different process, we create
// a new mapping (of the same heap),
// even if a mapping of the same heap may already exist in
// the same process. For pointer identity, it is essential
// that we find and use the existing heap.
// process A ---marshal---> process B ---marshal----> process A
// The "final" unmarshaled exposed object in process A should
// match the original pointer used when the exposed object
// was originally marshaled. To do this, we check the global
// context list, and if there's a percontext match, we use
// its allocator and heap mapping (and don't create a new one).
// However, to actually search the global context list (it
// lives in shared memory), we need a temporary mapping until
// a matching percontext can be found and reused.
// If not, then a new percontext is allocated and the temporary
// mapping becomes "permanent" for the lifetime of the new percontext.
}
#endif
//BUGBUG: Fix error path
if (fNewContext)
{
olMemTo(EH_Open,
ppc = new (pgc->GetMalloc()) CPerContext(pgc->GetMalloc()));
olChkTo(EH_ppc, ppc->InitFromGlobal(pgc));
}
#ifdef MULTIHEAP
// take the ownership of the heap away from the temporary
ppc->SetAllocatorState (NULL, &g_smAllocator);
//ppc from above may have used incorrect base (base of temporary heap).
// Since we're returning and storing an unbased pointer, we need to get
// the real absolute pointer here. At this point, ppc will always be in
// the context list, so we don't need to worry about a NULL return.
ppc = pgc->Find(GetCurrentContextId());
olAssert(ppc != NULL);
#endif
#ifdef ASYNC
if (fUnmarshalILBs)
{
#endif
#ifdef DCOM
// attempt to unmarshal all the interfaces first. this makes cleanup
// easier.
sc = CoUnmarshalInterface(pstm, IID_ILockBytes, (void **)&plkbBase);
sc2 = CoUnmarshalInterface(pstm, IID_ILockBytes, (void **)&pfstDirty);
sc = (SUCCEEDED(sc)) ? sc2 : sc; // sc = first failure code (if any)
if (fUnmarshalOriginal)
{
sc2 = CoUnmarshalInterface(pstm, IID_ILockBytes,
(void **)&plkbOriginal);
sc = (SUCCEEDED(sc)) ? sc2 : sc; // sc = first failure code (if any)
}
// cleanup if any failure so far
olChkTo(EH_plkbOriginal, sc);
if (ppc->GetBase() != NULL)
{
// already have context, just release the things we unmarshaled.
plkbBase->Release();
plkbBase = NULL;
}
if (ppc->GetDirty() != NULL)
{
pfstDirty->Release();
pfstDirty = NULL;
}
if ((plkbOriginal) && (ppc->GetOriginal() != NULL))
{
plkbOriginal->Release();
plkbOriginal = NULL;
}
else if ((NULL == plkbOriginal) && plkbBase)
{
plkbBase->AddRef();
plkbOriginal = plkbBase;
}
olAssert (plkbOriginal != NULL || ppc->GetOriginal() != NULL);
#else
olHChkTo(EH_pgc,
CoUnmarshalInterfaceEx(pstm, IID_ILockBytes, (void **)&plkbBase,
FALSE /*fNormalDoesRelease*/));
if (ppc->GetBase() != NULL)
{
plkbBase->Release();
plkbBase = NULL;
}
olHChkTo(EH_plkbBase,
CoUnmarshalInterfaceEx(pstm, IID_ILockBytes, (void **)&pfstDirty,
FALSE /*fNormalDoesRelease*/));
if (ppc->GetDirty() != NULL)
{
pfstDirty->Release();
pfstDirty = NULL;
}
if (fUnmarshalOriginal)
{
olHChkTo(EH_pfstDirty,
CoUnmarshalInterfaceEx(pstm, IID_ILockBytes,
(void **)&plkbOriginal,
FALSE /*fNormalDoesRelease*/));
if (ppc->GetOriginal() != NULL)
{
plkbOriginal->Release();
plkbOriginal = NULL;
}
}
else if (fNewContext || (ppc->GetOriginal() == NULL))
{
plkbBase->AddRef();
plkbOriginal = plkbBase;
}
else
{
plkbOriginal = NULL;
}
#endif
// Make sure there is a reserved handle if this is a root
// file-based lockbytes
if (fIsRoot)
{
IFileLockBytes *pflkb;
if (SUCCEEDED(DfGetScode((plkbOriginal ? plkbOriginal :
ppc->GetOriginal())->
QueryInterface(IID_IFileLockBytes,
(void **)&pflkb))))
{
sc = DfGetScode(pflkb->ReserveHandle());
pflkb->Release();
olChkTo(EH_plkbOriginal, sc);
}
}
#ifdef ASYNC
}
#endif
if (fNewContext)
{
olAssert(plkbOriginal != NULL);
// Take open locks if necessary
if (fIsRoot && pgc->TakeLock())
{
olChkTo(EH_plkbOriginal,
GetOpen(plkbOriginal, pgc->GetOpenLockFlags(),
FALSE, &ulOpenLock));
}
ppc->SetILBInfo(plkbBase, pfstDirty, plkbOriginal, ulOpenLock);
}
else
{
if (ppc->GetBase() == NULL)
{
//Fill in the ILB fields
ppc->SetILBInfo(plkbBase, pfstDirty, plkbOriginal, ulOpenLock);
}
ppc->AddRef();
}
*pppc = ppc;
olDebugOut((DEB_ITRACE, "Out UnmarshalContext => %p\n", *pppc));
return S_OK;
EH_ppc:
// Preserve plkbOriginal so the lock is released even after the
// context releases things;
plkbOriginal->AddRef();
ppc->Release();
pfstDirty = NULL;
plkbBase = NULL;
EH_Open:
if (ulOpenLock != 0)
{
olAssert(plkbOriginal != NULL);
ReleaseOpen(plkbOriginal, pgc->GetOpenLockFlags(), ulOpenLock);
}
EH_plkbOriginal:
if (plkbOriginal)
plkbOriginal->Release();
#ifdef DCOM
// compiler complains about unreferenced lables
if (pfstDirty)
pfstDirty->Release();
if (plkbBase)
plkbBase->Release();
#else
EH_pfstDirty:
if (pfstDirty)
pfstDirty->Release();
EH_plkbBase:
if (plkbBase)
plkbBase->Release();
EH_pgc:
#endif
return sc;
}
//+---------------------------------------------------------------------------
//
// Function: ReleaseContext, public
//
// Synopsis: Releases references for a context's marshal data
//
// Arguments: [pstm] - Marshal stream
// [fHasOriginal] - Original is marshalled
// [mshlflags] - Marshal flags
//
// Returns: Appropriate status code
//
// History: 20-Nov-92 DrewB Created
//
//----------------------------------------------------------------------------
#ifdef CODESEGMENTS
#pragma code_seg(SEG_ReleaseContext)
#endif
#ifdef WIN32
SCODE ReleaseContext(IStream *pstm,
#ifdef ASYNC
BOOL const fUnmarshalILBs,
#endif
BOOL const fHasOriginal,
DWORD mshlflags)
{
CGlobalContext *pgc;
SCODE sc;
olDebugOut((DEB_ITRACE, "In ReleaseContext(%p, %d, %lu)\n", pstm,
fHasOriginal, mshlflags));
olChk(UnmarshalPointer(pstm, (void **)&pgc));
if (fUnmarshalILBs)
{
olHChk(CoReleaseMarshalData(pstm));
olHChk(CoReleaseMarshalData(pstm));
if (fHasOriginal)
olHChk(CoReleaseMarshalData(pstm));
}
if (mshlflags != MSHLFLAGS_TABLEWEAK)
pgc->Release();
olDebugOut((DEB_ITRACE, "Out ReleaseContext\n"));
EH_Err:
return sc;
}
#endif
#ifdef MULTIHEAP
//+---------------------------------------------------------------------------
//
// Function: MarshalSharedMemory, public
//
// Synopsis: marshals the shared memory context
//
// Arguments: [pstm] - Marshal stream
// [ppc] - per context structure
//
// Returns: Appropriate status code
//
// History: 02-Dec-95 HenryLee Created
//
//----------------------------------------------------------------------------
SCODE MarshalSharedMemory (IStream *pstStm, CPerContext *ppc)
{
SCODE sc = S_OK;
ULONG cbWritten;
ULONG ulHeapName;
ContextId cntxid = GetCurrentContextId();
ulHeapName = g_smAllocator.GetHeapName();
olHChk(pstStm->Write((void*) &ulHeapName, sizeof(ulHeapName), &cbWritten));
if (cbWritten != sizeof(ulHeapName))
olErr(EH_Err, STG_E_WRITEFAULT);
olHChk(pstStm->Write((void*) &cntxid, sizeof(cntxid), &cbWritten));
if (cbWritten != sizeof(cntxid))
olErr(EH_Err, STG_E_WRITEFAULT);
olHChk(pstStm->Write((void*) &ppc, sizeof(ppc), &cbWritten));
if (cbWritten != sizeof(ppc))
olErr(EH_Err, STG_E_WRITEFAULT);
EH_Err:
return sc;
}
//+---------------------------------------------------------------------------
//
// Function: UnMarshalSharedMemory, public
//
// Synopsis: Unmarshals the shared memory context
//
// Arguments: [pstm] - Marshal stream
//
// Returns: Appropriate status code
//
// History: 02-Dec-95 HenryLee Created
//
//----------------------------------------------------------------------------
SCODE UnmarshalSharedMemory (IStream *pstStm, DWORD mshlflags,
CPerContext *ppcOwner, ContextId *pcntxid)
{
SCODE sc = S_OK;
ULONG cbRead;
ULONG ulHeapName;
ContextId cntxid;
CPerContext *ppc;
olHChk(pstStm->Read(&ulHeapName, sizeof(ulHeapName), &cbRead));
if (cbRead != sizeof(ulHeapName))
olErr(EH_Err, STG_E_READFAULT);
olHChk(pstStm->Read(&cntxid, sizeof(cntxid), &cbRead));
if (cbRead != sizeof(cntxid))
olErr(EH_Err, STG_E_READFAULT);
olHChk(pstStm->Read(&ppc, sizeof(ppc), &cbRead));
if (cbRead != sizeof(ppc))
olErr(EH_Err, STG_E_READFAULT);
*pcntxid = cntxid;
if (GetCurrentContextId() == cntxid)
{
// marshaling to the same process, reuse the per context and heap
// in the case of marshaling to another thread
// the per context takes ownership of the thread's allocator
ppc->SetThreadAllocatorState(NULL);
}
else
{
// marshaling to another process on the same machine
// if the name of heap is different that current one, open it
if (g_smAllocator.GetHeapName() != ulHeapName)
{
DfInitSharedMemBase();
olChk(DfSyncSharedMemory(ulHeapName));
}
// Because the unmarshaling code calls IStream::Read,
// possibly using another shared heap, we need a temporary
// owner until the real CPerContext is unmarshaled
ppcOwner->GetThreadAllocatorState();
ppcOwner->SetThreadAllocatorState(NULL);
}
EH_Err:
return sc;
}
#endif
#ifdef ASYNC
SCODE MarshalConnection(IStream *pstm,
CAsyncConnection *pcpoint,
DWORD dwDestContext,
LPVOID pvDestContext,
DWORD mshlflags)
{
SCODE sc;
ULONG cbWritten;
IDocfileAsyncConnectionPoint *pdacp = pcpoint->GetMarshalPoint();
BOOL fIsInitialized = (pdacp != NULL);
//Write out the pointer.
olHChk(pstm->Write(&fIsInitialized,
sizeof(BOOL),
&cbWritten));
if (cbWritten != sizeof(BOOL))
{
olErr(EH_Err, STG_E_READFAULT);
}
if (fIsInitialized)
{
//If the pointer was NULL, we don't need to worry about actually
//marshalling anything, and we can detect this in the unmarshal
//path. If it wasn't NULL, we need to store some additional
//information: The async flags and the actual connection point,
//which will be standard marshalled.
DWORD dwAsyncFlags = pcpoint->GetAsyncFlags();
olChk(pstm->Write(&dwAsyncFlags, sizeof(DWORD), &cbWritten));
if (cbWritten != sizeof(DWORD))
{
olErr(EH_Err, STG_E_WRITEFAULT);
}
//Finally, standard marshal the connection point itself.
olHChk(CoMarshalInterface(pstm,
IID_IDocfileAsyncConnectionPoint,
pdacp,
dwDestContext,
pvDestContext,
mshlflags));
}
EH_Err:
return sc;
}
SCODE UnmarshalConnection(IStream *pstm,
DWORD *pdwAsyncFlags,
IDocfileAsyncConnectionPoint **ppdacp,
DWORD mshlflags)
{
SCODE sc;
BOOL fIsInitialized;
ULONG cbRead;
*ppdacp = NULL;
*pdwAsyncFlags = 0;
olHChk(pstm->Read(&fIsInitialized, sizeof(BOOL), &cbRead));
if (cbRead != sizeof(BOOL))
{
olErr(EH_Err, STG_E_READFAULT);
}
if (fIsInitialized)
{
olChk(pstm->Read(pdwAsyncFlags, sizeof(DWORD), &cbRead));
if (cbRead != sizeof(DWORD))
{
olErr(EH_Err, STG_E_READFAULT);
}
sc = CoUnmarshalInterface(pstm,
IID_IDocfileAsyncConnectionPoint,
(void **)ppdacp);
}
EH_Err:
return sc;
}
SCODE ReleaseConnection(IStream *pstm, DWORD mshlflags)
{
SCODE sc;
ULONG cbRead;
BOOL fIsInitialized;
DWORD dwAsyncFlags;
olHChk(pstm->Read(&fIsInitialized, sizeof(BOOL), &cbRead));
if (cbRead != sizeof(BOOL))
{
olErr(EH_Err, STG_E_READFAULT);
}
if (fIsInitialized)
{
olChk(pstm->Read(&dwAsyncFlags, sizeof(DWORD), &cbRead));
if (cbRead != sizeof(DWORD))
{
olErr(EH_Err, STG_E_READFAULT);
}
olHChk(CoReleaseMarshalData(pstm));
}
EH_Err:
return sc;
}
#endif