mirror of https://github.com/tongzx/nt5src
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.
1357 lines
33 KiB
1357 lines
33 KiB
/*++
|
|
|
|
Copyright (C) Microsoft Corporation, 1996 - 1999
|
|
|
|
Module Name:
|
|
|
|
trans.hxx
|
|
|
|
Abstract:
|
|
|
|
Commen base for all NT transport interfaces.
|
|
|
|
Author:
|
|
|
|
Mario Goertzel [MarioGo]
|
|
|
|
Revision History:
|
|
|
|
MarioGo 4/12/1996 Bits 'n pieces
|
|
MarioGo 10/24/1996 Async RPC
|
|
EdwardR 07/04/1997 Falcon/RPC
|
|
|
|
--*/
|
|
|
|
#ifndef __TRANS_HXX
|
|
#define __TRANS_HXX
|
|
|
|
#include "Dbg.hxx"
|
|
|
|
//
|
|
// Winsock address types.
|
|
//
|
|
|
|
union WS_SOCKADDR
|
|
{
|
|
SOCKADDR generic;
|
|
SOCKADDR_IN inetaddr;
|
|
#ifndef SPX_IPX_OFF
|
|
SOCKADDR_IPX ipxaddr;
|
|
#endif
|
|
SOCKADDR_AT ataddr;
|
|
#ifdef NETBIOS_ON
|
|
SOCKADDR_NB nbaddr;
|
|
#endif
|
|
SOCKADDR_CLUSTER clusaddr;
|
|
SOCKADDR_STORAGE ipaddr;
|
|
};
|
|
|
|
inline void
|
|
RpcpSetIpPort (
|
|
IN WS_SOCKADDR *SockAddress,
|
|
IN USHORT Port
|
|
)
|
|
{
|
|
SS_PORT(SockAddress) = Port;
|
|
}
|
|
|
|
inline USHORT
|
|
RpcpGetIpPort (
|
|
IN WS_SOCKADDR *SockAddress
|
|
)
|
|
{
|
|
return SS_PORT(SockAddress);
|
|
}
|
|
|
|
inline void
|
|
RpcpCopyIPv6Address (
|
|
IN SOCKADDR_IN6 *SourceAddress,
|
|
OUT SOCKADDR_IN6 *TargetAddress
|
|
)
|
|
{
|
|
RpcpMemoryCopy(&TargetAddress->sin6_addr,
|
|
&SourceAddress->sin6_addr,
|
|
sizeof(in6_addr));
|
|
}
|
|
|
|
inline void
|
|
RpcpCopyIPv4Address (
|
|
IN SOCKADDR_IN *SourceAddress,
|
|
OUT SOCKADDR_IN *TargetAddress
|
|
)
|
|
{
|
|
TargetAddress->sin_addr.s_addr = SourceAddress->sin_addr.s_addr;
|
|
}
|
|
|
|
|
|
//
|
|
//
|
|
// Async object types
|
|
//
|
|
|
|
struct BASE_ASYNC_OBJECT;
|
|
|
|
//
|
|
// Every outstanding async IO has an overlapped structure associated
|
|
// with it.
|
|
//
|
|
struct BASE_OVERLAPPED
|
|
{
|
|
//
|
|
// Pointer to the transport object this IO is associated with.
|
|
// This is needed since a single object may have more than one
|
|
// pending IO request.
|
|
//
|
|
BASE_ASYNC_OBJECT *pAsyncObject;
|
|
//
|
|
// System overlapped structure associated with the async IO.
|
|
//
|
|
OVERLAPPED ol;
|
|
//
|
|
// RPC thread object of the thread which started the IO. This is used
|
|
// when the IO completes to keep the count of IO pending on a thread.
|
|
//
|
|
PVOID thread;
|
|
};
|
|
|
|
typedef BASE_OVERLAPPED *PBASE_OVERLAPPED;
|
|
|
|
|
|
#ifdef NCADG_MQ_ON
|
|
struct MQ_OVERLAPPED : BASE_OVERLAPPED
|
|
{
|
|
// MSMQ Messages have there own structure which must be maintained
|
|
// during the span of the async IO request:
|
|
MQMSGPROPS msgProps;
|
|
MSGPROPID aMsgPropID[MAX_RECV_VAR];
|
|
MQPROPVARIANT aMsgPropVar[MAX_RECV_VAR];
|
|
HRESULT aStatus[MAX_RECV_VAR];
|
|
};
|
|
|
|
typedef MQ_OVERLAPPED *PMQ_OVERLAPPED;
|
|
#endif
|
|
|
|
|
|
// BASE_ASYNC_OBJECT is the basis for all objects which
|
|
// are used in async I/O.
|
|
// There are three basic objects which are used in I/O,
|
|
// addresses, connections and datagrams.
|
|
//
|
|
// BASE_ADDRESS is the basis for all I/O which is
|
|
// used to listen for new client connections.
|
|
//
|
|
// BASE_CONNECTION is the basis for all I/O which is
|
|
// used to read/write to a connection.
|
|
//
|
|
// There are currently three flavors of address and
|
|
// and connections:
|
|
// WS_ (winsock connection base protocols)
|
|
// NB_ (not supported on Itaniums - winsock based, but unique historical reasons)
|
|
// NMP_ (named pipes)
|
|
//
|
|
// BASE_DATAGRAM is the basis for all I/O which is pending
|
|
// on a datagram port. The only flavor today is winsock.
|
|
//
|
|
//
|
|
// BASE_ASYNC_OBJECT
|
|
// |
|
|
// +---------------------|-------------------------+
|
|
// BASE_ADDRESS BASE_CONNECTION BASE_DATAGRAM
|
|
// | | | | | | |
|
|
// CO_ADDRESS | WS_DG_ADDR | WS_CONNECTION | WS_DATAGRAM
|
|
// | | MQ_DG_ADDR | | MQ_DATAGRAM
|
|
// NMP_ADDR | NMP_CONNECTION |
|
|
// WS_ADDR |
|
|
// | +---------+----------+
|
|
// NB_ADDR | |
|
|
// WS_CLIENT_CONNECTION WS_SAN_CONNECTION
|
|
// |
|
|
// |
|
|
// WS_SAN_CLIENT_CONNECTION
|
|
//
|
|
|
|
struct BASE_ASYNC_OBJECT;
|
|
|
|
struct BASE_ASYNC_OBJECT
|
|
{
|
|
//
|
|
// a placeholder for the vtbl of derived objects
|
|
// this makes casts safe and fast at the expense of some
|
|
// memory waste. That's ok
|
|
//
|
|
virtual void DoNothing(void)
|
|
{
|
|
}
|
|
//
|
|
// The type of this object. Used in determining where
|
|
// to send the completed I/O.
|
|
//
|
|
RPC_TRANSPORT_EVENT type;
|
|
//
|
|
// Identifies the protcol of the address/connection.
|
|
//
|
|
PROTOCOL_ID id;
|
|
|
|
//
|
|
// > 0 means that the object has been aborted
|
|
//
|
|
LONG fAborted;
|
|
|
|
//
|
|
// Used to chain objects belonging to the
|
|
// same protocol.
|
|
//
|
|
LIST_ENTRY ObjectList;
|
|
};
|
|
|
|
typedef BASE_ASYNC_OBJECT *PREQUEST;
|
|
|
|
struct BASE_ADDRESS;
|
|
class BASE_CONNECTION;
|
|
|
|
typedef void (*TRANS_ADDRESS_LISTEN)(BASE_ADDRESS *);
|
|
|
|
enum ADDRESS_STATE {
|
|
NotInList,
|
|
InTheList,
|
|
Inactive
|
|
};
|
|
|
|
#define TRANSPORT_POSTED_KEY UINT_PTR(0xFFFF0000)
|
|
|
|
//
|
|
// Address objects represent a connection oriented endpoint which
|
|
// is connected to by a client at which time a connection object
|
|
// is created for the specific client.
|
|
// There are relatively few address objects in the system
|
|
//
|
|
struct BASE_ADDRESS : BASE_ASYNC_OBJECT
|
|
{
|
|
//
|
|
// The endpoint this address is listening on.
|
|
//
|
|
RPC_CHAR *Endpoint;
|
|
//
|
|
// List of network addresses for this address
|
|
//
|
|
NETWORK_ADDRESS_VECTOR *pAddressVector;
|
|
//
|
|
// Function to call when a listen is aborted or when
|
|
// an address doesn't have an outstanding listens.
|
|
//
|
|
TRANS_ADDRESS_LISTEN SubmitListen;
|
|
//
|
|
// NotInList is address in not in the AddressManager list
|
|
// InTheList if the address has been inserted into the list
|
|
// InActive if it is in the list, but it is inactive
|
|
//
|
|
ADDRESS_STATE InAddressList;
|
|
//
|
|
// Endpoint flags used in conjunction with firewalls
|
|
//
|
|
ULONG EndpointFlags;
|
|
//
|
|
// If an address is unable to (or has not yet) submitted
|
|
// an listen request then it is stuck into a linked list.
|
|
// This is the forward list pointer.
|
|
//
|
|
BASE_ADDRESS *pNext;
|
|
|
|
//
|
|
// In order to callback into the runtime we must pass the
|
|
// first (runtime allocated) address when calling back.
|
|
//
|
|
struct BASE_ADDRESS *pFirstAddress;
|
|
|
|
//
|
|
// Each netbios address may represent several listen sockets (one
|
|
// for each lana). A list of these is maintained to facilitate aborting
|
|
// the setup of an entire address.
|
|
//
|
|
struct BASE_ADDRESS *pNextAddress;
|
|
|
|
//
|
|
// Static or dynamic ?
|
|
//
|
|
BOOL fDynamicEndpoint;
|
|
};
|
|
|
|
struct CO_ADDRESS;
|
|
|
|
typedef RPC_STATUS (*TRANS_NEW_CONNECTION)(CO_ADDRESS *, BASE_CONNECTION **);
|
|
|
|
struct CO_ADDRESS : BASE_ADDRESS
|
|
{
|
|
//
|
|
// Overlapped object associated with the pending accept/connect
|
|
//
|
|
BASE_OVERLAPPED Listen;
|
|
//
|
|
// Function to call when a connection notification arrives.
|
|
//
|
|
TRANS_NEW_CONNECTION NewConnection;
|
|
};
|
|
|
|
typedef CO_ADDRESS *PADDRESS;
|
|
|
|
struct NMP_ADDRESS : CO_ADDRESS
|
|
{
|
|
//
|
|
// The handle of the pipe instance currently
|
|
// avaliable for clients to connect to.
|
|
//
|
|
HANDLE hConnectPipe;
|
|
//
|
|
// When we disconnect a client we save an extra pipe instance here
|
|
// to use on the next client connection. This is a performance optimization,
|
|
// but also affects correctness. See NMP_ServerListen where the first
|
|
// spare pipe is created.
|
|
//
|
|
HandleCache sparePipes;
|
|
//
|
|
// The self relative security descriptor associated with
|
|
// this addresss.
|
|
//
|
|
PSECURITY_DESCRIPTOR SecurityDescriptor;
|
|
//
|
|
// The complete pipe name for this addresses
|
|
// endpoint including "\\."
|
|
//
|
|
RPC_CHAR *LocalEndpoint;
|
|
};
|
|
|
|
typedef NMP_ADDRESS *PNMP_ADDRESS;
|
|
|
|
struct WS_ADDRESS : CO_ADDRESS
|
|
{
|
|
//
|
|
// The socket listening on this addresses port.
|
|
//
|
|
SOCKET ListenSocket;
|
|
|
|
//
|
|
// The socket of the next client to connect. It is waiting
|
|
// in an AcceptEx for the client to connect.
|
|
//
|
|
SOCKET ConnectionSocket;
|
|
|
|
//
|
|
// Netbios requires that the a protocol be multiplied by the
|
|
// lana number. For other protocols this should be one.
|
|
//
|
|
DWORD ProtocolMultiplier;
|
|
|
|
//
|
|
// The PendingQueueSize parameter supplied when the address
|
|
// was created
|
|
//
|
|
INT QueueSize;
|
|
|
|
//
|
|
// Buffer for the address of the client which is part of the
|
|
// outstanding AcceptEx call on the listen socket.
|
|
//
|
|
BYTE AcceptBuffer[sizeof(WS_SOCKADDR) + 16];
|
|
|
|
//
|
|
// the sockaddr used to setup this address
|
|
//
|
|
WS_SOCKADDR ListenAddr;
|
|
|
|
union
|
|
{
|
|
struct
|
|
{
|
|
LPFN_ACCEPTEX pAcceptExFunction;
|
|
LPFN_GETACCEPTEXSOCKADDRS pGetAcceptExSockaddressFunction;
|
|
};
|
|
void *ExtensionFunctionPointers[2];
|
|
};
|
|
|
|
static const int AcceptExFunctionId;
|
|
static const int GetAcceptExSockAddressFunctionId;
|
|
|
|
static const UUID ExtensionFunctionsUuids[];
|
|
|
|
BOOL GetExtensionFunctionPointers(SOCKET sock);
|
|
BOOL GetExtensionFunctionPointerForFunction(SOCKET sock, int nFunctionCode);
|
|
};
|
|
|
|
inline DWORD
|
|
GetProtocolMultiplier (
|
|
IN WS_ADDRESS *Address
|
|
)
|
|
{
|
|
#ifdef NETBIOS_ON
|
|
return Address->ProtocolMultiplier;
|
|
#else
|
|
return 1;
|
|
#endif
|
|
}
|
|
|
|
inline void
|
|
SetProtocolMultiplier (
|
|
IN WS_ADDRESS *Address,
|
|
IN DWORD ProtocolMultiplier
|
|
)
|
|
{
|
|
#ifdef NETBIOS_ON
|
|
Address->ProtocolMultiplier = ProtocolMultiplier;
|
|
#endif
|
|
}
|
|
|
|
typedef WS_ADDRESS *PWS_ADDRESS;
|
|
|
|
struct WS_DATAGRAM;
|
|
typedef WS_DATAGRAM *PWS_DATAGRAM;
|
|
|
|
struct WS_DATAGRAM_ENDPOINT : BASE_ADDRESS
|
|
{
|
|
// WS_DATAGRAM_ENDPOINTs represent either a client or server
|
|
// port. There will be a small number of these on servers and
|
|
// O(N) active threads on sync clients.
|
|
|
|
//
|
|
// The socket we're listen on.
|
|
//
|
|
SOCKET Socket;
|
|
//
|
|
// Normally FALSE, set to true by a thread which is submitting
|
|
// new IOs on the endpoint.
|
|
//
|
|
LONG fSubmittingIos;
|
|
//
|
|
// Current number of WS_DATAGRAM's submitted on this endpoint.
|
|
// Must be changed via InterlockedInc/Dec. The count of
|
|
// non-null entries in aIdleDatagrams.
|
|
//
|
|
LONG cPendingIos;
|
|
//
|
|
// If cPendingIos is less than the minimum then recvs on any idle
|
|
// WS_DATAGRAMs should be posted.
|
|
//
|
|
// const after initialization.
|
|
//
|
|
LONG cMinimumIos;
|
|
//
|
|
// The number of WS_DATAGRAMs available for this endpoint to use.
|
|
//
|
|
// const after initialization.
|
|
//
|
|
LONG cMaximumIos;
|
|
//
|
|
// Array of cMaxIos datagrams. NULL in sync endpoints.
|
|
//
|
|
PWS_DATAGRAM aDatagrams;
|
|
//
|
|
// Client or Server ?
|
|
//
|
|
BOOL fClient;
|
|
//
|
|
// Sync or Async ?
|
|
//
|
|
BOOL fAsync;
|
|
//
|
|
// Sockaddr
|
|
//
|
|
WS_SOCKADDR ListenAddr;
|
|
};
|
|
|
|
#ifdef NCADG_MQ_ON
|
|
struct MQ_DATAGRAM;
|
|
typedef MQ_DATAGRAM *PMQ_DATAGRAM;
|
|
|
|
struct MQ_DATAGRAM_ENDPOINT : BASE_ADDRESS
|
|
{
|
|
// MQ_DATAGRAM_ENDPOINTs represent either a client or server
|
|
// port. There will be a small number of these on servers and
|
|
// O(N) active threads on sync clients.
|
|
|
|
//
|
|
// The queue we're listen on.
|
|
//
|
|
UUID uuidQType; //
|
|
QUEUEHANDLE hQueue;
|
|
QUEUEHANDLE hAdminQueue;
|
|
BOOL fAllowReceives;
|
|
RPC_CHAR wsMachine[MAX_COMPUTERNAME_LEN];
|
|
RPC_CHAR wsQName[MQ_MAX_Q_NAME_LEN];
|
|
RPC_CHAR wsQPathName[MAX_PATHNAME_LEN];
|
|
RPC_CHAR wsQFormat[MAX_FORMAT_LEN];
|
|
RPC_CHAR wsAdminQFormat[MAX_FORMAT_LEN];
|
|
ULONG ulDelivery;
|
|
ULONG ulPriority;
|
|
ULONG ulJournaling;
|
|
ULONG ulTimeToReachQueue; // Seconds.
|
|
ULONG ulTimeToReceive; // Seconds.
|
|
BOOL fAck;
|
|
BOOL fAuthenticate; // Server security tracking.
|
|
BOOL fEncrypt;
|
|
ULONG ulPrivacyLevel; // Server security tracking.
|
|
|
|
//
|
|
// Normally FALSE, set to true by a thread which is submitting
|
|
// new IOs on the endpoint.
|
|
//
|
|
LONG fSubmittingIos;
|
|
//
|
|
// Current number of WS_DATAGRAM's submitted on this endpoint.
|
|
// Must be changed via InterlockedInc/Dec. The count of
|
|
// non-null entries in aIdleDatagrams.
|
|
//
|
|
LONG cPendingIos;
|
|
//
|
|
// If cPendingIos is less than the minimum then recvs on any idle
|
|
// WS_DATAGRAMs should be posted.
|
|
//
|
|
// const after initialization.
|
|
//
|
|
LONG cMinimumIos;
|
|
//
|
|
// The number of WS_DATAGRAMs available for this endpoint to use.
|
|
//
|
|
// const after initialization.
|
|
//
|
|
LONG cMaximumIos;
|
|
//
|
|
// Array of cMaxIos datagrams. NULL in sync endpoints.
|
|
//
|
|
PMQ_DATAGRAM aDatagrams;
|
|
};
|
|
#endif
|
|
|
|
//
|
|
// Structure allocated by the runtime and associated with
|
|
// a pending send operation.
|
|
//
|
|
|
|
struct CO_SEND_CONTEXT
|
|
{
|
|
//
|
|
// Overlapped object associated with the pending async write.
|
|
//
|
|
BASE_OVERLAPPED Write;
|
|
//
|
|
// The buffer which is currently being written
|
|
//
|
|
BUFFER pWriteBuffer;
|
|
//
|
|
// Size of the write buffer (as far as we know)
|
|
//
|
|
DWORD maxWriteBuffer;
|
|
};
|
|
|
|
#if defined(_ALPHA_)
|
|
#ifdef __cplusplus
|
|
extern "C" { __int64 __asm(char *,...); };
|
|
#pragma intrinsic(__asm)
|
|
#endif
|
|
|
|
#define MBInstruction __asm("mb")
|
|
#endif
|
|
|
|
//
|
|
// Connection objects are used for async reads (future: writes)
|
|
// from a single client based on that client's connection to
|
|
// this server.
|
|
//
|
|
// There maybe 100's or 1000's of connection objects allocated.
|
|
//
|
|
|
|
class BASE_CONNECTION : public BASE_ASYNC_OBJECT
|
|
{
|
|
public:
|
|
BASE_CONNECTION(void)
|
|
{
|
|
}
|
|
|
|
void
|
|
Initialize (
|
|
void
|
|
);
|
|
|
|
//
|
|
// The socket (or handle) of the client connection.
|
|
// We use a union to avoid type casting this everywhere.
|
|
//
|
|
union {
|
|
SOCKET Socket;
|
|
HANDLE Handle;
|
|
} Conn;
|
|
private:
|
|
//
|
|
// Incremented when a thread is just about to start an IO, just
|
|
// before it checks fAborted. The aborting thread must wait for
|
|
// this to reach 0 before closing the connection.
|
|
//
|
|
LONG StartingWriteIo;
|
|
LONG StartingReadIo;
|
|
|
|
public:
|
|
//
|
|
// We use a heuristic for choosing the size of receives to post. This
|
|
// starts are CO_MIN_RECV size and is increased as we see larger
|
|
// IOs on the connection.
|
|
UINT iPostSize;
|
|
//
|
|
// The size of the outstanding read buffer.
|
|
//
|
|
DWORD maxReadBuffer;
|
|
//
|
|
// The number of bytes in pReadBuffer that where read
|
|
// as part of a previous read.
|
|
//
|
|
UINT iLastRead;
|
|
//
|
|
// Overlapped object associated with the pending async read
|
|
//
|
|
BASE_OVERLAPPED Read;
|
|
//
|
|
// A buffer for the outstanding read of maxReadBuffer bytes.
|
|
// Also used a flag, if 0 then no read is pending.
|
|
//
|
|
BUFFER pReadBuffer;
|
|
|
|
inline void StartingWriteIO(void)
|
|
{
|
|
InterlockedIncrement(&StartingWriteIo);
|
|
}
|
|
|
|
inline void StartingReadIO(void)
|
|
{
|
|
#if defined(i386) || defined (_ALPHA_)
|
|
// if we are the first, we know there won't be other guys around
|
|
if (StartingReadIo == 0)
|
|
{
|
|
StartingReadIo = 1;
|
|
#if defined (_ALPHA_)
|
|
MBInstruction;
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
// there may be other guys around - be safe
|
|
InterlockedIncrement(&StartingReadIo);
|
|
}
|
|
#else
|
|
InterlockedIncrement(&StartingReadIo);
|
|
#endif
|
|
}
|
|
|
|
inline void StartingOtherIO(void)
|
|
{
|
|
// we use the StartingWriteIo because it
|
|
// doesn't shortcut transition from 0 to 1st
|
|
InterlockedIncrement(&StartingWriteIo);
|
|
}
|
|
|
|
inline void WriteIOFinished(void)
|
|
{
|
|
InterlockedDecrement(&StartingWriteIo);
|
|
}
|
|
|
|
// ********************* NOTE *************************************
|
|
// After you return from this function, if the read is successful, you
|
|
// can no longer touch the connection (transport level or runtime) -
|
|
// it may have been destroyed
|
|
inline void ReadIOFinished(void)
|
|
{
|
|
InterlockedDecrement(&StartingReadIo);
|
|
}
|
|
|
|
inline void OtherIOFinished(void)
|
|
{
|
|
InterlockedDecrement(&StartingWriteIo);
|
|
}
|
|
|
|
inline BOOL IsIoStarting(void)
|
|
{
|
|
return StartingWriteIo || StartingReadIo;
|
|
}
|
|
|
|
inline void InitIoCounter(void)
|
|
{
|
|
StartingWriteIo = 0;
|
|
StartingReadIo = 0;
|
|
}
|
|
|
|
virtual RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped) = 0;
|
|
|
|
virtual RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped) = 0;
|
|
|
|
virtual RPC_STATUS ProcessRead(IN DWORD bytes, OUT BUFFER *pBuffer,
|
|
OUT PUINT pBufferLength);
|
|
|
|
virtual RPC_STATUS Abort(void) = 0;
|
|
|
|
};
|
|
|
|
typedef BASE_CONNECTION *PCONNECTION;
|
|
|
|
RPC_STATUS
|
|
UTIL_ReadFile(
|
|
IN HANDLE hFile,
|
|
IN LPVOID lpBuffer,
|
|
IN DWORD nNumberOfBytesToRead,
|
|
OUT LPDWORD lpNumberOfBytesRead,
|
|
IN OUT LPOVERLAPPED lpOverlapped
|
|
);
|
|
|
|
RPC_STATUS
|
|
UTIL_WriteFile(
|
|
IN HANDLE hFile,
|
|
IN LPCVOID lpBuffer,
|
|
IN DWORD nNumberOfBytesToWrite,
|
|
OUT LPDWORD lpNumberOfBytesWritten,
|
|
IN OUT LPOVERLAPPED lpOverlapped
|
|
);
|
|
|
|
class NMP_CONNECTION : public BASE_CONNECTION
|
|
{
|
|
public:
|
|
//
|
|
// Pointer to my address used to store any extra pipe
|
|
// instance when closed.
|
|
//
|
|
PNMP_ADDRESS pAddress;
|
|
|
|
RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return UTIL_ReadFile(hFile, lpBuffer, nNumberOfBytesToRead,
|
|
lpNumberOfBytesRead, lpOverlapped);
|
|
}
|
|
|
|
RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return UTIL_WriteFile(hFile, lpBuffer, nNumberOfBytesToWrite,
|
|
lpNumberOfBytesWritten, lpOverlapped);
|
|
}
|
|
|
|
virtual RPC_STATUS Abort(void);
|
|
|
|
};
|
|
|
|
typedef NMP_CONNECTION *PNMP_CONNECTION;
|
|
|
|
class WS_CONNECTION : public BASE_CONNECTION
|
|
{
|
|
public:
|
|
WS_CONNECTION(void)
|
|
{
|
|
}
|
|
//
|
|
// The address of the client is returned as part of
|
|
// the connection and saved here to support
|
|
// *_QueryClientAddress().
|
|
//
|
|
WS_SOCKADDR saClientAddress;
|
|
WS_ADDRESS *pAddress;
|
|
|
|
virtual RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped);
|
|
|
|
virtual RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped);
|
|
|
|
virtual RPC_STATUS Abort(void);
|
|
|
|
};
|
|
|
|
typedef WS_CONNECTION *PWS_CONNECTION;
|
|
|
|
class SAN_CONNECTION
|
|
{
|
|
public:
|
|
SAN_CONNECTION(void)
|
|
{
|
|
}
|
|
|
|
RPC_STATUS SANReceive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped);
|
|
|
|
RPC_STATUS SANSend(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped);
|
|
|
|
private:
|
|
DWORD m_dwFlags;
|
|
};
|
|
|
|
class HTTP2SocketTransportChannel; // forward
|
|
class WS_HTTP2_CONNECTION;
|
|
|
|
typedef RPC_STATUS
|
|
(RPC_ENTRY *HTTP2_READ_HEADER)
|
|
(
|
|
IN WS_HTTP2_CONNECTION *Connection,
|
|
IN ULONG BytesRead,
|
|
OUT ULONG *NewBytesRead
|
|
);
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Read a channel HTTP header (usually some string). In success
|
|
case, there is real data in Connection->pReadBuffer. The
|
|
number of bytes there is in NewBytesRead
|
|
|
|
Arguments:
|
|
|
|
Connection - the connection on which the header arrived.
|
|
|
|
BytesRead - the bytes received from the net
|
|
|
|
NewBytesRead - the bytes read from the channel (success only)
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK or other RPC_S_* errors for error
|
|
|
|
--*/
|
|
|
|
|
|
class WS_HTTP2_CONNECTION : public WS_CONNECTION, public SAN_CONNECTION
|
|
{
|
|
public:
|
|
HTTP2SocketTransportChannel *Channel;
|
|
|
|
RPC_STATUS ProcessReceiveFailed (IN RPC_STATUS EventStatus);
|
|
|
|
virtual RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped);
|
|
|
|
virtual RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped);
|
|
|
|
RPC_STATUS ProcessSendComplete (
|
|
IN RPC_STATUS EventStatus,
|
|
IN CO_SEND_CONTEXT *SendContext
|
|
);
|
|
|
|
// no-op for compatibility with common transport layer
|
|
virtual RPC_STATUS ProcessRead(
|
|
IN DWORD bytes,
|
|
OUT BUFFER *pBuffer,
|
|
OUT PUINT pBufferLength
|
|
);
|
|
|
|
// the actual read
|
|
RPC_STATUS ProcessReceiveComplete(
|
|
IN DWORD bytes,
|
|
OUT BUFFER *pBuffer,
|
|
OUT PUINT pBufferLength
|
|
);
|
|
|
|
void Initialize (
|
|
void
|
|
);
|
|
|
|
void Free (
|
|
void
|
|
);
|
|
|
|
// no-op for HTTP2 connections. They get
|
|
// aborted from RealAbort
|
|
virtual RPC_STATUS Abort(void);
|
|
|
|
// actual code to abort the connection
|
|
void RealAbort(void);
|
|
|
|
BOOL HeaderRead;
|
|
|
|
HTTP2_READ_HEADER ReadHeaderFn;
|
|
|
|
void *RuntimeConnectionPtr; // the transport connection from
|
|
// runtime perspective. Never called
|
|
// directly - just a token to pass
|
|
// to runtime
|
|
};
|
|
|
|
// WS_HTTP2_INITIAL_CONNECTION - a version of WS_HTTP2_CONNECTION that
|
|
// is used before the type of connection HTTP2 or HTTP is known on the
|
|
// server. It has the capability to recognize the first packet and morph
|
|
// into WS_HTTP2_CONNECTION (for HTTP2) or WS_CONNECTION (for HTTP)
|
|
class WS_HTTP2_INITIAL_CONNECTION : public WS_HTTP2_CONNECTION
|
|
{
|
|
public:
|
|
virtual RPC_STATUS ProcessRead(
|
|
IN DWORD bytes,
|
|
OUT BUFFER *pBuffer,
|
|
OUT PUINT pBufferLength
|
|
);
|
|
|
|
virtual RPC_STATUS Abort(void);
|
|
};
|
|
|
|
class WS_CLIENT_CONNECTION : public WS_CONNECTION
|
|
{
|
|
public:
|
|
// Additional state needed in client-side sync calls on TCP/IP to
|
|
// handle shutdowns.
|
|
//
|
|
// State of the connection, used to keep track of shutdowns for TCP/IP.
|
|
// Reset after a recv() call so that the next send operation will
|
|
// check for shutdowns.
|
|
//
|
|
BOOL fCallStarted;
|
|
//
|
|
// True if we received a shutdown packet during the last send operation.
|
|
//
|
|
BOOL fShutdownReceived;
|
|
//
|
|
// True if we posted a receive to check for a shutdown the receive
|
|
// is (was) still pending.
|
|
//
|
|
BOOL fReceivePending;
|
|
//
|
|
// The time of the last RPC call finished on this connection. This
|
|
// is used to determine if a shutdown check is needed.
|
|
//
|
|
DWORD dwLastCallTime;
|
|
|
|
//
|
|
// Com timeout
|
|
//
|
|
UINT Timeout;
|
|
};
|
|
|
|
typedef WS_CLIENT_CONNECTION *PWS_CCONNECTION;
|
|
|
|
class WS_SAN_CONNECTION : public WS_CONNECTION, public SAN_CONNECTION
|
|
{
|
|
RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return SANReceive(hFile, lpBuffer, nNumberOfBytesToRead, lpNumberOfBytesRead, lpOverlapped);
|
|
}
|
|
|
|
RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return SANSend(hFile, lpBuffer, nNumberOfBytesToWrite, lpNumberOfBytesWritten, lpOverlapped);
|
|
}
|
|
};
|
|
|
|
class WS_SAN_CLIENT_CONNECTION : public WS_CLIENT_CONNECTION, public SAN_CONNECTION
|
|
{
|
|
RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return SANReceive(hFile, lpBuffer, nNumberOfBytesToRead, lpNumberOfBytesRead, lpOverlapped);
|
|
}
|
|
|
|
RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return SANSend(hFile, lpBuffer, nNumberOfBytesToWrite, lpNumberOfBytesWritten, lpOverlapped);
|
|
}
|
|
};
|
|
|
|
#ifdef NETBIOS_ON
|
|
class NB_CONNECTION : public WS_CONNECTION
|
|
{
|
|
public:
|
|
// In order to support the old Netbios (3.1/Dos) style programming
|
|
// interface netbios fragments sent from the client to the server
|
|
// must contain a sequence number. It allowed for in-order delivery
|
|
// of fragments on such a system. While not required today it is still
|
|
// part of the wire protocol. The sequence numbers must be incremented
|
|
// on each fragment sent. The sequence number is reset after each call.
|
|
|
|
// REVIEW: When multiple async calls are outstanding the sequence number
|
|
// will be wrong.
|
|
|
|
ULONG SequenceNumber;
|
|
|
|
// the next members need to be aligned in a way that will ensure that
|
|
// fReceivePending is at the same offset as fReceivePending in
|
|
// WS_CLIENT_CONNECTION. Some functions manipulate them in the same
|
|
// way and we need to make sure they are consistent. This is just
|
|
// documenting an existing code idiosyncracy
|
|
BOOL Reserved[1];
|
|
BOOL fReceivePending;
|
|
|
|
virtual RPC_STATUS ProcessRead(IN DWORD bytes, OUT BUFFER *pBuffer,
|
|
OUT PUINT pBufferLength);
|
|
|
|
};
|
|
typedef NB_CONNECTION *PNB_CONNECTION;
|
|
|
|
class NB_SAN_CONNECTION : public NB_CONNECTION, SAN_CONNECTION
|
|
{
|
|
RPC_STATUS Receive(HANDLE hFile, LPVOID lpBuffer,
|
|
DWORD nNumberOfBytesToRead, LPDWORD lpNumberOfBytesRead,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return SANReceive(hFile, lpBuffer, nNumberOfBytesToRead, lpNumberOfBytesRead, lpOverlapped);
|
|
}
|
|
|
|
RPC_STATUS Send(HANDLE hFile, LPCVOID lpBuffer,
|
|
DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten,
|
|
LPOVERLAPPED lpOverlapped)
|
|
{
|
|
return SANSend(hFile, lpBuffer, nNumberOfBytesToWrite, lpNumberOfBytesWritten, lpOverlapped);
|
|
}
|
|
};
|
|
#endif
|
|
|
|
//
|
|
// Datagram object represent both addresses and connections for
|
|
// datagram. Since all client IO is multiplexed through a single
|
|
// port there are not per client objects.
|
|
//
|
|
// There maybe O(# processors) of these objects allocated for
|
|
// each address (endpoint). This is not very many.
|
|
//
|
|
|
|
struct BASE_DATAGRAM : BASE_ASYNC_OBJECT
|
|
{
|
|
//
|
|
// The endpoint on which requests will be received. This will
|
|
// actually be either a WS_DATAGRAM_ENDPOINT or a MQ_DATAGRAM_ENDPOINT.
|
|
//
|
|
BASE_ADDRESS *pEndpoint;
|
|
};
|
|
|
|
struct WS_DATAGRAM : BASE_DATAGRAM
|
|
{
|
|
// If FALSE then the datagram is available to submit a recv on.
|
|
BOOL Busy;
|
|
|
|
// Size of the address received (ignored but needs to be
|
|
// writeable during the async IO completion.)
|
|
INT cRecvAddr;
|
|
|
|
// Address object to receive the remote/local address of the datagram.
|
|
// The object itself is in the runtime packet object
|
|
DatagramTransportPair *AddressPair;
|
|
|
|
// The IO buffer used to receive the packet.
|
|
WSABUF Packet;
|
|
|
|
// Async IO control information
|
|
BASE_OVERLAPPED Read;
|
|
|
|
// The message used to control the operation
|
|
WSAMSG Msg;
|
|
|
|
// The control information for the message
|
|
char MessageAncillaryData[WSA_CMSG_SPACE(sizeof(in_pktinfo))];
|
|
};
|
|
|
|
typedef WS_DATAGRAM *PWS_DATAGRAM;
|
|
|
|
|
|
#ifdef NCADG_MQ_ON
|
|
struct MQ_DATAGRAM : BASE_DATAGRAM
|
|
{
|
|
// If FALSE then the datagram is available to submit a recv on.
|
|
BOOL Busy;
|
|
|
|
// Size of the address received.
|
|
INT cRecvAddr;
|
|
|
|
// Address object to receive the source address of the datagram.
|
|
MQ_ADDRESS *pAddress;
|
|
|
|
// The IO buffer used to receive the packet.
|
|
ULONG dwPacketSize;
|
|
UCHAR *pPacket;
|
|
|
|
// Async IO control information
|
|
MQ_OVERLAPPED Read;
|
|
};
|
|
|
|
typedef MQ_DATAGRAM *PMQ_DATAGRAM;
|
|
#endif
|
|
|
|
//
|
|
// Finds the async object based on the overlapped structure an io
|
|
// completed on.
|
|
//
|
|
inline PBASE_OVERLAPPED FindOverlapped(LPOVERLAPPED pol)
|
|
{
|
|
return(CONTAINING_RECORD(pol, BASE_OVERLAPPED, ol));
|
|
}
|
|
|
|
inline PREQUEST FindRequest(LPOVERLAPPED pol)
|
|
{
|
|
return (pol) ? (FindOverlapped(pol)->pAsyncObject) : 0;
|
|
}
|
|
|
|
//
|
|
// Connection protocol loaders
|
|
//
|
|
|
|
const RPC_CONNECTION_TRANSPORT *WS_TransportLoad(PROTOCOL_ID);
|
|
|
|
#ifdef NETBIOS_ON
|
|
const RPC_CONNECTION_TRANSPORT *NB_TransportLoad(PROTOCOL_ID);
|
|
#endif
|
|
|
|
const RPC_CONNECTION_TRANSPORT *NMP_TransportLoad();
|
|
|
|
//
|
|
// Datagram protocol loaders
|
|
//
|
|
|
|
const RPC_DATAGRAM_TRANSPORT *DG_TransportLoad(PROTOCOL_ID);
|
|
|
|
//
|
|
// Misc functions
|
|
//
|
|
|
|
extern RPC_STATUS IP_BuildAddressVector(
|
|
OUT NETWORK_ADDRESS_VECTOR **,
|
|
IN ULONG NICFlags,
|
|
IN RPC_CHAR *NetworkAddress OPTIONAL,
|
|
IN WS_ADDRESS *Address OPTIONAL);
|
|
|
|
extern RPC_STATUS CDP_BuildAddressVector(NETWORK_ADDRESS_VECTOR **);
|
|
|
|
extern RPC_ADDRESS_CHANGE_FN * SpxAddressChangeFn;
|
|
|
|
//
|
|
// IP name resolver
|
|
//
|
|
|
|
const int IP_RETAIL_BUFFER_SIZE = 3*0x38;
|
|
const int IP_BUFFER_SIZE = DEBUG_MIN(1, IP_RETAIL_BUFFER_SIZE);
|
|
|
|
typedef enum tagIPVersionToUse
|
|
{
|
|
ipvtuIPv4 = 0,
|
|
ipvtuIPv6,
|
|
ipvtuIPAny
|
|
} IPVersionToUse;
|
|
|
|
typedef enum tagClientOrServer
|
|
{
|
|
cosClient,
|
|
cosServer
|
|
} ClientOrServer;
|
|
|
|
class IP_ADDRESS_RESOLVER
|
|
/*++
|
|
|
|
Class Description:
|
|
|
|
Converts the string address to an IP address.
|
|
|
|
--*/
|
|
{
|
|
public:
|
|
IP_ADDRESS_RESOLVER(
|
|
IN RPC_CHAR *Name,
|
|
IN ClientOrServer cos,
|
|
IN IPVersionToUse IPvToUse
|
|
)
|
|
/*++
|
|
|
|
Arguments:
|
|
|
|
Name - The name (dotted ip address or DNS name) to resolve.
|
|
fUseIPv6 - if TRUE, an IPv6 address will be resolved. If
|
|
FALSE, an IPv4 address will be resolved.
|
|
|
|
--*/
|
|
{
|
|
if (Name && (*Name == 0))
|
|
Name = 0;
|
|
this->Name = Name;
|
|
this->IPvToUse = IPvToUse;
|
|
this->cos = cos;
|
|
AddrInfo = NULL;
|
|
LoopbacksReturned = 0;
|
|
CurrentAddrInfo = NULL;
|
|
RpcpMemorySet(&Hint, 0, sizeof(ADDRINFO));
|
|
}
|
|
|
|
inline RPC_STATUS
|
|
NextAddress(
|
|
OUT SOCKADDR_IN *pAddress
|
|
)
|
|
{
|
|
ASSERT(IPvToUse == ipvtuIPv4);
|
|
return NextAddress((SOCKADDR_STORAGE *)pAddress);
|
|
}
|
|
|
|
RPC_STATUS
|
|
NextAddress(
|
|
OUT SOCKADDR_STORAGE *pAddress
|
|
);
|
|
|
|
~IP_ADDRESS_RESOLVER();
|
|
|
|
private:
|
|
RPC_CHAR *Name;
|
|
IPVersionToUse IPvToUse;
|
|
ADDRINFO *AddrInfo; // the start of the addr info list. NULL if enumeration hasn't
|
|
// started
|
|
ADDRINFO *CurrentAddrInfo; // the current element in the addr info list enumeration. NULL
|
|
// if enumeration hasn't started or has finished.
|
|
ClientOrServer cos;
|
|
ADDRINFO Hint;
|
|
int LoopbacksReturned;
|
|
};
|
|
|
|
//
|
|
// Common functions exported by each protocol
|
|
//
|
|
|
|
extern RPC_STATUS RPC_ENTRY
|
|
COMMON_ProcessCalls(
|
|
IN INT Timeout,
|
|
OUT RPC_TRANSPORT_EVENT *pEvent,
|
|
OUT RPC_STATUS *pEventStatus,
|
|
OUT PVOID *ppEventContext,
|
|
OUT UINT *pBufferLength,
|
|
OUT BUFFER *pBuffer,
|
|
OUT PVOID *ppSourceContext
|
|
);
|
|
|
|
extern RPC_STATUS
|
|
COMMON_PostNonIoEvent(
|
|
RPC_TRANSPORT_EVENT Event,
|
|
DWORD Type,
|
|
PVOID Context
|
|
);
|
|
|
|
extern void
|
|
COMMON_RemoveAddress (
|
|
IN BASE_ADDRESS *Address
|
|
);
|
|
|
|
extern RPC_STATUS RPC_ENTRY
|
|
COMMON_TowerConstruct(
|
|
IN PCHAR Protseq,
|
|
IN PCHAR NetworkAddress,
|
|
IN PCHAR Endpoint,
|
|
OUT PUSHORT Floors,
|
|
OUT PULONG ByteCount,
|
|
OUT PUCHAR *Tower
|
|
);
|
|
|
|
extern RPC_STATUS RPC_ENTRY
|
|
COMMON_TowerExplode(
|
|
IN PUCHAR Tower,
|
|
OUT PCHAR *Protseq,
|
|
OUT PCHAR *NetworkAddress,
|
|
OUT PCHAR *Endpoint
|
|
);
|
|
|
|
extern VOID RPC_ENTRY
|
|
COMMON_ServerCompleteListen(
|
|
IN RPC_TRANSPORT_ADDRESS
|
|
);
|
|
|
|
#ifndef NO_PLUG_AND_PLAY
|
|
|
|
extern VOID RPC_ENTRY
|
|
COMMON_ListenForPNPNotifications (
|
|
);
|
|
|
|
extern VOID RPC_ENTRY
|
|
COMMON_StartPnpNotifications (
|
|
);
|
|
|
|
#endif
|
|
|
|
// Internal to transport interface
|
|
|
|
extern RPC_STATUS
|
|
COMMON_PrepareNewHandle(
|
|
IN HANDLE hAdd
|
|
);
|
|
|
|
extern VOID
|
|
COMMON_AddressManager(
|
|
IN BASE_ADDRESS *
|
|
);
|
|
|
|
extern RPC_STATUS RPC_ENTRY
|
|
WS_Abort(
|
|
IN RPC_TRANSPORT_CONNECTION Connection
|
|
);
|
|
|
|
extern RPC_STATUS
|
|
WS_ReactivateAddress (
|
|
IN WS_ADDRESS *pAddress
|
|
);
|
|
|
|
extern VOID
|
|
WS_DeactivateAddress (
|
|
IN WS_ADDRESS *pAddress
|
|
);
|
|
|
|
extern RPC_STATUS
|
|
DG_ReactivateAddress (
|
|
IN WS_DATAGRAM_ENDPOINT *pAddress
|
|
);
|
|
|
|
extern VOID
|
|
DG_DeactivateAddress (
|
|
IN WS_DATAGRAM_ENDPOINT *pAddress
|
|
);
|
|
|
|
|
|
inline
|
|
void
|
|
TransConnectionFreePacket(
|
|
RPC_TRANSPORT_CONNECTION ThisConnection,
|
|
BUFFER Ptr
|
|
)
|
|
{
|
|
I_RpcTransConnectionFreePacket(ThisConnection, Ptr);
|
|
}
|
|
|
|
inline
|
|
BUFFER
|
|
TransConnectionAllocatePacket(
|
|
RPC_TRANSPORT_CONNECTION ThisConnection,
|
|
UINT Size
|
|
)
|
|
{
|
|
return (I_RpcTransConnectionAllocatePacket(ThisConnection, Size));
|
|
}
|
|
|
|
inline
|
|
RPC_STATUS
|
|
TransConnectionReallocPacket(
|
|
IN RPC_TRANSPORT_CONNECTION ThisConnection,
|
|
IN BUFFER *ppBuffer,
|
|
IN UINT OldSize,
|
|
IN UINT NewSize
|
|
)
|
|
{
|
|
return(I_RpcTransConnectionReallocPacket(ThisConnection, ppBuffer, OldSize, NewSize));
|
|
}
|
|
|
|
#define InitReadEvent(p) \
|
|
hEvent = I_RpcTransGetThreadEvent(); \
|
|
\
|
|
p->Read.ol.hEvent = (HANDLE)((ULONG_PTR)hEvent | 0x1)
|
|
|
|
#define ASSERT_READ_EVENT_IS_THERE(p) \
|
|
ASSERT( p->Read.ol.hEvent == (HANDLE) ((ULONG_PTR)I_RpcTransGetThreadEvent() | 0x1))
|
|
|
|
extern HMODULE hWinsock2;
|
|
|
|
#endif // __TRANS_HXX
|