324 lines
12 KiB
C++
324 lines
12 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: master header file for socketlib.lib
|
|
//
|
|
// $Header: $
|
|
// $NoKeywords: $
|
|
//===========================================================================//
|
|
|
|
#ifndef SOCKETLIB_H
|
|
#define SOCKETLIB_H
|
|
|
|
#ifdef _WIN32
|
|
#pragma once
|
|
#endif
|
|
|
|
#include "tier0/basetypes.h"
|
|
#include "utlbuffer.h"
|
|
|
|
typedef int64 SocketHandle_t;
|
|
|
|
|
|
//===========================================================================
|
|
// Error Codes used by socketlib
|
|
//===========================================================================
|
|
enum SocketErrorCode_t
|
|
{
|
|
SOCKET_SUCCESS = 0,
|
|
SOCKET_ERR_OPERATION_NOT_SUPPORTED,
|
|
SOCKET_ERR_CREATE_FAILED,
|
|
SOCKET_ERR_READ_OPERATION_FAILED,
|
|
SOCKET_ERR_WRITE_OPERATION_FAILED,
|
|
|
|
SOCKET_ERR_CONNECT_FAILED,
|
|
SOCKET_ERR_LISTEN_FAILED,
|
|
SOCKET_ERR_ACCEPT_FAILED,
|
|
SOCKET_ERR_POLLING_OPERATION_FAILED,
|
|
SOCKET_ERR_BIND_OPERATION_FAILED,
|
|
|
|
SOCKET_ERR_ENABLE_NON_BLOCKING_MODE_FAILED,
|
|
SOCKET_ERR_HOST_NOT_FOUND,
|
|
SOCKET_ERR_GENERAL_SOCKET_ERROR,
|
|
SOCKET_ERR_READ_OPERATION_WOULD_BLOCK,
|
|
SOCKET_ERR_WRITE_OPERATION_WOULD_BLOCK,
|
|
|
|
SOCKET_ERR_CONNECTION_CLOSED,
|
|
SOCKET_ERR_CONNECTION_RESET,
|
|
SOCKET_ERR_NO_INCOMING_CONNECTIONS,
|
|
SOCKET_ERR_NO_AVAILABLE_ENDPOINTS,
|
|
|
|
SOCKET_ERR_BAD_USER_DATA,
|
|
SOCKET_ERR_INVALID_CONNECTION,
|
|
SOCKET_ERR_CANT_WRITE,
|
|
|
|
SOCKET_ERR_MIXING_PACKET_SENDS,
|
|
SOCKET_ERR_PARTIAL_PACKET_OVERFLOW,
|
|
};
|
|
|
|
|
|
const int MAX_SERVER_CONNECTIONS = 4; // Maximum number of concurrent connections allowed to a server.
|
|
// In the future, this can be set higher and class CSocketConnection should use a growable array of Sockets.
|
|
|
|
const int MAX_SERVER_CONNECTION_BACKLOG = 16; // Maximum number of connections allowed in the listening backlog.
|
|
|
|
const int INVALID_ENDPOINT_INDEX = -1; // Represents an invalid endpoint index.
|
|
|
|
|
|
|
|
enum ConnectionType_t // Type of connection: client or server.
|
|
{
|
|
CT_INDETERMINATE = 0,
|
|
CT_CLIENT,
|
|
CT_SERVER,
|
|
};
|
|
|
|
enum SocketProtocol_t // Protocol to use for the socket.
|
|
{
|
|
SP_INDETERMINATE = 0,
|
|
SP_UDP,
|
|
SP_VDP,
|
|
SP_TCP,
|
|
};
|
|
|
|
enum SocketState_t // State of a given socket.
|
|
{
|
|
SSTATE_UNINITIALIZED = 0, // Socket is in completely uninitialized state.
|
|
SSTATE_LISTENING, // Server socket is listening for connections.
|
|
SSTATE_CONNECTION_IN_PROGRESS, // Socket is initialized and in the process of connecting to a client/server.
|
|
SSTATE_CONNECTED, // Socket is initialized and connected to a client/server.
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Platform and build specific defines, change these as needed
|
|
//-----------------------------------------------------------------------
|
|
|
|
#if defined( PLATFORM_X360 )
|
|
#define VDP_SUPPORT_ENABLED 1
|
|
#else
|
|
#define VDP_SUPPORT_ENABLED 0
|
|
#endif
|
|
|
|
#if defined( _DEBUG )
|
|
#define UNSECURE_SOCKETS_ENABLED 1
|
|
#else
|
|
#define UNSECURE_SOCKETS_ENABLED 0
|
|
#endif
|
|
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// Basic platform-specific socket definitions go here to avoid
|
|
// polluting other source files with common #defines.
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#if defined( PLATFORM_X360 )
|
|
#include <xtl.h>
|
|
#include <winsockx.h>
|
|
#elif defined( PLATFORM_WINDOWS_PC )
|
|
#include <winsock2.h>
|
|
#else
|
|
#error No build platform macro (PLATFORM_*) defined
|
|
#endif
|
|
|
|
inline SocketHandle_t GetSocketHandle( SOCKET socket )
|
|
{
|
|
return static_cast<SocketHandle_t>( socket );
|
|
}
|
|
|
|
inline SOCKET GetPlatformSocket( SocketHandle_t handle )
|
|
{
|
|
return static_cast<SOCKET>( handle );
|
|
}
|
|
|
|
static const SocketHandle_t InvalidSocketHandle = static_cast<SocketHandle_t>( INVALID_SOCKET );
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// CSocketConnection
|
|
// Represents a client or server network connection.
|
|
//
|
|
// todo: Rework connection class to treat endpoints as standalone objects and unify
|
|
// interface with named pipes.
|
|
// ----------------------------------------------------------------------------
|
|
class CSocketConnection
|
|
{
|
|
public:
|
|
CSocketConnection();
|
|
~CSocketConnection();
|
|
|
|
SocketErrorCode_t Init( ConnectionType_t connectionType, SocketProtocol_t socketProtocol ); // Initializes the connection class as either client or server with specified protocol.
|
|
// Does not acquire any system resources.
|
|
|
|
void Cleanup(); // Cleans up the class and restores it to the default uninitialized state.
|
|
|
|
SocketErrorCode_t Listen( uint16 localPort, int numAllowedConnections ); // (Server-only) Listens for active client connections.
|
|
|
|
SocketErrorCode_t TryAcceptIncomingConnection( int *newEndpointIndex ); // (Server-only) Attempts to accept an incoming connection, if one is available.
|
|
|
|
int GetFirstAvailableListeningEndpoint(); // (Server-only) Gets the index of the first endpoint which is listening for connections.
|
|
|
|
SocketState_t GetListeningSocketState(); // (Server-only) Gets the status of the listening socket on a server.
|
|
|
|
SocketErrorCode_t ConnectToServer( const char *hostName, uint16 hostPort ); // (Client-only) Connects to a remote host using endpoint 0.
|
|
// In most cases, the status of endpoint 0 will be SSTATE_CONNECTION_IN_PROGRESS.
|
|
// The caller must periodically call PollClientConnectionState until the socket is connected before it is usable.
|
|
|
|
SocketErrorCode_t PollClientConnectionState( bool *isConnected ); // (Client-only) Checks the status of endpoint 0 to see if it is connected successfully to a server.
|
|
|
|
SocketState_t GetEndpointSocketState( int endpointIndex ); // Gets the status of the specified endpoint. (Clients only use endpoint 0.)
|
|
|
|
SocketErrorCode_t CanReadFromEndpoint( int endpointIndex, bool *canRead ); // Gets a value indicating whether the endpoint can be successfully read from.
|
|
|
|
SocketErrorCode_t CanWriteToEndpoint( int endpointIndex, bool *canWrite ); // Gets a value indicating whether the endpoint can be successfully written to.
|
|
|
|
SocketErrorCode_t ReadFromEndpoint( int endpointIndex, byte *destinationBuffer, int bufferSize, int *bytesRead ); // Reads data from the specified endpoint.
|
|
|
|
SocketErrorCode_t WriteToEndpoint( int endpointIndex, byte *sourceBuffer, int bufferSize, int *bytesWritten ); // Writes data to the specified endpoint.
|
|
|
|
const char* GetLastSystemErrorString() const; // Gets a string containing the last underlying system error reported.
|
|
// Only valid immediately after a function returns an unsuccessful error code.
|
|
const char* GetLastErrorString() const; // Gets a string containing the last underlying system error reported.
|
|
|
|
SocketHandle_t GetListeningSocket() const;
|
|
|
|
SocketHandle_t GetEndpointSocket( int nEndpointIndex ) const;
|
|
|
|
void ResetEndpoint( int endpointIndex );
|
|
|
|
SocketErrorCode_t SetSocketOpt( int endpointIndex, int level, int optname, const void *optval, int optlen );
|
|
|
|
private:
|
|
SocketHandle_t CreateNewSocket();
|
|
|
|
SocketHandle_t m_ListeningSocket;
|
|
SocketHandle_t m_EndpointSockets[ MAX_SERVER_CONNECTIONS ];
|
|
|
|
SocketState_t m_ListeningSocketState;
|
|
SocketState_t m_EndpointStates[ MAX_SERVER_CONNECTIONS ];
|
|
|
|
ConnectionType_t m_ConnectionType;
|
|
SocketProtocol_t m_SocketProtocol;
|
|
|
|
SocketErrorCode_t m_LastError;
|
|
int m_LastSystemError;
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Inline functions from CSocketConnection
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
inline SocketHandle_t CSocketConnection::GetListeningSocket() const
|
|
{
|
|
return m_ListeningSocket;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
inline SocketHandle_t CSocketConnection::GetEndpointSocket( int nEndpointIndex ) const
|
|
{
|
|
AssertMsg( nEndpointIndex >= 0 && nEndpointIndex < MAX_SERVER_CONNECTIONS, "Endpoint index out of range" );
|
|
|
|
return m_EndpointSockets[ nEndpointIndex ];
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// Start up and shutdown functions
|
|
// Calls to these functions have to bracket any usage of this socket code
|
|
// ----------------------------------------------------------------------------
|
|
void SocketLibInit();
|
|
void SocketLibShutdown();
|
|
|
|
// Handy conversion to text of those nasty HRESULT codes
|
|
const char* ConvertWinsockErrorToString( int errorCode );
|
|
const char* ConvertSocketLibErrorToString( SocketErrorCode_t errorCode );
|
|
|
|
|
|
|
|
// Bitwise representation of a network message header.
|
|
struct MessageHeader_t
|
|
{
|
|
uint32 m_nLength;
|
|
};
|
|
|
|
// Byte swaps (in-place) a message header between system byte-order and network byte-order.
|
|
void ByteSwapInPlaceMessageHeader( MessageHeader_t* messageHeader );
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// Signature of a callback invoked once per message parsed by the CSocketMessageBuilder::FeedData function.
|
|
// header = Header of the message read.
|
|
// message = Message payload (m_nLength is given by the header)
|
|
// userContext = Context provided by caller to FeedData
|
|
// ----------------------------------------------------------------------------
|
|
typedef void ( *NetworkMessageHandler )( const MessageHeader_t& header, const byte* message, void* userContext );
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// Helper class to parse messages from a stream source.
|
|
// This class maintains state between successive calls to FeedData so that partial messages
|
|
// can be stored until fully parsed and dispatched.
|
|
// ----------------------------------------------------------------------------
|
|
class CSocketMessageBuilder
|
|
{
|
|
public:
|
|
CSocketMessageBuilder( int initialSize = 0, int growSize = 0 );
|
|
~CSocketMessageBuilder();
|
|
|
|
void SetMaxExpectedMsgSize( int expectedSize );
|
|
|
|
// Parses the given stream data and fires a callback for each full message parsed.
|
|
// This function has the side-effect of updating the CSocketMessageBuilder's internal parsing state
|
|
// so that partial messages can be glued together.
|
|
void FeedData( const void *data, int dataLength, NetworkMessageHandler networkMessageHandlerFunc, void *userContext );
|
|
|
|
void AssignConnection( CSocketConnection* pConnection, int endpoint );
|
|
SocketErrorCode_t SendDataPacket( const void* RESTRICT data, int dataLength );
|
|
SocketErrorCode_t SendDataPacket( CSocketConnection* pConnection, int endpoint, const void* RESTRICT data, int dataLength );
|
|
|
|
SocketErrorCode_t BeginSendPartialDataPacket( uint32 totalSize, const void* RESTRICT data, int dataLength );
|
|
SocketErrorCode_t BeginSendPartialDataPacket( CSocketConnection* pConnection, int endpoint, uint32 totalSize, const void* RESTRICT data, int dataLength );
|
|
|
|
bool WaitForIncomingMessage( DWORD timeOutValue );
|
|
void* GetIncomingMessageData();
|
|
uint32 GetIncomingMessageLen();
|
|
|
|
void FeedDataManual( const void* RESTRICT data, int dataLength );
|
|
bool HasCompleteMessageManual( );
|
|
bool GetCurrentMessageManual( void*& msgData, uint32& msgSize );
|
|
bool DiscardCurrentMessageManual();
|
|
|
|
bool IsSendingPartialMessage() { return m_bSendingPartialMessage; }
|
|
int32 PartialMessageBytesRemaining() { return m_bSendingPartialMessage ? m_PartialMessageBytesTotal - m_PartialMessageBytesSent : 0; }
|
|
|
|
private:
|
|
MessageHeader_t m_MessageHeader;
|
|
uint32 m_nHeaderBytesRead;
|
|
uint32 m_nMessageBytesRead;
|
|
|
|
CSocketConnection* m_pConnection;
|
|
int m_nConnectionEndpoint;
|
|
|
|
CUtlBuffer m_MessageData;
|
|
|
|
uint32 m_PartialMessageBytesSent; // used to combine multiple calls to SendDataPacket/BeginSendPartialDataPacket into one message packet
|
|
uint32 m_PartialMessageBytesTotal;
|
|
bool m_bSendingPartialMessage;
|
|
|
|
int m_nRecvBufSize;
|
|
byte* m_pRecvBuf;
|
|
bool m_bSwappedHeader;
|
|
|
|
|
|
};
|
|
|
|
|
|
#endif // SOCKETLIB_H
|
|
|