source-engine/utils/vmpi/loopback_channel.cpp
FluorescentCIAAfricanAmerican 3bf9df6b27 1
2020-04-22 12:56:21 -04:00

99 lines
2.0 KiB
C++

//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "loopback_channel.h"
#include "utllinkedlist.h"
#include "iphelpers.h"
// -------------------------------------------------------------------------------- //
// CLoopbackChannel.
// -------------------------------------------------------------------------------- //
typedef struct
{
int m_Len;
unsigned char m_Data[1];
} LoopbackMsg_t;
class CLoopbackChannel : public IChannel
{
public:
virtual ~CLoopbackChannel()
{
FOR_EACH_LL( m_Messages, i )
{
free( m_Messages[i] );
}
m_Messages.Purge();
}
virtual void Release()
{
delete this;
}
virtual bool Send( const void *pData, int len )
{
const void *pChunks[1] = { pData };
int chunkLengths[1] = { len };
return SendChunks( pChunks, chunkLengths, 1 );
}
virtual bool SendChunks( void const * const *pChunks, const int *pChunkLengths, int nChunks )
{
CChunkWalker walker( pChunks, pChunkLengths, nChunks );
LoopbackMsg_t *pMsg = (LoopbackMsg_t*)malloc( sizeof( LoopbackMsg_t ) - 1 + walker.GetTotalLength() );
walker.CopyTo( pMsg->m_Data, walker.GetTotalLength() );
pMsg->m_Len = walker.GetTotalLength();
m_Messages.AddToTail( pMsg );
return true;
}
virtual bool Recv( CUtlVector<unsigned char> &data, double flTimeout )
{
int iNext = m_Messages.Head();
if ( iNext == m_Messages.InvalidIndex() )
{
return false;
}
else
{
LoopbackMsg_t *pMsg = m_Messages[iNext];
data.CopyArray( pMsg->m_Data, pMsg->m_Len );
free( pMsg );
m_Messages.Remove( iNext );
return true;
}
}
virtual bool IsConnected()
{
return true;
}
virtual void GetDisconnectReason( CUtlVector<char> &reason )
{
}
private:
CUtlLinkedList<LoopbackMsg_t*,int> m_Messages; // FIFO for messages we've sent.
};
IChannel* CreateLoopbackChannel()
{
return new CLoopbackChannel;
}