2020-09-30 16:53:55 +02:00

655 lines
20 KiB
C++

//----------------------------------------------------------------------------
//
// Remoting support.
//
// Copyright (C) Microsoft Corporation, 1999-2002.
//
//----------------------------------------------------------------------------
#ifndef __DBGRPC_HPP__
#define __DBGRPC_HPP__
#include <wincrypt.h>
#include <security.h>
#ifdef _WIN32_WCE
#include <winsock.h>
#else
#include <winsock2.h>
#include <schannel.h>
#endif
#include <pparse.hpp>
#include <portio.h>
#define DEBUG_SERVER_KEY "Software\\Microsoft\\Debug Engine\\Servers"
#define INT_ALIGN2(Val, Pow2) \
(((Val) + (Pow2) - 1) & ~((Pow2) - 1))
#define PTR_ALIGN2(Type, Ptr, Pow2) \
((Type)INT_ALIGN2((ULONG64)(Ptr), Pow2))
#define DBGRPC_MAX_IDENTITY 128
typedef ULONG64 DbgRpcObjectId;
//
// Stub functions are indexed out of per-interface tables.
// The function indices are encoded as part interface index
// and part method index.
//
#define DBGRPC_STUB_INDEX_INTERFACE_SHIFT 8
#define DBGRPC_STUB_INDEX_INTERFACE_MAX \
((1 << (8 * sizeof(USHORT) - DBGRPC_STUB_INDEX_INTERFACE_SHIFT)) - 1)
#define DBGRPC_STUB_INDEX_METHOD_MAX \
((1 << DBGRPC_STUB_INDEX_INTERFACE_SHIFT) - 1)
#define DBGRPC_STUB_INDEX(Interface, Method) \
((USHORT)(((Interface) << DBGRPC_STUB_INDEX_INTERFACE_SHIFT) | (Method)))
#define DBGRPC_STUB_INDEX_INTERFACE(StubIndex) \
((StubIndex) >> DBGRPC_STUB_INDEX_INTERFACE_SHIFT)
#define DBGRPC_STUB_INDEX_METHOD(StubIndex) \
((StubIndex) & DBGRPC_STUB_INDEX_METHOD_MAX)
//
// Interface indices for stub indices are given here
// rather than generated as they must stay constant
// for compatibility.
//
// IMPORTANT: New interfaces must be added at the end of
// the section for that header. New headers must be
// well separated from each other to allow expansion.
//
enum
{
// The first dbgeng interface must always be zero.
DBGRPC_SIF_IDebugAdvanced,
DBGRPC_SIF_IDebugBreakpoint,
DBGRPC_SIF_IDebugClient,
DBGRPC_SIF_IDebugControl,
DBGRPC_SIF_IDebugDataSpaces,
DBGRPC_SIF_IDebugEventCallbacks,
DBGRPC_SIF_IDebugInputCallbacks,
DBGRPC_SIF_IDebugOutputCallbacks,
DBGRPC_SIF_IDebugRegisters,
DBGRPC_SIF_IDebugSymbolGroup,
DBGRPC_SIF_IDebugSymbols,
DBGRPC_SIF_IDebugSystemObjects,
DBGRPC_SIF_IDebugClient2,
DBGRPC_SIF_IDebugControl2,
DBGRPC_SIF_IDebugDataSpaces2,
DBGRPC_SIF_IDebugSymbols2,
DBGRPC_SIF_IDebugSystemObjects2,
DBGRPC_SIF_IDebugClient3,
DBGRPC_SIF_IDebugSystemObjects3,
DBGRPC_SIF_IDebugControl3,
DBGRPC_SIF_IDebugDataSpaces3,
DBGRPC_SIF_IDebugClient4,
// Add new dbgeng interfaces here.
DBGRPC_SIF_DBGENG_AFTER_LAST,
DBGRPC_SIF_IUserDebugServices = 192,
// Add new dbgsvc interfaces here.
DBGRPC_SIF_DBGSVC_AFTER_LAST,
};
#define DBGRPC_SIF_DBGENG_FIRST 0
#define DBGRPC_SIF_DBGENG_LAST (DBGRPC_SIF_DBGENG_AFTER_LAST - 1)
#define DBGRPC_SIF_DBGSVC_FIRST DBGRPC_SIF_IUserDebugServices
#define DBGRPC_SIF_DBGSVC_LAST (DBGRPC_SIF_DBGSVC_AFTER_LAST - 1)
#define DBGRPC_RETURN 0x0001
#define DBGRPC_NO_RETURN 0x0002
#define DBGRPC_LOCKED 0x0004
struct DbgRpcCall
{
DbgRpcObjectId ObjectId;
USHORT StubIndex;
USHORT Flags;
ULONG InSize;
ULONG OutSize;
HRESULT Status;
ULONG Sequence;
ULONG Reserved1;
};
//
// These functions and tables are automatically generated.
//
typedef HRESULT (*DbgRpcStubFunction)
(IUnknown* If, class DbgRpcConnection* Conn, DbgRpcCall* Call,
PUCHAR InData, PUCHAR OutData);
struct DbgRpcStubFunctionTable
{
DbgRpcStubFunction* Functions;
ULONG Count;
};
// These functions are provided by a caller of dbgrpc with
// implementations specific to the caller.
void DbgRpcInitializeClient(void);
DbgRpcStubFunction DbgRpcGetStub(USHORT StubIndex);
#if DBG
PCSTR DbgRpcGetStubName(USHORT StubIndex);
#endif
HRESULT DbgRpcPreallocProxy(REFIID InterfaceId, PVOID* Interface,
class DbgRpcProxy** Proxy, PULONG IfUnique);
void DbgRpcDeleteProxy(class DbgRpcProxy* Proxy);
HRESULT DbgRpcServerThreadInitialize(void);
void DbgRpcServerThreadUninitialize(void);
void DbgRpcError(char* Format, ...);
//----------------------------------------------------------------------------
//
// DbgRpcTransport.
//
//----------------------------------------------------------------------------
#define MAX_SERVER_NAME MAX_PARAM_VALUE
#define MAX_PASSWORD_BUFFER 32
enum
{
TRANS_TCP,
TRANS_NPIPE,
TRANS_SSL,
TRANS_SPIPE,
TRANS_1394,
TRANS_COM,
TRANS_COUNT
};
extern PCSTR g_DbgRpcTransportNames[TRANS_COUNT];
class DbgRpcTransport : public ParameterStringParser
{
public:
DbgRpcTransport(void)
{
m_ServerName[0] = 0;
m_PasswordGiven = FALSE;
m_Hidden = FALSE;
m_ClientConnect = FALSE;
m_ClientConnectAttempts = 0;
}
virtual ~DbgRpcTransport(void);
virtual ULONG GetNumberParameters(void);
virtual void GetParameter(ULONG Index,
PSTR Name, ULONG NameSize,
PSTR Value, ULONG ValueSize);
virtual void ResetParameters(void);
virtual BOOL SetParameter(PCSTR Name, PCSTR Value);
virtual DbgRpcTransport* Clone(void) = 0;
virtual HRESULT CreateServer(void) = 0;
virtual HRESULT AcceptConnection(DbgRpcTransport** ClientTrans,
PSTR Identity, ULONG IdentitySize) = 0;
virtual HRESULT ConnectServer(void) = 0;
virtual ULONG Read(ULONG Seq, PVOID Buffer, ULONG Len) = 0;
virtual ULONG Write(ULONG Seq, PVOID Buffer, ULONG Len) = 0;
void CloneData(DbgRpcTransport* Trans);
char m_ServerName[MAX_SERVER_NAME];
BOOL m_PasswordGiven;
BOOL m_Hidden;
BOOL m_ClientConnect;
ULONG m_ClientConnectAttempts;
UCHAR m_HashedPassword[MAX_PASSWORD_BUFFER];
};
class DbgRpcTcpTransport : public DbgRpcTransport
{
public:
DbgRpcTcpTransport(void);
virtual ~DbgRpcTcpTransport(void);
// DbgRpcTransport.
virtual ULONG GetNumberParameters(void);
virtual void GetParameter(ULONG Index,
PSTR Name, ULONG NameSize,
PSTR Value, ULONG ValueSize);
virtual void ResetParameters(void);
virtual BOOL SetParameter(PCSTR Name, PCSTR Value);
virtual DbgRpcTransport* Clone(void);
virtual HRESULT CreateServer(void);
virtual HRESULT AcceptConnection(DbgRpcTransport** ClientTrans,
PSTR Identity, ULONG IdentitySize);
virtual HRESULT ConnectServer(void);
virtual ULONG Read(ULONG Seq, PVOID Buffer, ULONG Len);
virtual ULONG Write(ULONG Seq, PVOID Buffer, ULONG Len);
HRESULT InitOl(void);
void GetAddressIdentity(PSOCKADDR_STORAGE Addr,
int AddrLength,
PSTR Identity,
ULONG IdentitySize);
HRESULT CreateServerSocket(void);
HRESULT AcceptSocketConnection(SOCKET ServSock);
HRESULT ConnectSocket(void);
SOCKADDR_STORAGE m_Addr;
int m_AddrLength;
SOCKET m_Sock;
WSAOVERLAPPED m_OlRead, m_OlWrite;
ULONG m_TopPort;
char m_ClientConnectName[MAX_SERVER_NAME];
SOCKADDR_STORAGE m_ClientConnectAddr;
int m_ClientConnectAddrLength;
};
#ifndef _WIN32_WCE
class DbgRpcNamedPipeTransport : public DbgRpcTransport
{
public:
DbgRpcNamedPipeTransport(void)
{
m_Name = g_DbgRpcTransportNames[TRANS_NPIPE];
m_Handle = NULL;
ZeroMemory(&m_ReadOlap, sizeof(m_ReadOlap));
ZeroMemory(&m_WriteOlap, sizeof(m_WriteOlap));
}
virtual ~DbgRpcNamedPipeTransport(void);
// DbgRpcTransport.
virtual ULONG GetNumberParameters(void);
virtual void GetParameter(ULONG Index,
PSTR Name, ULONG NameSize,
PSTR Value, ULONG ValueSize);
virtual void ResetParameters(void);
virtual BOOL SetParameter(PCSTR Name, PCSTR Value);
virtual DbgRpcTransport* Clone(void);
virtual HRESULT CreateServer(void);
virtual HRESULT AcceptConnection(DbgRpcTransport** ClientTrans,
PSTR Identity, ULONG IdentitySize);
virtual HRESULT ConnectServer(void);
virtual ULONG Read(ULONG Seq, PVOID Buffer, ULONG Len);
virtual ULONG Write(ULONG Seq, PVOID Buffer, ULONG Len);
char m_Pipe[MAX_PARAM_VALUE];
HANDLE m_Handle;
OVERLAPPED m_ReadOlap, m_WriteOlap;
};
// This class is a generic schannel-based wrapper for
// a normal transport.
#define DBGRPC_SCHAN_BUFFER 16384
class DbgRpcSecureChannelTransport : public DbgRpcTransport
{
public:
DbgRpcSecureChannelTransport(ULONG ThisTransport,
ULONG BaseTransport);
virtual ~DbgRpcSecureChannelTransport(void);
// DbgRpcTransport.
virtual ULONG GetNumberParameters(void);
virtual void GetParameter(ULONG Index,
PSTR Name, ULONG NameSize,
PSTR Value, ULONG ValueSize);
virtual void ResetParameters(void);
virtual BOOL SetParameter(PCSTR Name, PCSTR Value);
virtual DbgRpcTransport* Clone(void);
virtual HRESULT CreateServer(void);
virtual HRESULT AcceptConnection(DbgRpcTransport** ClientTrans,
PSTR Identity, ULONG IdentitySize);
virtual HRESULT ConnectServer(void);
virtual ULONG Read(ULONG Seq, PVOID Buffer, ULONG Len);
virtual ULONG Write(ULONG Seq, PVOID Buffer, ULONG Len);
// DbgRpcSecureChannelTransport.
HRESULT GetSizes(void);
HRESULT AuthenticateClientConnection(void);
HRESULT InitiateServerConnection(LPSTR pszServerName);
HRESULT AuthenticateServerConnection(void);
void GetNewClientCredentials(void);
void DisconnectFromClient(void);
void DisconnectFromServer(void);
ULONG StreamRead(ULONG Seq, PVOID Buffer, ULONG MaxSize)
{
ULONG Size;
if (m_Stream->Read(Seq, &Size, sizeof(Size)) != sizeof(Size) ||
Size > MaxSize)
{
return 0;
}
return m_Stream->Read(Seq, Buffer, Size);
}
ULONG StreamWrite(ULONG Seq, PVOID Buffer, ULONG Size)
{
if (m_Stream->Write(Seq, &Size, sizeof(Size)) != sizeof(Size))
{
return 0;
}
return m_Stream->Write(Seq, Buffer, Size);
}
ULONG m_ThisTransport;
ULONG m_BaseTransport;
DbgRpcTransport* m_Stream;
SCHANNEL_CRED m_ScCreds;
CredHandle m_Creds;
BOOL m_OwnCreds;
CtxtHandle m_Context;
BOOL m_OwnContext;
ULONG m_Protocol;
char m_User[64];
BOOL m_MachineStore;
UCHAR m_Buffer[DBGRPC_SCHAN_BUFFER];
ULONG m_BufferUsed;
SecPkgContext_StreamSizes m_Sizes;
ULONG m_MaxChunk;
BOOL m_Server;
};
class DbgRpc1394Transport : public DbgRpcTransport
{
public:
DbgRpc1394Transport(void)
{
m_Name = g_DbgRpcTransportNames[TRANS_1394];
m_Handle = NULL;
}
virtual ~DbgRpc1394Transport(void);
// DbgRpcTransport.
virtual ULONG GetNumberParameters(void);
virtual void GetParameter(ULONG Index,
PSTR Name, ULONG NameSize,
PSTR Value, ULONG ValueSize);
virtual void ResetParameters(void);
virtual BOOL SetParameter(PCSTR Name, PCSTR Value);
virtual DbgRpcTransport* Clone(void);
virtual HRESULT CreateServer(void);
virtual HRESULT AcceptConnection(DbgRpcTransport** ClientTrans,
PSTR Identity, ULONG IdentitySize);
virtual HRESULT ConnectServer(void);
virtual ULONG Read(ULONG Seq, PVOID Buffer, ULONG Len);
virtual ULONG Write(ULONG Seq, PVOID Buffer, ULONG Len);
ULONG m_AcceptChannel;
ULONG m_StreamChannel;
HANDLE m_Handle;
};
class DbgRpcComTransport : public DbgRpcTransport
{
public:
DbgRpcComTransport(void)
{
m_Name = g_DbgRpcTransportNames[TRANS_COM];
m_Handle = NULL;
ZeroMemory(&m_ReadOlap, sizeof(m_ReadOlap));
ZeroMemory(&m_WriteOlap, sizeof(m_WriteOlap));
m_Timeout = INFINITE;
}
virtual ~DbgRpcComTransport(void);
// DbgRpcTransport.
virtual ULONG GetNumberParameters(void);
virtual void GetParameter(ULONG Index,
PSTR Name, ULONG NameSize,
PSTR Value, ULONG ValueSize);
virtual void ResetParameters(void);
virtual BOOL SetParameter(PCSTR Name, PCSTR Value);
virtual DbgRpcTransport* Clone(void);
virtual HRESULT CreateServer(void);
virtual HRESULT AcceptConnection(DbgRpcTransport** ClientTrans,
PSTR Identity, ULONG IdentitySize);
virtual HRESULT ConnectServer(void);
virtual ULONG Read(ULONG Seq, PVOID Buffer, ULONG Len);
virtual ULONG Write(ULONG Seq, PVOID Buffer, ULONG Len);
USHORT ScanQueue(UCHAR Chan, PVOID Buffer, USHORT Len);
USHORT ScanPort(UCHAR Chan, PVOID Buffer, USHORT Len,
BOOL ScanForAck, UCHAR AckChan);
USHORT ChanRead(UCHAR Chan, PVOID Buffer, USHORT Len);
USHORT ChanWrite(UCHAR Chan, PVOID Buffer, USHORT Len);
char m_PortName[MAX_PARAM_VALUE];
ULONG m_BaudRate;
UCHAR m_AcceptChannel;
UCHAR m_StreamChannel;
HANDLE m_Handle;
OVERLAPPED m_ReadOlap, m_WriteOlap;
COM_PORT_TYPE m_PortType;
ULONG m_Timeout;
static HRESULT InitializeChannels(void);
static BOOL s_ChanInitialized;
static CRITICAL_SECTION s_QueueLock;
static HANDLE s_QueueChangedEvent;
static LONG s_PortReadOwned;
static CRITICAL_SECTION s_PortWriteLock;
static CRITICAL_SECTION s_WriteAckLock;
static HANDLE s_WriteAckEvent;
static struct DbgRpcComQueue* s_QueueHead;
static struct DbgRpcComQueue* s_QueueTail;
};
#endif // #ifndef _WIN32_WCE
//----------------------------------------------------------------------------
//
// DbgRpcConnection.
//
//----------------------------------------------------------------------------
// Special value indicating no data was actually allocated.
// NULL is not used to make it easy to catch access.
#define DBGRPC_NO_DATA ((PUCHAR)(ULONG64)-1)
#define DBGRPC_CONN_BUFFER_SIZE 4096
#define DBGRPC_CONN_BUFFER_ALIGN 16
#define DBGRPC_CONN_BUFFER_DYNAMIC_LIMIT 1024
#define DBGRPC_IN_ASYNC_CALL 0x00000001
#define DBGRPC_FULL_REMOTE_UNKNOWN 0x00000002
class DbgRpcConnection
{
public:
DbgRpcConnection(class DbgRpcTransport* Trans);
~DbgRpcConnection(void);
PUCHAR StartCall(DbgRpcCall* Call, DbgRpcObjectId ObjectId,
ULONG StubIndex, ULONG InSize, ULONG OutSize);
HRESULT SendReceive(DbgRpcCall* Call, PUCHAR* InOutData);
void FreeData(PUCHAR Data)
{
if (Data != NULL && Data != DBGRPC_NO_DATA)
{
Free(Data);
}
}
PVOID MallocAligned(ULONG Size);
void FreeAligned(PVOID Ptr);
PVOID Alloc(ULONG Size);
void Free(PVOID Ptr);
void Disconnect(void);
class DbgRpcTransport* m_Trans;
DbgRpcConnection* m_Next;
ULONG m_ThreadId;
UCHAR m_UnalignedBuffer[DBGRPC_CONN_BUFFER_SIZE +
DBGRPC_CONN_BUFFER_ALIGN];
PUCHAR m_Buffer;
ULONG m_BufferUsed;
ULONG m_Flags;
ULONG m_Objects;
};
//----------------------------------------------------------------------------
//
// DbgRpcProxy.
//
//----------------------------------------------------------------------------
class DbgRpcProxy
{
public:
DbgRpcProxy(ULONG InterfaceIndex);
~DbgRpcProxy(void);
IUnknown* InitializeProxy(DbgRpcConnection* Conn,
DbgRpcObjectId ObjectId,
IUnknown* ExistingProxy);
DbgRpcConnection* m_Conn;
DbgRpcObjectId m_ObjectId;
ULONG m_InterfaceIndex;
ULONG m_OwningThread;
ULONG m_LocalRefs, m_RemoteRefs;
};
//----------------------------------------------------------------------------
//
// DbgRpcClientObject.
//
//----------------------------------------------------------------------------
class DbgRpcClientObject
{
public:
virtual HRESULT RpcInitialize(PSTR ClientIdentity, PSTR TransIdentity,
PVOID* Interface) = 0;
// Base implementation does nothing.
virtual void RpcFinalize(void);
virtual void RpcUninitialize(void) = 0;
};
//----------------------------------------------------------------------------
//
// DbgRpcClientObjectFactory.
//
//----------------------------------------------------------------------------
class DbgRpcClientObjectFactory
{
public:
virtual HRESULT CreateInstance(const GUID* DesiredObject,
DbgRpcClientObject** Object) = 0;
virtual void GetServerTypeName(PSTR Buffer, ULONG BufferSize) = 0;
};
#define DBGRPC_SIMPLE_FACTORY(Class, Guid, Name, CtorArgs) \
class Class##Factory : public DbgRpcClientObjectFactory \
{ \
public: \
virtual HRESULT CreateInstance(const GUID* DesiredObject, \
DbgRpcClientObject** Object); \
virtual void GetServerTypeName(PSTR Buffer, ULONG BufferSize); \
}; \
HRESULT \
Class##Factory::CreateInstance(const GUID* DesiredObject, \
DbgRpcClientObject** Object) \
{ \
if (DbgIsEqualIID(Guid, *DesiredObject)) \
{ \
*Object = (DbgRpcClientObject*)new Class CtorArgs; \
return *Object != NULL ? S_OK : E_OUTOFMEMORY; \
} \
else \
{ \
return E_NOINTERFACE; \
} \
} \
void \
Class##Factory::GetServerTypeName(PSTR Buffer, ULONG BufferSize) \
{ \
CopyString(Buffer, Name, BufferSize); \
}
//----------------------------------------------------------------------------
//
// Functions.
//
//----------------------------------------------------------------------------
extern CRITICAL_SECTION g_DbgRpcLock;
DbgRpcTransport*
DbgRpcNewTransport(ULONG Trans);
DbgRpcTransport*
DbgRpcInitializeTransport(PCSTR Options);
PVOID
DbgRpcEnumActiveServers(PVOID Cookie,
PULONG Id, PSTR Buffer, ULONG BufferSize);
HRESULT
DbgRpcDisableServer(ULONG Id);
void
DbgRpcDeregisterServers(void);
HRESULT
DbgRpcCreateServerConnection(DbgRpcTransport* Trans,
const GUID* DesiredObject,
IUnknown** ClientObject);
HRESULT
DbgRpcCreateServer(PCSTR Options, DbgRpcClientObjectFactory* Factory,
BOOL Wait);
HRESULT
DbgRpcConnectServer(PCSTR Options, const GUID* DesiredObject,
IUnknown** ClientObject);
DbgRpcConnection*
DbgRpcGetConnection(ULONG ThreadId);
#define DRPC_ERR(Args) g_NtDllCalls.DbgPrint Args
#if 0
#define DBG_RPC
#define DRPC(Args) g_NtDllCalls.DbgPrint Args
#else
#define DRPC(Args)
#endif
#if 0
#define DRPC_REF(Args) g_NtDllCalls.DbgPrint Args
#else
#define DRPC_REF(Args)
#endif
#endif // #ifndef __DBGRPC_HPP__