Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

2010 lines
61 KiB

//+--------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1997
//
// File: pipes.cxx
//
// History:
// RichN 10/30/97 Created.
//
//---------------------------------------------------------------------------
#include <ole2int.h>
#include "pipes.hxx"
/////////////////////////////////////////////////////////////////////////////
// Externals
EXTERN_C HRESULT PrxDllGetClassObject(REFCLSID clsid, REFIID iid, void **ppv);
extern HRESULT RemUnkPSGetClassObject(REFIID riid, PVOID* ppv);
//////////////////////////////////////////////////////////////////////////////
// Constants
const DWORD WAIT_INFINITE = DWORD(-1);
const ULONG FRAGMENT_SIZE = 1300;
typedef struct
{
const char *key;
const char *value;
} RegistryKeyValue;
const RegistryKeyValue REG_CONST_KEY[] =
{
"CLSID\\{0000032E-0000-0000-C000-000000000046}", "PipePSFactory",
"CLSID\\{0000032E-0000-0000-C000-000000000046}\\InprocServer32", "ole32.dll",
"Interface\\{DB2F3ACA-2F86-11d1-8E04-00C04FB9989A}", "IPipeByte",
"Interface\\{DB2F3ACA-2F86-11d1-8E04-00C04FB9989A}\\ProxyStubClsid32", "{0000032E-0000-0000-C000-000000000046}",
"Interface\\{DB2F3ACC-2F86-11d1-8E04-00C04FB9989A}", "IPipeLong",
"Interface\\{DB2F3ACC-2F86-11d1-8E04-00C04FB9989A}\\ProxyStubClsid32", "{0000032E-0000-0000-C000-000000000046}",
"Interface\\{DB2F3ACE-2F86-11d1-8E04-00C04FB9989A}", "IPipeDouble",
"Interface\\{DB2F3ACE-2F86-11d1-8E04-00C04FB9989A}\\ProxyStubClsid32", "{0000032E-0000-0000-C000-000000000046}",
// Indicates end of list.
"", ""
};
/////////////////////////////////////////////////////////////////////////////
// Macros
inline ULONG MIN( ULONG a, ULONG b )
{
return a < b ? a : b;
}
inline ULONG MAX( ULONG a, ULONG b )
{
return a < b ? b : a;
}
inline HRESULT MAKE_WIN32( HRESULT status )
{
return MAKE_SCODE( SEVERITY_ERROR, FACILITY_WIN32, status );
}
inline HRESULT FIX_WIN32( HRESULT result )
{
if ((ULONG) result > 0xfffffff7 || (ULONG) result < 0x2000)
return MAKE_WIN32( result );
else
return result;
}
/////////////////////////////////////////////////////////////////////////////
// Globals
#define DISABLEASYNC 0
/////////////////////////////////////////////////////////////////////////////
// Prototypes
//+**************************************************************************
// FixUpPipeRegistry(void)
//
// Description: Modifies the registry to have the pipe interface point
// to a different class ID for the PSFactory. Adds to the
// registry the new ID for the PipePSFactory.
//
// History:
// Date: Time: Developer: Action:
// 12/3/97 10:14:48 AM RichN Created.
//
// Notes: We do not change the async interfaces. They should still
// be handled by ole32 directly. Only modify the synchronous varity.
//
//-**************************************************************************
HRESULT FixUpPipeRegistry(void)
{
HRESULT result = ERROR_SUCCESS;
// Create the Pipe interfaces and add the clsid for the PipePSFactory.
for (int i = 0; (REG_CONST_KEY[i].key[0] != '\0') && result == ERROR_SUCCESS; i++)
{
// Use Ascii so it works on Win95.
result = RegSetValueA(
HKEY_CLASSES_ROOT,
REG_CONST_KEY[i].key,
REG_SZ,
REG_CONST_KEY[i].value,
strlen(REG_CONST_KEY[i].value) );
}
if( result != ERROR_SUCCESS )
return FIX_WIN32( result );
return S_OK;
}
//+**************************************************************************
// ProxyDllGetClassObject(REFCLSID clsid, REFIID iid, void **ppv)
//
// Description: Creates a proxy. Trys PrxDllGetClassObject first since that
// is the most likely. If not, then sees if it is the pipe
// proxy being created.
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 11:43:55 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
EXTERN_C HRESULT ProxyDllGetClassObject(REFCLSID clsid, REFIID riid, void **ppv)
{
ComDebOut(( DEB_MARSHAL, "ProxyDllGetClassObject, clsid: %l, riid: %l \n",
clsid, riid));
HRESULT hr = PrxDllGetClassObject(clsid, riid, ppv);
if( FAILED(hr) )
{
// Not a well known one, maybe it is the pipe factory.
if( clsid == CLSID_PipePSFactory )
{
// Create the pipe proxy/stub class factory
CPipePSFactory *pPipePSFactory = new CPipePSFactory();
if( NULL != pPipePSFactory )
{
// Get the interface Requested.
hr = pPipePSFactory->QueryInterface(riid, ppv);
pPipePSFactory->Release();
}
else
{
*ppv = NULL;
hr = E_OUTOFMEMORY;
}
}
else if (clsid == CLSID_RemoteUnknownPSFactory)
{
hr = RemUnkPSGetClassObject(riid, ppv);
}
}
return hr;
}
//+**************************************************************************
// CPipePSFactory()
//
// Description: CTOR
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 12:55:55 PM RichN Created.
//
// Notes:
//
//-**************************************************************************
CPipePSFactory::CPipePSFactory() :
m_cRef(1)
{
ComDebOut(( DEB_MARSHAL, "CPipePSFactory - ctor, this:%x \n", this));
}
//+**************************************************************************
// CPipePSFactory()
//
// Description: DTOR
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 12:56:19 PM RichN Created.
//
// Notes:
//
//-**************************************************************************
CPipePSFactory::~CPipePSFactory()
{
ComDebOut(( DEB_MARSHAL, "CPipePSFactory - dtor, this:%x \n"));
}
//+**************************************************************************
// QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
//
// Description: Standard QI
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 11:10:58 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP CPipePSFactory::QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
{
ComDebOut(( DEB_MARSHAL, "CPipePSFactory::QueryInterface, this:%x, riid:%i \n",
this, riid));
if (IsEqualIID(riid, IID_IUnknown) ||
IsEqualIID(riid, IID_IPSFactoryBuffer) )
{
*ppvObj = (IPSFactoryBuffer *) this;
}
else
{
*ppvObj = NULL;
return E_NOINTERFACE;
}
((IUnknown *)*ppvObj)->AddRef();
return S_OK;
}
//+**************************************************************************
// CPipePSFactory::AddRef()
//
// Description: Standard AddRef
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 11:11:16 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP_(ULONG) CPipePSFactory::AddRef()
{
ComDebOut(( DEB_MARSHAL, "CPipePSFactory::AddRef, this:%x, m_cRef:%d \n",
this, m_cRef + 1));
return InterlockedIncrement( &m_cRef );
}
//+**************************************************************************
// CPipePSFactory::Release()
//
// Description: Standard Release
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 11:11:31 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP_(ULONG) CPipePSFactory::Release()
{
ComDebOut(( DEB_MARSHAL, "CPipePSFactory::Release, this:%x, m_cRef:%d \n",
this, m_cRef - 1));
ULONG lRef;
if( (lRef = InterlockedDecrement( &m_cRef )) == 0)
{
delete this;
return 0;
}
return lRef;
}
//+**************************************************************************
// CreateProxy
//
// Description: Creates a pipe proxy.
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 11:11:45 AM RichN Created.
//
// Notes: We will pass back to the call a ptr to our object and we will
// hold a pointer to the real proxy. When the client calls us, we
// can do whatever we want/need to do and then call the real proxy.
//
//-**************************************************************************
STDMETHODIMP CPipePSFactory::CreateProxy( IUnknown *pUnkOuter,
REFIID riid,
IRpcProxyBuffer **ppProxy,
void **ppv )
{
ComDebOut(( DEB_MARSHAL, "CreateProxy, pUnkOuter:%x riid:%I \n",
pUnkOuter, riid));
if( NULL == ppv || NULL == ppProxy )
return E_INVALIDARG;
*ppProxy = NULL;
*ppv = NULL;
IPSFactoryBuffer *pPSFactory;
// Create the real PSFactory for the pipe interface.
HRESULT hr = PrxDllGetClassObject( riid,
IID_IPSFactoryBuffer,
(void **) &pPSFactory);
IUnknown *pNDRPipeProxy = NULL;
IRpcProxyBuffer *pInternalProxyBuffer = NULL;
if( SUCCEEDED(hr) )
{
// Create the real proxy.
hr = pPSFactory->CreateProxy( pUnkOuter,
riid,
&pInternalProxyBuffer,
(void **)&pNDRPipeProxy);
pPSFactory->Release();
}
if( FAILED(hr) )
return hr;
if( IID_IPipeByte == riid )
{
*ppv = new CPipeProxy<BYTE,
&IID_IPipeByte,
&IID_AsyncIPipeByte,
IPipeByte,
AsyncIPipeByte>
(pUnkOuter, pNDRPipeProxy);
}
else if( IID_IPipeLong == riid )
{
*ppv = new CPipeProxy<LONG,
&IID_IPipeLong,
&IID_AsyncIPipeLong,
IPipeLong,
AsyncIPipeLong>
(pUnkOuter, pNDRPipeProxy);
}
else if( IID_IPipeDouble == riid )
{
*ppv = new CPipeProxy<DOUBLE,
&IID_IPipeDouble,
&IID_AsyncIPipeDouble,
IPipeDouble,
AsyncIPipeDouble>
(pUnkOuter, pNDRPipeProxy);
}
else
{
hr = E_NOINTERFACE;
}
if( SUCCEEDED(hr) && NULL == *ppv)
hr = E_OUTOFMEMORY;
// Create the object that contains the IRpcProxyBuffer
// and the pipe interface. Created with refcount of 1.
if( SUCCEEDED(hr) )
{
CPipeProxyImp *pProxyImp = new CPipeProxyImp(pUnkOuter,
pInternalProxyBuffer,
pNDRPipeProxy,
(IUnknown*) *ppv,
riid);
if( NULL == pProxyImp )
{
hr = E_OUTOFMEMORY;
}
else
*ppProxy = (IRpcProxyBuffer *) pProxyImp;
}
// Clean up failure.
if( FAILED(hr) )
{
if( NULL != *ppv )
{
delete *ppv;
*ppv = NULL;
}
pNDRPipeProxy->Release();
pInternalProxyBuffer->Release();
}
return hr;
}
//+**************************************************************************
// CreateStub
//
// Description: Creates a pipe stub.
//
// History:
// Date: Time: Developer: Action:
// 10/30/97 11:12:46 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP CPipePSFactory::CreateStub( REFIID riid,
IUnknown *pUnkServer,
IRpcStubBuffer **ppStub )
{
ComDebOut(( DEB_MARSHAL, "CreateStub, riid:%x pUnkServer:%x \n",
riid, pUnkServer ));
HRESULT hr = S_OK;
IPSFactoryBuffer *pPSFactory;
// Create the real PSFactory for the pipe interface.
hr = PrxDllGetClassObject( riid,
IID_IPSFactoryBuffer,
(void **) &pPSFactory);
// Call real factory to get stub.
if( SUCCEEDED(hr) )
{
hr = pPSFactory->CreateStub(riid, pUnkServer, ppStub);
pPSFactory->Release();
}
return hr;
}
//+**************************************************************************
// CPipePoxyImp(IRpcProxyBuffer *pInternalPB, IUnknown *pPipe)
//
// Description: CTOR
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:31:43 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
CPipeProxyImp::CPipeProxyImp(IUnknown *pUnkOuter,
IRpcProxyBuffer *pInternalPB,
IUnknown *pRealPipeProxy,
IUnknown *pInternalPipeProxy,
IID iid) :
m_cRef (1),
m_pInternalPipeProxy(pInternalPipeProxy),
m_pInternalPB (pInternalPB),
m_pRealPipeProxy (pRealPipeProxy),
m_pUnkOuter (pUnkOuter),
m_IidOfPipe (iid)
{
ComDebOut(( DEB_MARSHAL, "CPipeProxyImp ctor, this:%x \n"));
Win4Assert(NULL != m_pInternalPB);
Win4Assert(NULL != m_pRealPipeProxy);
Win4Assert(NULL != m_pInternalPipeProxy);
}
//+**************************************************************************
// CPipeProxyImp()
//
// Description: DTOR
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:32:02 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
CPipeProxyImp::~CPipeProxyImp()
{
ComDebOut(( DEB_MARSHAL, "~CPipeProxyImp, this:%x \n", this));
// AddRef the outer because we are aggregated.
m_pUnkOuter->AddRef();
// Delete the internal proxy.
if( NULL != m_pInternalPipeProxy )
{
delete m_pInternalPipeProxy;
m_pInternalPipeProxy = NULL;
}
// Release the real proxy.
if( NULL != m_pRealPipeProxy )
{
m_pRealPipeProxy->Release();
m_pRealPipeProxy = NULL;
}
// Release the pointer to the IRpcProxyBuffer
if( NULL != m_pInternalPB )
m_pInternalPB->Release();
}
//+**************************************************************************
// QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
//
// Description: QI
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:36:15 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP CPipeProxyImp::QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
{
ComDebOut(( DEB_MARSHAL, "QueryInterface, this:%x \n", this));
HRESULT hr = S_OK;
if( NULL == ppvObj )
return E_INVALIDARG;
if (IsEqualIID(riid, IID_IUnknown) ||
IsEqualIID(riid, IID_IRpcProxyBuffer) )
{
*ppvObj = (IUnknown *) this;
}
else if( IsEqualIID(riid, m_IidOfPipe) )
{
*ppvObj = m_pInternalPipeProxy;
}
else
{
return m_pInternalPB->QueryInterface(riid, ppvObj);
}
((IUnknown *)(*ppvObj))->AddRef();
return hr;
}
//+**************************************************************************
// CPipeProxyImp::AddRef()
//
// Description: AddRef
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:36:34 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP_(ULONG) CPipeProxyImp::AddRef()
{
return InterlockedIncrement( &m_cRef );
}
//+**************************************************************************
// CPipeProxyImp::Release()
//
// Description:
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:36:48 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP_(ULONG) CPipeProxyImp::Release()
{
ULONG lRef;
if( (lRef = InterlockedDecrement( &m_cRef )) == 0)
{
delete this;
return 0;
}
return lRef;
}
//+**************************************************************************
// Connect(IRpcChannelBuffer *pRpcChannelBuffer)
//
// Description: Simple pass through.
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:36:59 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP CPipeProxyImp::Connect(IRpcChannelBuffer *pRpcChannelBuffer)
{
return m_pInternalPB->Connect(pRpcChannelBuffer);
}
//+**************************************************************************
// Disconnect( void )
//
// Description: Simple pass through.
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:37:25 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP_(void) CPipeProxyImp::Disconnect( void )
{
m_pInternalPB->Disconnect();
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// CPipeProxy( void * pProxy ):
//
// Description:CTOR
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:43:00 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
CPipeProxy<T, ID, AsyncID, I, AsyncI>
::CPipeProxy(IUnknown *pUnkOuter, void * pProxy ):
m_cFreeSpace (0), m_cKeepBufferSize (0),
m_cKeepDataSize (0), m_cLastRead (0),
m_cPushBufferSize (0), m_cReadAhead (0),
m_cRef (0), m_pAsyncPullPipe (NULL),
m_pAsyncPushPipe (NULL), m_pFreeSpace (NULL),
m_pISyncPull (NULL), m_pISyncPush (NULL),
m_pKeepBuffer (NULL), m_pKeepData (NULL),
m_pRealProxy ((I *)pProxy), m_pUnkOuter (pUnkOuter),
m_pPushBuffer (NULL), m_PullState (PULLSTATE0_ENTRY),
m_PushState (PUSHSTATE0_ENTRY)
{
ComDebOut(( DEB_MARSHAL, "CPipeProxy, pUnkOuter:%x pProxy:%x p \n",
pUnkOuter, pProxy));
Win4Assert(NULL != m_pUnkOuter);
Win4Assert(NULL != m_pRealProxy);
// Fill in the array of functions for the pull states.
PullStateFunc[0] = NULL; // Should never execute in state zero.
PullStateFunc[1] = NbNaRgtRA1; // No Buffer, No async outstanding, Request > Read ahead
PullStateFunc[2] = NbaRltRA2; // No Buffer, async call outstanding, Request < Read Ahead
PullStateFunc[3] = NbaRgtRA3; // No Buffer, async, Req >= Read ahead
PullStateFunc[4] = baRltB4; // Buffer, async, Request < Buffer size
PullStateFunc[5] = baRgtB5; // Buffer, async, Request >= Buffer size
PullStateFunc[6] = PullDone6; // done.
// Fill in the array of functions for the push states.
PushStateFunc[0] = NULL; // Should never execute in state zero.
PushStateFunc[1] = NbNf1; // No Buffer, No free buffer space
PushStateFunc[2] = bfPgtF2; // Buffer, free space in buffer, push size >= free size
PushStateFunc[3] = bfPltF3; // Buffer, free, push < free
PushStateFunc[4] = bPSz4; // Buffer, push size zero
PushStateFunc[5] = PushDone5; // Done
#if DBG==1
for(int i = 1; i < MAX_PULL_STATES; i++)
Win4Assert(PullStateFunc[i] != NULL);
for(i = 1; i < MAX_PUSH_STATES; i++)
Win4Assert(PushStateFunc[i] != NULL);
#endif
return;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// CPipeProxy( void )
//
// Description: DTOR
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:43:20 AM RichN Created.
//
// Notes: Addref the outer unknown and then release the pointer to the
// real proxy.
//
//-**************************************************************************
CPipeProxy<T, ID, AsyncID, I, AsyncI>
::~CPipeProxy( void )
{
Win4Assert(NULL != m_pUnkOuter);
Win4Assert(NULL != m_pRealProxy);
}
//+**************************************************************************
// QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
//
// Description: IUnknown implementation. All delegate to outer unknown.
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:48:42 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
STDMETHODIMP CPipeProxy<T, ID, AsyncID, I, AsyncI>
::QueryInterface(REFIID riid, LPVOID FAR* ppvObj)
{
return m_pUnkOuter->QueryInterface(riid, ppvObj);
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
STDMETHODIMP_(ULONG) CPipeProxy<T, ID, AsyncID, I, AsyncI>
::AddRef()
{
m_cRef++;
return m_pUnkOuter->AddRef();
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
STDMETHODIMP_(ULONG) CPipeProxy<T, ID, AsyncID, I, AsyncI>
::Release()
{
m_cRef--;
return m_pUnkOuter->Release();
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// Pull( T *Buf, ULONG Request, ULONG *Received)
//
// Description: Pull the data from the server.
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:49:18 AM RichN Created.
//
// Notes: We read data ahead by using async calls. The size of the
// read ahead can be controled by the user by implementing
// the IPipeHueristic and setting it on the interface.
//
//-**************************************************************************
STDMETHODIMP CPipeProxy<T, ID, AsyncID, I, AsyncI>
::Pull( T *Buf, ULONG Request, ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "Pull, this:%x, Buf:%x, Request:%d, Received:%x \n",
this, Buf, Request, Received));
if( 0 == Request )
return E_UNEXPECTED;
HRESULT hr;
// For debugging it is sometimes useful to disable
// the async read ahead.
#if DISABLEASYNC==1
hr = m_pRealProxy->Pull(Buf, Request, Received);
return hr;
#endif
*Received = 0;
// Transition to the next state.
hr = PullStateTransition( Request );
// Should never see state 0.
Win4Assert(0 != m_PullState);
// Call the function for the new state.
if( SUCCEEDED(hr) )
{
hr = (this->*(PullStateFunc[m_PullState]))( Buf, Request, Received );
}
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// Push( T *Buf, ULONG count)
//
// Description: Pushes data to the server.
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 11:49:39 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
STDMETHODIMP CPipeProxy<T, ID, AsyncID, I, AsyncI>
::Push( T *Buf, ULONG Count)
{
ComDebOut(( DEB_MARSHAL, "Push, this:%x, Buf:%x, Count:%u \n",
this, Buf, Count));
HRESULT hr;
// For debugging it is sometimes useful to disable
// write behind.
#if DISABLEASYNC==1
hr = m_pRealProxy->Push(Buf, Count);
return hr;
#endif
// Transition to the next state.
hr = PushStateTransition( Count );
// Should never see state 0.
Win4Assert(0 != m_PushState);
// Call the function for the new state.
if( SUCCEEDED(hr) )
{
hr = (this->*(PushStateFunc[m_PushState]))( Buf, Count );
}
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// InitAsync(void)
//
// Description: Initializes, gets, the pointers to the async parts.
//
// History:
// Date: Time: Developer: Action:
// 12/8/97 4:45:22 PM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::InitAsync(IUnknown** ppCallObj,
AsyncI** ppAsyncPipe,
ISynchronize** ppISync)
{
ComDebOut(( DEB_MARSHAL, "InitAsync, this:%x \n", this));
Win4Assert(NULL != m_pRealProxy);
Win4Assert(NULL == (*ppAsyncPipe));
Win4Assert(NULL == (*ppISync));
HRESULT hr;
ICallFactory *pCF = NULL;
hr = m_pRealProxy->QueryInterface(IID_ICallFactory, (void **) &pCF);
if( FAILED(hr) )
return hr;
hr = pCF->CreateCall(*AsyncID, NULL, IID_IUnknown, ppCallObj);
pCF->Release();
if( FAILED(hr) )
return hr;
hr = (*ppCallObj)->QueryInterface(*AsyncID, (void **) ppAsyncPipe);
if( FAILED(hr) )
goto ErrorCallObj;
hr = (*ppCallObj)->QueryInterface(IID_ISynchronize, (void **) ppISync);
if( FAILED(hr) )
goto ErrorAsyncPipe;
return S_OK;
ErrorAsyncPipe:
(*ppAsyncPipe)->Release();
(*ppAsyncPipe) = NULL;
ErrorCallObj:
(*ppCallObj)->Release();
(*ppCallObj) = NULL;
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// CleanupProxy(IUnknown* pCallObj, IUnknown* pAsyncPipe, ISynchronize* pISync)
//
// Description: Cleans up all the async interfaces acquired.
//
// History:
// Date: Time: Developer: Action:
// 12/16/97 11:42:42 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
void CPipeProxy<T, ID, AsyncID, I, AsyncI>
::CleanupProxy(T ** ppBuffer,
IUnknown** ppCallObj,
AsyncI** ppAsyncPipe,
ISynchronize** ppISync)
{
if( *ppBuffer )
{
delete (*ppBuffer);
(*ppBuffer) = NULL;
}
if( NULL != (*ppISync) )
{
(*ppISync)->Release();
*ppISync = NULL;
}
if( NULL != (*ppAsyncPipe) )
{
(*ppAsyncPipe)->Release();
*ppAsyncPipe = NULL;
}
if( NULL != (*ppCallObj) )
{
(*ppCallObj)->Release();
*ppCallObj = NULL;
}
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// CancelTheCall(DWORD delay)
//
// Description: Cancel the currently outstanding call.
//
// History:
// Date: Time: Developer: Action:
// 12/9/97 10:59:54 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
void CPipeProxy<T, ID, AsyncID, I, AsyncI>
::CancelTheCall(IUnknown *pCallObj, DWORD delay)
{
ComDebOut(( DEB_MARSHAL, "CancelTheCall, this:%x \n", this));
ICancelMethodCalls *pICancel;
HRESULT hr = pCallObj->QueryInterface(IID_ICancelMethodCalls,
(void **) &pICancel);
if( FAILED(hr) )
return;
pICancel->Cancel(delay);
pICancel->Release();
return;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// SetReadAhead(ULONG Request)
//
// Description: Determine the size of the read ahead.
//
// History:
// Date: Time: Developer: Action:
// 11/20/97 10:01:10 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
void CPipeProxy<T, ID, AsyncID, I, AsyncI>
::SetReadAhead(ULONG Request)
{
ComDebOut(( DEB_MARSHAL, "SetReadAhead, this:%x Request:%u \n", this, Request));
Win4Assert(Request != 0);
switch(m_PullState)
{
case PULLSTATE1_FIRST_CALL :
// On the first call just set the read ahead to the request.
// This assumes that the request will be constant and
// we will be one call ahead all the time.
m_cReadAhead = Request;
break;
case PULLSTATE2_NS_RQlsRA :
// We haven't had a zero read or we wouldn't be here
Win4Assert(m_cLastRead != 0);
// Set the read ahead to the lesser of the request and the
// amount last read. We are trying to match the read ahead with
// the request by assuming a constant request, but the server
// may only return a given amount regardless of what we
// request.
m_cReadAhead = MIN(Request, m_cLastRead);
break;
case PULLSTATE3_NS_RQgeRA :
case PULLSTATE5_S_RQgeBS :
// No zero read
Win4Assert(m_cLastRead != 0);
// The request is greater than what was asked for last time. So
// we increase the read ahead to the max of the request or what
// was actually read last time.
m_cReadAhead = MAX(Request, m_cLastRead);
break;
default :
// For all other states we should not be making read ahead calls.
// Mostly because we read zero elements last time which indicates
// the end of the data. The PULLSTATE4_S_RQlsBS doesn't do a read
// ahead so we shouldn't be here while in that state.
Win4Assert(FALSE && "Request read ahead in wrong state.");
}
Win4Assert( 0 != m_cReadAhead );
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// CheckAndSetKeepBuffer()
//
// Description: Check to see if the Buffer is the correct size and if not
// make it the correct size.
//
// History:
// Date: Time: Developer: Action:
// 11/19/97 3:27:25 PM RichN Created.
//
// Notes: The buffer will never get smaller.
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::CheckAndSetKeepBuffer(void)
{
Win4Assert(0 != m_cReadAhead);
// Create a keep buffer the size of the read ahead.
// We assume here that the user will not change the
// request size and that the amount of data on hand
// will never be larger. When it is we will re-allocate.
// The buffer will never get smaller. Something to look at.
if( m_cKeepBufferSize >= m_cReadAhead )
return S_OK;
T *temp = new T[m_cReadAhead];
if( NULL == temp )
{
delete[] m_pKeepBuffer;
m_pKeepBuffer = NULL;
return E_OUTOFMEMORY;
}
if( m_pKeepBuffer != NULL )
{
// Copy the data into the new buffer
memcpy(temp, m_pKeepBuffer, m_cKeepDataSize * sizeof(T));
// Delete the old buffer and reset the bookkeeping.
delete[] m_pKeepBuffer;
}
m_pKeepBuffer = temp;
m_cKeepBufferSize = m_cReadAhead;
m_pKeepData = m_pKeepBuffer + m_cKeepDataSize;
return S_OK;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// PullStateTransition(ULONG Request)
//
// Description: Transition from one state to the next. See pipes document
// for a description of the state machine.
//
// History:
// Date: Time: Developer: Action:
// 11/11/97 4:20:19 PM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::PullStateTransition(ULONG Request)
{
ComDebOut(( DEB_MARSHAL, "PullStateTransition \n"));
Win4Assert(Request > 0);
switch( m_PullState )
{
case PULLSTATE0_ENTRY:
// Transition to the first call state.
m_PullState = PULLSTATE1_FIRST_CALL;
break;
case PULLSTATE1_FIRST_CALL:
// If the last read was zero we are done. We have no stored
// data so go to the state that handles request
// that are either less than or greater than or
// equal to the last read ahead. Realize the amount of
// data returned from the last read ahead may
// not equal the requeat and could be greater than or less than.
if( 0 == m_cLastRead )
m_PullState = PULLSTATE6_DONE;
else
m_PullState = (Request < m_cReadAhead) ?
PULLSTATE2_NS_RQlsRA : PULLSTATE3_NS_RQgeRA;
break;
case PULLSTATE2_NS_RQlsRA:
// In this state, either the last read is not zero or
// the amount of data remaining is zero. Whatever state
// we get to this one from does not go here on a last read
// of zero nor if there is any data in the buffer (kept data).
// Possible we where in this state and got a last
// read of zero, in which case the amount of held data will
// be = zero. We are in a state with no held data and got
// a zero read of data, there can't be any kept data.
Win4Assert(!( m_cLastRead == 0 && m_cKeepDataSize > 0));
// If we had a zero read, we cleaned up in
// this state and go to the state that just returns zero.
if( 0 == m_cLastRead)
{
Win4Assert( 0 == m_cKeepDataSize );
m_PullState = PULLSTATE6_DONE;
}
else
{
// If the kept data is zero, then we need to go to a state
// that understands an empty buffer. Determine which one
// by the request and the read ahead.
// If there is kept data then go to a state that understands
// a non empty buffer. This time the correct one depends on
// the amount of data in the buffer.
if( 0 == m_cKeepDataSize )
m_PullState = ( Request < m_cReadAhead ) ?
PULLSTATE2_NS_RQlsRA : PULLSTATE3_NS_RQgeRA;
else
m_PullState = ( Request < m_cKeepDataSize) ?
PULLSTATE4_S_RQlsBS : PULLSTATE5_S_RQgeBS;
}
break;
case PULLSTATE3_NS_RQgeRA:
// We can never leave this state with data in the buffer. The
// request is greater than the read ahead and the returned amount
// of data can never be greater than the requested data, but it
// could be less.
Win4Assert(m_cKeepDataSize == 0);
// If the last read was zero go to the done state.
// else go to the state that understands empty buffers depending
// on the request and the read ahead.
if( 0 == m_cLastRead )
m_PullState = PULLSTATE6_DONE;
else
m_PullState = ( Request < m_cReadAhead ) ?
PULLSTATE2_NS_RQlsRA : PULLSTATE3_NS_RQgeRA;
break;
case PULLSTATE4_S_RQlsBS:
// When this state was entered there was data in the buffer. The
// request was for less than the buffer size so when we returned
// there should have still been data in the buffer. No read is done.
Win4Assert(m_cKeepDataSize > 0);
Win4Assert(m_cLastRead > 0);
// Go to the state that handles data in the buffer
// depending on the request and the amount of data in the buffer.
m_PullState = ( Request < m_cKeepDataSize ) ?
PULLSTATE4_S_RQlsBS : PULLSTATE5_S_RQgeBS;
break;
case PULLSTATE5_S_RQgeBS:
// Because we can fulfill at aleast part of the request from
// the buffer, we don't wait on the async call to finish. If it
// did finish (wait 0 tells us that) then there is data in the
// buffer (assuming it returned data) otherwise it is empty.
// So when the call finished last time the buffer
// could be empty or not. If the read was zero the buffer is empty.
Win4Assert( (m_cLastRead == 0 && m_cKeepDataSize == 0) ||
m_cLastRead != 0 );
// If the buffer is empty then on a zero last read go to done.
// Otherwise go to a state that understands empty buffers depending
// on the request size and the read ahead.
if( 0 == m_cKeepDataSize )
if( 0 == m_cLastRead )
m_PullState = PULLSTATE6_DONE;
else
m_PullState = ( Request < m_cReadAhead ) ?
PULLSTATE2_NS_RQlsRA : PULLSTATE3_NS_RQgeRA;
else
// Otherwise go to one that understands having data in the
// buffer depending on the request and the amount of data in
// the buffer.
m_PullState = ( Request < m_cKeepDataSize ) ?
PULLSTATE4_S_RQlsBS : PULLSTATE5_S_RQgeBS;
break;
case PULLSTATE6_DONE:
// When in this state there better not be any data left and
// the last read must be zero.
Win4Assert(m_cKeepDataSize == 0);
Win4Assert(m_cLastRead == 0);
m_PullState = PULLSTATE1_FIRST_CALL;
break;
default:
return E_UNEXPECTED;
}
return S_OK;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// NbNaRgtRA1
//
// Description: Pull, No data in buffer, no async call outstanding and
// request is >= read ahead. State 1.
//
// History:
// Date: Time: Developer: Action:
// 11/19/97 2:30:11 PM RichN Created.
//
// Notes: Make a sync call to get some data and then make an async call to
// read ahead.
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::NbNaRgtRA1(T *Buf,
ULONG Request,
ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "NbNaRgtRA1 Request:%d\n", Request));
Win4Assert(1 == m_PullState);
Win4Assert(NULL != Buf);
// State conditions
Win4Assert(0 == m_cLastRead );
Win4Assert(0 == m_cKeepDataSize);
HRESULT hr = S_OK;
// We are only in this state one time. There will always be
// an out standing async call, so we init async here.
hr = InitAsync(&m_pPullCallObj, &m_pAsyncPullPipe, &m_pISyncPull);
if( FAILED(hr) )
return hr;
// make a sync call to get started.
hr = m_pRealProxy->Pull(Buf, Request, Received);
m_cLastRead = *Received;
if( m_cLastRead > 0 && SUCCEEDED(hr))
{
SetReadAhead(Request);
// Make the async call.
hr = m_pAsyncPullPipe->Begin_Pull(m_cReadAhead);
}
else
{
CleanupProxy(&m_pKeepBuffer,
&m_pPullCallObj,
&m_pAsyncPullPipe,
&m_pISyncPull);
}
// Post condition
// Wouldn't expect the last read to be zero here, but no reason
// it couldn't be.
Win4Assert( 0 == m_cKeepDataSize );
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// NbaRltRA2 (T *Buf, ULONG Request, ULONG *Received)
//
// Description: Pull, No Buffer, read ahead call outstanding.
// State 2. We have to be prepared
// for the amount returned to be greater than, less than or equal
// to the amount requested. This works for both states 2 and 3.
//
// History:
// Date: Time: Developer: Action:
// 11/19/97 2:32:13 PM RichN Created.
//
// Notes: wait on the sync object,
// Finish the async call,
// Copy the data into the user Buffer
// make another async call
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::NbaRltRA2 (T *Buf, ULONG Request, ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "NbaRltRA2 Request:%d\n", Request));
Win4Assert(2 == m_PullState);
Win4Assert(NULL != Buf);
// State conditions.
Win4Assert(0 == m_cKeepDataSize); // No data in keep Buffer.
Win4Assert(0 < m_cLastRead);
Win4Assert(Request < m_cReadAhead);
HRESULT hr = S_OK;
bool bDoCleanup = false;
// There might be a Bug here. Bug!
// We should just be able to call the finish method, but the
// bug requires us to wait first.
hr = m_pISyncPull->Wait(0, WAIT_INFINITE);
if( SUCCEEDED(hr) )
{
hr = CheckAndSetKeepBuffer();
if( SUCCEEDED(hr) )
{
// Get the data requested last time. Remember the amount returned
// could be less than we requested.
hr = m_pAsyncPullPipe->Finish_Pull(m_pKeepBuffer, &m_cLastRead);
// We can't return more than requested, the buffer may not be
// large enough.
*Received = MIN(Request, m_cLastRead);
if( SUCCEEDED(hr) && m_cLastRead > 0 )
{
// Copy the data to the users Buffer and updata bookkeeping.
memcpy(Buf, m_pKeepBuffer, (*Received) * sizeof(T));
m_pKeepData = m_pKeepBuffer + *Received;
m_cKeepDataSize = m_cLastRead - *Received;
SetReadAhead(Request);
// Make another read ahead
hr = m_pAsyncPullPipe->Begin_Pull(m_cReadAhead);
}
else
{
// If the call failed or we received no data, we
// need to clean up since we won't be called again.
bDoCleanup = true;
}
}
else
{
//Cancel the call here.
CancelTheCall(m_pPullCallObj, 0);
}
}
if( FAILED(hr) || bDoCleanup )
CleanupProxy(&m_pKeepBuffer,
&m_pPullCallObj,
&m_pAsyncPullPipe,
&m_pISyncPull);
// Post condition
Win4Assert(!(m_cLastRead == 0 && m_cKeepDataSize > 0));
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// NbaRgtRA3 (T *Buf, ULONG Request, ULONG *Received)
//
// Description: Pull, No Buffered data, async call outstanding and request
// is greater than read ahead.
//
// History:
// Date: Time: Developer: Action:
// 11/20/97 10:50:22 AM RichN Created.
//
// Notes: Difference between this and the previous state: we know
// we don't need a keep Buffer here.
// wait on the sync object,
// Finish the async call,
// Copy the data into the user Buffer
// make another async call
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::NbaRgtRA3 (T *Buf, ULONG Request, ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "NbaRgtRA3, Request:%d \n", Request));
Win4Assert(3 == m_PullState);
Win4Assert(NULL != Buf);
// State conditions.
Win4Assert(0 == m_cKeepDataSize); // No data in keep Buffer.
Win4Assert(0 < m_cLastRead);
Win4Assert(Request >= m_cReadAhead);
bool bDoCleanup = false;
HRESULT hr = m_pISyncPull->Wait(0, WAIT_INFINITE);
if( SUCCEEDED(hr))
{
// Get the data requested last time.
hr = m_pAsyncPullPipe->Finish_Pull(Buf, &m_cLastRead);
*Received = m_cLastRead;
if( SUCCEEDED(hr) && m_cLastRead > 0 )
{
// Reset the amount of data remaining.
m_cKeepDataSize = 0;
m_pKeepData = NULL;
SetReadAhead(Request);
// Make another read ahead
hr = m_pAsyncPullPipe->Begin_Pull(m_cReadAhead);
}
else
{
// If the call failed or we received no data, we
// need to clean up since we won't be called again.
bDoCleanup = TRUE;
}
}
if( FAILED(hr) || bDoCleanup )
CleanupProxy(&m_pKeepBuffer,
&m_pPullCallObj,
&m_pAsyncPullPipe,
&m_pISyncPull);
// Post condition
Win4Assert( 0 < m_cReadAhead);
Win4Assert(0 == m_cKeepDataSize);
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// baRltB4 (T *Buf, ULONG Request, ULONG *Received)
//
// Description: Pull, Data in Buffer, async call outstanding and Request is
// less than the data in the keep Buffer.
//
// History:
// Date: Time: Developer: Action:
// 11/20/97 1:22:41 PM RichN Created.
//
// Notes: Copy data from the keep Buffer to the users Buffer.
// Update state variables.
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::baRltB4 (T *Buf, ULONG Request, ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "baRltB4 \n"));
Win4Assert(NULL != Buf);
Win4Assert(4 == m_PullState);
// State conditions.
Win4Assert(0 < m_cKeepDataSize); // Data in keep Buffer.
Win4Assert(0 < m_cLastRead);
Win4Assert(Request < m_cKeepDataSize);
memcpy(Buf, m_pKeepData, Request * (sizeof(T)));
m_cKeepDataSize -= Request;
m_pKeepData += Request;
// Post condition
Win4Assert(m_cKeepDataSize > 0);
Win4Assert(m_cLastRead > 0);
// Post condition
Win4Assert(0 < m_cKeepDataSize); // Data in keep Buffer.
Win4Assert(0 < m_cLastRead);
return S_OK;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// baRgtB5 (T *Buf, ULONG Request, ULONG *Received)
//
// Description: Pull, Data in Buffer, async call outstanding and Request is
// greater than or equal the data in the keep Buffer.
//
// History:
// Date: Time: Developer: Action:
// 11/20/97 1:24:25 PM RichN Created.
//
// Notes: Copy the keep Buffer data to the users Buffer.
// Wait 0
// if the call has completed.
// Finish the async call (keep Buffer, RA)
// Copy data into users Buffer to fill request
// Update keep data size.
// if we didn't read zero
// set read ahead
// Begin async call(RA)
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::baRgtB5 (T *Buf, ULONG Request, ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "baRgtB5 \n"));
Win4Assert(NULL != Buf);
Win4Assert(5 == m_PullState);
// State conditions.
Win4Assert(0 < m_cKeepDataSize); // Data in keep buffer.
Win4Assert(0 < m_cLastRead);
Win4Assert(Request >= m_cKeepDataSize);
HRESULT hr = S_OK;
// Give whatever data we already have.
T *tempBuf = Buf;
memcpy(tempBuf, m_pKeepData, m_cKeepDataSize * (sizeof(T)) );
// Remainder of the request.
ULONG Remainder = Request - m_cKeepDataSize;
tempBuf += m_cKeepDataSize;
m_cKeepDataSize = 0;
m_pKeepData = NULL;
hr = m_pISyncPull->Wait(0, 0);
// If the call is finished get the data and
// copy up to the total request or as much as
// we have into the buffer.
if( SUCCEEDED(hr) && RPC_S_CALLPENDING != hr)
{
hr = CheckAndSetKeepBuffer();
if( SUCCEEDED(hr) )
{
hr = m_pAsyncPullPipe->Finish_Pull(m_pKeepBuffer, &m_cLastRead);
if( SUCCEEDED(hr) )
{
// Copy the smaller of the remainder of the
// request or what was actually received.
ULONG CopySize = MIN(Remainder, m_cLastRead);
memcpy(tempBuf, m_pKeepBuffer, CopySize * sizeof(T));
m_cKeepDataSize = m_cLastRead - CopySize;
m_pKeepData = m_pKeepBuffer + CopySize;
if( m_cLastRead > 0 )
{
SetReadAhead(Request);
hr = m_pAsyncPullPipe->Begin_Pull(m_cReadAhead);
}
}
}
else
CancelTheCall(m_pPullCallObj, 0);
}
else
{
if( RPC_S_CALLPENDING == hr )
hr = S_OK;
else
CancelTheCall(m_pPullCallObj, 0);
}
*Received = (ULONG) (tempBuf - Buf);
if( FAILED(hr) )
CleanupProxy(&m_pKeepBuffer,
&m_pPullCallObj,
&m_pAsyncPullPipe,
&m_pISyncPull);
// Post condition
Win4Assert( (m_cLastRead == 0 && m_cKeepDataSize == 0) ||
m_cLastRead != 0 );
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// PullDone6 (T *Buf, ULONG Request, ULONG *Received)
//
// Description: Pull Done, no data in the Buffer and no outstanding calls.
//
// History:
// Date: Time: Developer: Action:
// 11/20/97 3:30:14 PM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::PullDone6 (T *Buf, ULONG Request, ULONG *Received)
{
ComDebOut(( DEB_MARSHAL, "PullDone6 \n"));
Win4Assert(6 == m_PullState);
CleanupProxy(&m_pKeepBuffer,
&m_pPullCallObj,
&m_pAsyncPullPipe,
&m_pISyncPull);
HRESULT hr = S_OK;
if (Request > 0)
{
m_PullState = PULLSTATE1_FIRST_CALL;
hr = (this->*(PullStateFunc[m_PullState]))( Buf, Request, Received );
}
else
{
*Received = 0;
// Post condition
Win4Assert( 0 == m_cKeepDataSize );
Win4Assert( 0 == m_cLastRead );
}
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// SetPushBuffer(ULONG PushSize)
//
// Description: Allocates a buffer for push, or reallocates if it
// needs to grow.
//
// History:
// Date: Time: Developer: Action:
// 11/21/97 10:10:50 AM RichN Created.
//
// Notes: The buffer will never get smaller. We might want
// to reduce it by some algorithm, but not this time.
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::SetPushBuffer(ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "SetPushBuffer, PushSize:%l \n"));
// If it is already big enough just return.
if( m_cPushBufferSize >= PushSize )
return S_OK;
ULONG NewSize = MAX(PushSize, (FRAGMENT_SIZE / sizeof(T)) + 1);
T *pTtemp = new T[NewSize];
if( NULL == pTtemp )
{
delete[] m_pPushBuffer;
m_pPushBuffer = NULL;
return E_OUTOFMEMORY;
}
ULONG BufferedDataSize = m_cPushBufferSize - m_cFreeSpace;
if( m_pPushBuffer != NULL )
{
// Copy data over and reset bookkeeping.
memcpy(pTtemp, m_pPushBuffer, BufferedDataSize * sizeof(T));
delete[] m_pPushBuffer;
}
m_pPushBuffer = pTtemp;
m_cPushBufferSize = NewSize;
m_pFreeSpace = m_pPushBuffer + BufferedDataSize;
m_cFreeSpace = m_cPushBufferSize - BufferedDataSize;
return S_OK;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// PushStateTransition(ULONG Request)
//
// Description: Implements the transition table for push. See the pipes
// document for a description of the state machine.
//
// History:
// Date: Time: Developer: Action:
// 11/13/97 4:36:43 PM RichN Created.
//
// 02/05/99 JohnStra Modified Push state machine to
// allow multiple Push operations
// on a pipe.
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::PushStateTransition(ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "PushStateTransition, PushSize:%l \n"));
switch( m_PushState )
{
case PUSHSTATE0_ENTRY:
// From the entry state we always go to the first call state.
m_PushState = PUSHSTATE1_FIRSTCALL;
break;
case PUSHSTATE1_FIRSTCALL:
case PUSHSTATE2_FS_PSgeFS:
case PUSHSTATE3_FS_PSltFS:
// If the push size is zero transition to state that
// does a zero send.
if( 0 == PushSize )
m_PushState = PUSHSTATE4_FS_PSZERO;
else
// Go to state that either puts the data in free space
// or one that handles a push greater than the free space.
m_PushState = (PushSize < m_cFreeSpace) ?
PUSHSTATE3_FS_PSltFS : PUSHSTATE2_FS_PSgeFS;
break;
case PUSHSTATE4_FS_PSZERO:
// If we are in the state that handles a zero push we may
// be called again with a positive buffer size to execute
// another push. If we are called with any other
// buffer size, go to the state that returns an error.
if( 0 < PushSize )
m_PushState = PUSHSTATE1_FIRSTCALL;
else
m_PushState = PUSHSTATE5_DONE_ERROR;
break;
case PUSHSTATE5_DONE_ERROR:
// Stay in state PUSHSTATE_DONE_ERROR.
break;
default:
return E_UNEXPECTED;
}
return S_OK;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// NbNf1(T *Buf, ULONG PushSize)
//
// Description: Push, No buffer, no free, state 1.
//
// History:
// Date: Time: Developer: Action:
// 11/21/97 9:42:57 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::NbNf1(T *Buf, ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "NbNf1, PushSize:%l \n", PushSize));
Win4Assert(1 == m_PushState);
// This is the first call to push so PushSize shouldn't be zero.
if( PushSize == 0 || NULL == Buf)
return E_INVALIDARG;
HRESULT hr;
// We are only in this state one time so init the async stuff.
hr = InitAsync(&m_pPushCallObj, &m_pAsyncPushPipe, &m_pISyncPush);
if( FAILED(hr) )
return hr;
hr = m_pAsyncPushPipe->Begin_Push(Buf, PushSize);
if( FAILED(hr) )
CleanupProxy(&m_pPushBuffer,
&m_pPushCallObj,
&m_pAsyncPushPipe,
&m_pISyncPush);
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// bfPgtF2(T *Buf, ULONG PushSize)
//
// Description: Push, Have a buffer with free space and the push size
// is greater than or equal to the free space. State 2.
//
// History:
// Date: Time: Developer: Action:
// 11/21/97 10:52:41 AM RichN Created.
//
// Notes: This may grow the buffer, look at reducing it in the next method.
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::bfPgtF2(T *Buf, ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "bfPgtF2, PushSize:%l \n", PushSize));
Win4Assert(2 == m_PushState);
Win4Assert( PushSize >= m_cFreeSpace );
Win4Assert( PushSize > 0 );
Win4Assert( (LONG) m_cFreeSpace >= 0 );
if( PushSize == 0 || NULL == Buf)
return E_INVALIDARG;
// There might be a BUG here. BUG! Shouldn't have to wait.
HRESULT hr = m_pISyncPush->Wait(0, WAIT_INFINITE);
if( SUCCEEDED(hr) )
hr = m_pAsyncPushPipe->Finish_Push();
if( SUCCEEDED(hr) )
{
ULONG TotalData = PushSize + (m_cPushBufferSize - m_cFreeSpace);
hr = SetPushBuffer( TotalData );
if( SUCCEEDED(hr) )
{
// Append the data to the buffer.
memcpy(m_pFreeSpace, Buf, PushSize * sizeof(T));
hr = m_pAsyncPushPipe->Begin_Push(m_pPushBuffer, TotalData);
m_pFreeSpace = m_pPushBuffer;
m_cFreeSpace = m_cPushBufferSize;
}
}
if( FAILED(hr) )
CleanupProxy(&m_pPushBuffer,
&m_pPushCallObj,
&m_pAsyncPushPipe,
&m_pISyncPush);
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// bfPltF3(T *Buf, ULONG PushSize)
//
// Description: Push, Have buffer and pushed data is less than free space.
//
// History:
// Date: Time: Developer: Action:
// 11/21/97 11:03:19 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::bfPltF3(T *Buf, ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "bfPltF3, PushSize:%l \n", PushSize));
Win4Assert(3 == m_PushState);
Win4Assert( m_cFreeSpace > PushSize );
Win4Assert( PushSize > 0 );
Win4Assert( m_cFreeSpace > 0 );
if( NULL == Buf)
return E_INVALIDARG;
// Copy the data into the buffer.
memcpy(m_pFreeSpace, Buf, PushSize * sizeof(T));
m_cFreeSpace -= PushSize;
m_pFreeSpace += PushSize;
return S_OK;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// bPSz4(T *Buf, ULONG PushSize)
//
// Description: Push, Have buffer and push size is zero.
//
// History:
// Date: Time: Developer: Action:
// 11/21/97 11:33:32 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::bPSz4(T *Buf, ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "bPSz4, PushSize:$l \n", PushSize));
Win4Assert(4 == m_PushState);
Win4Assert( PushSize == 0 );
Win4Assert( (LONG) m_cFreeSpace >= 0 );
// There might be a BUG here. BUG! Shouldn't have to wait.
HRESULT hr = m_pISyncPush->Wait(0, WAIT_INFINITE);
if( SUCCEEDED(hr) )
hr = m_pAsyncPushPipe->Finish_Push();
if( SUCCEEDED(hr) )
{
if( (m_cPushBufferSize - m_cFreeSpace) > 0 )
{
// Data in buffer so send it.
hr = m_pAsyncPushPipe->Begin_Push(m_pPushBuffer,
m_cPushBufferSize - m_cFreeSpace);
if( FAILED(hr) )
goto asyncFailed;
hr = m_pISyncPush->Wait(0, WAIT_INFINITE);
if( FAILED(hr) )
goto asyncFailed;
hr = m_pAsyncPushPipe->Finish_Push();
}
if( SUCCEEDED(hr) )
{
// Push a zero size buffer to signal end of data.
hr = m_pAsyncPushPipe->Begin_Push(Buf, PushSize);
if( FAILED(hr) )
goto asyncFailed;
hr = m_pISyncPush->Wait(0, WAIT_INFINITE);
if( FAILED(hr) )
goto asyncFailed;
hr = m_pAsyncPushPipe->Finish_Push();
}
}
asyncFailed:
// Last call regardless of success or failure so clean async up.
CleanupProxy(&m_pPushBuffer,
&m_pPushCallObj,
&m_pAsyncPushPipe,
&m_pISyncPush);
return hr;
}
template<class T, const IID* ID, const IID *AsyncID, class I, class AsyncI>
//+**************************************************************************
// PushDone5(T *Buf, ULONG PushSize)
//
// Description: Push Done, so this should never be called.
//
// History:
// Date: Time: Developer: Action:
// 11/21/97 11:42:08 AM RichN Created.
//
// Notes:
//
//-**************************************************************************
HRESULT CPipeProxy<T, ID, AsyncID, I, AsyncI>
::PushDone5(T *Buf, ULONG PushSize)
{
ComDebOut(( DEB_MARSHAL, "PushDone5, PushSize:%u \n"));
Win4Assert(FALSE && "Push call after completion.");
Win4Assert(5 == m_PushState);
return E_UNEXPECTED;
}