814 lines
19 KiB
C
814 lines
19 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* RPC over HTTP
|
|
*
|
|
* Copyright 2012 Fujitsu Technology Solutions GmbH
|
|
* Copyright 2012 Dmitrij Jasnov <dmitrij.jasnov@ts.fujitsu.com>
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#ifndef FREERDP_CORE_RPC_H
|
|
#define FREERDP_CORE_RPC_H
|
|
|
|
#include <winpr/wtypes.h>
|
|
#include <winpr/collections.h>
|
|
#include <winpr/interlocked.h>
|
|
|
|
typedef struct rdp_rpc rdpRpc;
|
|
|
|
#define DEFINE_RPC_COMMON_FIELDS() \
|
|
BYTE rpc_vers; \
|
|
BYTE rpc_vers_minor; \
|
|
BYTE ptype; \
|
|
BYTE pfc_flags; \
|
|
BYTE packed_drep[4]; \
|
|
UINT16 frag_length; \
|
|
UINT16 auth_length; \
|
|
UINT32 call_id
|
|
|
|
#define RPC_COMMON_FIELDS_LENGTH 16
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
UINT16 Flags;
|
|
UINT16 NumberOfCommands;
|
|
} rpcconn_rts_hdr_t;
|
|
|
|
#define RTS_PDU_HEADER_LENGTH 20
|
|
|
|
#define RPC_PDU_FLAG_STUB 0x00000001
|
|
|
|
typedef struct _RPC_PDU
|
|
{
|
|
BYTE* Buffer;
|
|
UINT32 Size;
|
|
UINT32 Length;
|
|
DWORD Flags;
|
|
} RPC_PDU, *PRPC_PDU;
|
|
|
|
#include "tcp.h"
|
|
#include "rts.h"
|
|
#include "http.h"
|
|
#include "ntlm.h"
|
|
#include "transport.h"
|
|
|
|
#include <time.h>
|
|
|
|
#include <winpr/sspi.h>
|
|
#include <winpr/interlocked.h>
|
|
|
|
#include <freerdp/types.h>
|
|
#include <freerdp/settings.h>
|
|
#include <freerdp/crypto/tls.h>
|
|
#include <freerdp/crypto/crypto.h>
|
|
#include <freerdp/utils/sleep.h>
|
|
#include <freerdp/utils/debug.h>
|
|
#include <freerdp/utils/stream.h>
|
|
#include <freerdp/utils/hexdump.h>
|
|
|
|
/**
|
|
* CAE Specification
|
|
* DCE 1.1: Remote Procedure Call
|
|
* Document Number: C706
|
|
* http://pubs.opengroup.org/onlinepubs/9629399/
|
|
*/
|
|
|
|
#define PTYPE_REQUEST 0x00
|
|
#define PTYPE_PING 0x01
|
|
#define PTYPE_RESPONSE 0x02
|
|
#define PTYPE_FAULT 0x03
|
|
#define PTYPE_WORKING 0x04
|
|
#define PTYPE_NOCALL 0x05
|
|
#define PTYPE_REJECT 0x06
|
|
#define PTYPE_ACK 0x07
|
|
#define PTYPE_CL_CANCEL 0x08
|
|
#define PTYPE_FACK 0x09
|
|
#define PTYPE_CANCEL_ACK 0x0A
|
|
#define PTYPE_BIND 0x0B
|
|
#define PTYPE_BIND_ACK 0x0C
|
|
#define PTYPE_BIND_NAK 0x0D
|
|
#define PTYPE_ALTER_CONTEXT 0x0E
|
|
#define PTYPE_ALTER_CONTEXT_RESP 0x0F
|
|
#define PTYPE_RPC_AUTH_3 0x10
|
|
#define PTYPE_SHUTDOWN 0x11
|
|
#define PTYPE_CO_CANCEL 0x12
|
|
#define PTYPE_ORPHANED 0x13
|
|
#define PTYPE_RTS 0x14
|
|
|
|
#define PFC_FIRST_FRAG 0x01
|
|
#define PFC_LAST_FRAG 0x02
|
|
#define PFC_PENDING_CANCEL 0x04
|
|
#define PFC_SUPPORT_HEADER_SIGN 0x04
|
|
#define PFC_RESERVED_1 0x08
|
|
#define PFC_CONC_MPX 0x10
|
|
#define PFC_DID_NOT_EXECUTE 0x20
|
|
#define PFC_MAYBE 0x40
|
|
#define PFC_OBJECT_UUID 0x80
|
|
|
|
/* Minimum fragment sizes */
|
|
#define RPC_CO_MUST_RECV_FRAG_SIZE 1432
|
|
#define RPC_CL_MUST_RECV_FRAG_SIZE 1464
|
|
|
|
/**
|
|
* The PDU maximum header length is enough
|
|
* to contain either the RPC common fields
|
|
* or all fields up to the stub data in PDUs
|
|
* that use it (request, response, fault)
|
|
*/
|
|
#define RPC_PDU_HEADER_MAX_LENGTH 32
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
} rpcconn_common_hdr_t;
|
|
|
|
typedef UINT16 p_context_id_t;
|
|
typedef UINT16 p_reject_reason_t;
|
|
|
|
typedef struct
|
|
{
|
|
UINT32 time_low;
|
|
UINT16 time_mid;
|
|
UINT16 time_hi_and_version;
|
|
BYTE clock_seq_hi_and_reserved;
|
|
BYTE clock_seq_low;
|
|
BYTE node[6];
|
|
} p_uuid_t;
|
|
|
|
#define ndr_c_int_big_endian 0
|
|
#define ndr_c_int_little_endian 1
|
|
#define ndr_c_float_ieee 0
|
|
#define ndr_c_float_vax 1
|
|
#define ndr_c_float_cray 2
|
|
#define ndr_c_float_ibm 3
|
|
#define ndr_c_char_ascii 0
|
|
#define ndr_c_char_ebcdic 1
|
|
|
|
typedef struct
|
|
{
|
|
BYTE int_rep;
|
|
BYTE char_rep;
|
|
BYTE float_rep;
|
|
BYTE reserved;
|
|
} ndr_format_t, *ndr_format_p_t;
|
|
|
|
typedef struct ndr_context_handle
|
|
{
|
|
UINT32 context_handle_attributes;
|
|
p_uuid_t context_handle_uuid;
|
|
} ndr_context_handle;
|
|
|
|
typedef struct
|
|
{
|
|
p_uuid_t if_uuid;
|
|
UINT32 if_version;
|
|
} p_syntax_id_t;
|
|
|
|
typedef struct
|
|
{
|
|
p_context_id_t p_cont_id;
|
|
BYTE n_transfer_syn; /* number of items */
|
|
BYTE reserved; /* alignment pad, m.b.z. */
|
|
p_syntax_id_t abstract_syntax; /* transfer syntax list */
|
|
p_syntax_id_t* transfer_syntaxes; /* size_is(n_transfer_syn) */
|
|
} p_cont_elem_t;
|
|
|
|
typedef struct
|
|
{
|
|
BYTE n_context_elem; /* number of items */
|
|
BYTE reserved; /* alignment pad, m.b.z. */
|
|
UINT16 reserved2; /* alignment pad, m.b.z. */
|
|
p_cont_elem_t* p_cont_elem; /* size_is(n_cont_elem) */
|
|
} p_cont_list_t;
|
|
|
|
typedef enum
|
|
{
|
|
acceptance,
|
|
user_rejection,
|
|
provider_rejection
|
|
} p_cont_def_result_t;
|
|
|
|
typedef enum
|
|
{
|
|
reason_not_specified,
|
|
abstract_syntax_not_supported,
|
|
proposed_transfer_syntaxes_not_supported,
|
|
local_limit_exceeded
|
|
} p_provider_reason_t;
|
|
|
|
typedef struct
|
|
{
|
|
p_cont_def_result_t result;
|
|
p_provider_reason_t reason;
|
|
p_syntax_id_t transfer_syntax;
|
|
} p_result_t;
|
|
|
|
/* Same order and number of elements as in bind request */
|
|
|
|
typedef struct
|
|
{
|
|
BYTE n_results; /* count */
|
|
BYTE reserved; /* alignment pad, m.b.z. */
|
|
UINT16 reserved2; /* alignment pad, m.b.z. */
|
|
p_result_t* p_results; /* size_is(n_results) */
|
|
} p_result_list_t;
|
|
|
|
typedef struct
|
|
{
|
|
BYTE major;
|
|
BYTE minor;
|
|
} version_t;
|
|
typedef version_t p_rt_version_t;
|
|
|
|
typedef struct
|
|
{
|
|
BYTE n_protocols; /* count */
|
|
p_rt_version_t* p_protocols; /* size_is(n_protocols) */
|
|
} p_rt_versions_supported_t;
|
|
|
|
typedef struct
|
|
{
|
|
UINT16 length;
|
|
char* port_spec; /* port string spec; size_is(length) */
|
|
} port_any_t;
|
|
|
|
#define REASON_NOT_SPECIFIED 0
|
|
#define TEMPORARY_CONGESTION 1
|
|
#define LOCAL_LIMIT_EXCEEDED 2
|
|
#define CALLED_PADDR_UNKNOWN 3
|
|
#define PROTOCOL_VERSION_NOT_SUPPORTED 4
|
|
#define DEFAULT_CONTEXT_NOT_SUPPORTED 5
|
|
#define USER_DATA_NOT_READABLE 6
|
|
#define NO_PSAP_AVAILABLE 7
|
|
|
|
typedef UINT16 rpcrt_reason_code_t;
|
|
|
|
typedef struct
|
|
{
|
|
BYTE rpc_vers;
|
|
BYTE rpc_vers_minor;
|
|
BYTE reserved[2]; /* must be zero */
|
|
BYTE packed_drep[4];
|
|
UINT32 reject_status;
|
|
BYTE reserved2[4];
|
|
} rpcrt_optional_data_t;
|
|
|
|
typedef struct
|
|
{
|
|
rpcrt_reason_code_t reason_code; /* 0..65535 */
|
|
rpcrt_optional_data_t rpc_info; /* may be RPC specific */
|
|
} rpcconn_reject_optional_data_t;
|
|
|
|
typedef struct
|
|
{
|
|
rpcrt_reason_code_t reason_code; /* 0..65535 */
|
|
rpcrt_optional_data_t rpc_info; /* may be RPC-specific */
|
|
} rpcconn_disc_optional_data_t;
|
|
|
|
typedef struct
|
|
{
|
|
BYTE signature[8];
|
|
} rpc_sec_verification_trailer;
|
|
|
|
typedef struct
|
|
{
|
|
/* restore 4-byte alignment */
|
|
|
|
BYTE* auth_pad; /* align(4); [size_is(auth_pad_length)] */
|
|
|
|
BYTE auth_type; /* :01 which authent service */
|
|
BYTE auth_level; /* :01 which level within service */
|
|
BYTE auth_pad_length; /* :01 */
|
|
BYTE auth_reserved; /* :01 reserved, m.b.z. */
|
|
UINT32 auth_context_id; /* :04 */
|
|
|
|
BYTE* auth_value; /* credentials; [size_is(auth_length)] */
|
|
} auth_verifier_co_t;
|
|
|
|
/* Connection-oriented PDU Definitions */
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
UINT16 max_xmit_frag; /* ignored */
|
|
UINT16 max_recv_frag; /* ignored */
|
|
UINT32 assoc_group_id; /* ignored */
|
|
|
|
/* presentation context list */
|
|
|
|
p_cont_list_t p_context_elem; /* variable size */
|
|
|
|
/* optional authentication verifier */
|
|
|
|
/* following fields present if auth_length != 0 */
|
|
|
|
auth_verifier_co_t auth_verifier;
|
|
|
|
} rpcconn_alter_context_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
UINT16 max_xmit_frag;
|
|
UINT16 max_recv_frag;
|
|
UINT32 assoc_group_id;
|
|
port_any_t sec_addr;
|
|
|
|
/* restore 4-octet alignment */
|
|
|
|
BYTE* pad2; /* size_is(align(4)) */
|
|
|
|
/* presentation context result list, including hints */
|
|
|
|
p_result_list_t p_result_list; /* variable size */
|
|
|
|
/* optional authentication verifier */
|
|
/* following fields present iff auth_length != 0 */
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_alter_context_response_hdr_t;
|
|
|
|
/* bind header */
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
UINT16 max_xmit_frag; /* 16:02 max transmit frag size, bytes */
|
|
UINT16 max_recv_frag; /* 18:02 max receive frag size, bytes */
|
|
UINT32 assoc_group_id; /* 20:04 incarnation of client-server assoc group */
|
|
|
|
/* presentation context list */
|
|
|
|
p_cont_list_t p_context_elem; /* variable size */
|
|
|
|
/* optional authentication verifier */
|
|
/* following fields present if auth_length != 0 */
|
|
|
|
auth_verifier_co_t auth_verifier;
|
|
} rpcconn_bind_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
UINT16 max_xmit_frag; /* 16:02 max transmit frag size */
|
|
UINT16 max_recv_frag; /* 18:02 max receive frag size */
|
|
UINT32 assoc_group_id; /* 20:04 returned assoc_group_id */
|
|
|
|
port_any_t sec_addr; /* 24:yy optional secondary address for process incarnation; local port part of address only */
|
|
|
|
/* restore 4-octet alignment */
|
|
|
|
BYTE* pad2; /* size_is(align(4)) */
|
|
|
|
/* presentation context result list, including hints */
|
|
|
|
p_result_list_t p_result_list; /* variable size */
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_bind_ack_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
UINT16 max_xmit_frag; /* 16:02 max transmit frag size */
|
|
UINT16 max_recv_frag; /* 18:02 max receive frag size */
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_rpc_auth_3_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
p_reject_reason_t provider_reject_reason;
|
|
|
|
p_rt_versions_supported_t versions; /* 18:yy array of protocol versions supported */
|
|
} rpcconn_bind_nak_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
|
|
} rpcconn_cancel_hdr_t;
|
|
|
|
/* fault codes */
|
|
|
|
struct _RPC_FAULT_CODE
|
|
{
|
|
UINT32 code;
|
|
char* name;
|
|
};
|
|
typedef struct _RPC_FAULT_CODE RPC_FAULT_CODE;
|
|
|
|
#define DEFINE_RPC_FAULT_CODE(_code) { _code , #_code },
|
|
|
|
#define nca_s_fault_object_not_found 0x1C000024
|
|
#define nca_s_fault_cancel 0x1C00000D
|
|
#define nca_s_fault_addr_error 0x1C000002
|
|
#define nca_s_fault_context_mismatch 0x1C00001A
|
|
#define nca_s_fault_fp_div_zero 0x1C000003
|
|
#define nca_s_fault_fp_error 0x1C00000F
|
|
#define nca_s_fault_fp_overflow 0x1C000005
|
|
#define nca_s_fault_fp_underflow 0x1C000004
|
|
#define nca_s_fault_ill_inst 0x1C00000E
|
|
#define nca_s_fault_int_div_by_zero 0x1C000001
|
|
#define nca_s_fault_int_overflow 0x1C000010
|
|
#define nca_s_fault_invalid_bound 0x1C000007
|
|
#define nca_s_fault_invalid_tag 0x1C000006
|
|
#define nca_s_fault_pipe_closed 0x1C000015
|
|
#define nca_s_fault_pipe_comm_error 0x1C000018
|
|
#define nca_s_fault_pipe_discipline 0x1C000017
|
|
#define nca_s_fault_pipe_empty 0x1C000014
|
|
#define nca_s_fault_pipe_memory 0x1C000019
|
|
#define nca_s_fault_pipe_order 0x1C000016
|
|
#define nca_s_fault_remote_no_memory 0x1C00001B
|
|
#define nca_s_fault_user_defined 0x1C000021
|
|
#define nca_s_fault_tx_open_failed 0x1C000022
|
|
#define nca_s_fault_codeset_conv_error 0x1C000023
|
|
#define nca_s_fault_no_client_stub 0x1C000025
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
/* needed for request, response, fault */
|
|
|
|
UINT32 alloc_hint; /* 16:04 allocation hint */
|
|
p_context_id_t p_cont_id; /* 20:02 pres context, i.e. data rep */
|
|
|
|
/* needed for response or fault */
|
|
|
|
BYTE cancel_count; /* 22:01 received cancel count */
|
|
BYTE reserved; /* 23:01 reserved, m.b.z. */
|
|
|
|
/* fault code */
|
|
|
|
UINT32 status; /* 24:04 run-time fault code or zero */
|
|
|
|
/* always pad to next 8-octet boundary */
|
|
|
|
BYTE reserved2[4]; /* 28:04 reserved padding, m.b.z. */
|
|
|
|
/* stub data here, 8-octet aligned */
|
|
BYTE* stub_data;
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_fault_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_orphaned_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
/* needed on request, response, fault */
|
|
|
|
UINT32 alloc_hint; /* 16:04 allocation hint */
|
|
|
|
p_context_id_t p_cont_id; /* 20:02 pres context, i.e. data rep */
|
|
UINT16 opnum; /* 22:02 operation number within the interface */
|
|
|
|
/* optional field for request, only present if the PFC_OBJECT_UUID field is non-zero */
|
|
|
|
p_uuid_t object; /* 24:16 object UUID */
|
|
|
|
/* stub data, 8-octet aligned */
|
|
|
|
BYTE* stub_data;
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_request_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
|
|
/* needed for request, response, fault */
|
|
|
|
UINT32 alloc_hint; /* 16:04 allocation hint */
|
|
p_context_id_t p_cont_id; /* 20:02 pres context, i.e. data rep */
|
|
|
|
/* needed for response or fault */
|
|
|
|
BYTE cancel_count; /* 22:01 cancel count */
|
|
BYTE reserved; /* 23:01 reserved, m.b.z. */
|
|
|
|
/* stub data here, 8-octet aligned */
|
|
BYTE* stub_data;
|
|
|
|
auth_verifier_co_t auth_verifier; /* xx:yy */
|
|
} rpcconn_response_hdr_t;
|
|
|
|
typedef struct
|
|
{
|
|
DEFINE_RPC_COMMON_FIELDS();
|
|
} rpcconn_shutdown_hdr_t;
|
|
|
|
typedef union
|
|
{
|
|
rpcconn_common_hdr_t common;
|
|
rpcconn_alter_context_hdr_t alter_context;
|
|
rpcconn_alter_context_response_hdr_t alter_context_response;
|
|
rpcconn_bind_hdr_t bind;
|
|
rpcconn_bind_ack_hdr_t bind_ack;
|
|
rpcconn_rpc_auth_3_hdr_t rpc_auth_3;
|
|
rpcconn_bind_nak_hdr_t bind_nak;
|
|
rpcconn_cancel_hdr_t cancel;
|
|
rpcconn_fault_hdr_t fault;
|
|
rpcconn_orphaned_hdr_t orphaned;
|
|
rpcconn_request_hdr_t request;
|
|
rpcconn_response_hdr_t response;
|
|
rpcconn_shutdown_hdr_t shutdown;
|
|
rpcconn_rts_hdr_t rts;
|
|
} rpcconn_hdr_t;
|
|
|
|
struct _RPC_SECURITY_PROVIDER_INFO
|
|
{
|
|
UINT32 Id;
|
|
LONG EvenLegs;
|
|
LONG NumLegs;
|
|
};
|
|
typedef struct _RPC_SECURITY_PROVIDER_INFO RPC_SECURITY_PROVIDER_INFO;
|
|
|
|
enum _RPC_CLIENT_STATE
|
|
{
|
|
RPC_CLIENT_STATE_INITIAL,
|
|
RPC_CLIENT_STATE_ESTABLISHED,
|
|
RPC_CLIENT_STATE_WAIT_SECURE_BIND_ACK,
|
|
RPC_CLIENT_STATE_WAIT_UNSECURE_BIND_ACK,
|
|
RPC_CLIENT_STATE_WAIT_SECURE_ALTER_CONTEXT_RESPONSE,
|
|
RPC_CLIENT_STATE_CONTEXT_NEGOTIATED,
|
|
RPC_CLIENT_STATE_WAIT_RESPONSE,
|
|
RPC_CLIENT_STATE_FINAL
|
|
};
|
|
typedef enum _RPC_CLIENT_STATE RPC_CLIENT_STATE;
|
|
|
|
enum _RPC_CLIENT_CALL_STATE
|
|
{
|
|
RPC_CLIENT_CALL_STATE_INITIAL,
|
|
RPC_CLIENT_CALL_STATE_SEND_PDUS,
|
|
RPC_CLIENT_CALL_STATE_DISPATCHED,
|
|
RPC_CLIENT_CALL_STATE_RECEIVE_PDU,
|
|
RPC_CLIENT_CALL_STATE_COMPLETE,
|
|
RPC_CLIENT_CALL_STATE_FAULT,
|
|
RPC_CLIENT_CALL_STATE_FINAL
|
|
};
|
|
typedef enum _RPC_CLIENT_CALL_STATE RPC_CLIENT_CALL_STATE;
|
|
|
|
enum _TSG_CHANNEL
|
|
{
|
|
TSG_CHANNEL_IN,
|
|
TSG_CHANNEL_OUT
|
|
};
|
|
typedef enum _TSG_CHANNEL TSG_CHANNEL;
|
|
|
|
/* Ping Originator */
|
|
|
|
struct rpc_ping_originator
|
|
{
|
|
UINT32 ConnectionTimeout;
|
|
UINT32 LastPacketSentTimestamp;
|
|
UINT32 KeepAliveInterval;
|
|
};
|
|
typedef struct rpc_ping_originator RpcPingOriginator;
|
|
|
|
/* Client In Channel */
|
|
|
|
enum _CLIENT_IN_CHANNEL_STATE
|
|
{
|
|
CLIENT_IN_CHANNEL_STATE_INITIAL,
|
|
CLIENT_IN_CHANNEL_STATE_OPENED,
|
|
CLIENT_IN_CHANNEL_STATE_OPENED_A4W,
|
|
CLIENT_IN_CHANNEL_STATE_FINAL
|
|
};
|
|
typedef enum _CLIENT_IN_CHANNEL_STATE CLIENT_IN_CHANNEL_STATE;
|
|
|
|
struct rpc_in_channel
|
|
{
|
|
/* Sending Channel */
|
|
|
|
CLIENT_IN_CHANNEL_STATE State;
|
|
|
|
HANDLE Mutex;
|
|
|
|
UINT32 PlugState;
|
|
void* SendQueue;
|
|
UINT32 BytesSent;
|
|
UINT32 SenderAvailableWindow;
|
|
UINT32 PeerReceiveWindow;
|
|
|
|
/* Ping Originator */
|
|
|
|
RpcPingOriginator PingOriginator;
|
|
};
|
|
typedef struct rpc_in_channel RpcInChannel;
|
|
|
|
/* Client Out Channel */
|
|
|
|
enum _CLIENT_OUT_CHANNEL_STATE
|
|
{
|
|
CLIENT_OUT_CHANNEL_STATE_INITIAL,
|
|
CLIENT_OUT_CHANNEL_STATE_OPENED,
|
|
CLIENT_OUT_CHANNEL_STATE_OPENED_A6W,
|
|
CLIENT_OUT_CHANNEL_STATE_OPENED_A10W,
|
|
CLIENT_OUT_CHANNEL_STATE_OPENED_B3W,
|
|
CLIENT_OUT_CHANNEL_STATE_FINAL
|
|
};
|
|
typedef enum _CLIENT_OUT_CHANNEL_STATE CLIENT_OUT_CHANNEL_STATE;
|
|
|
|
struct rpc_out_channel
|
|
{
|
|
/* Receiving Channel */
|
|
|
|
CLIENT_OUT_CHANNEL_STATE State;
|
|
|
|
HANDLE Mutex;
|
|
|
|
UINT32 ReceiveWindow;
|
|
UINT32 ReceiveWindowSize;
|
|
UINT32 ReceiverAvailableWindow;
|
|
UINT32 BytesReceived;
|
|
UINT32 AvailableWindowAdvertised;
|
|
};
|
|
typedef struct rpc_out_channel RpcOutChannel;
|
|
|
|
/* Client Virtual Connection */
|
|
|
|
enum _VIRTUAL_CONNECTION_STATE
|
|
{
|
|
VIRTUAL_CONNECTION_STATE_INITIAL,
|
|
VIRTUAL_CONNECTION_STATE_OUT_CHANNEL_WAIT,
|
|
VIRTUAL_CONNECTION_STATE_WAIT_A3W,
|
|
VIRTUAL_CONNECTION_STATE_WAIT_C2,
|
|
VIRTUAL_CONNECTION_STATE_OPENED,
|
|
VIRTUAL_CONNECTION_STATE_FINAL
|
|
};
|
|
typedef enum _VIRTUAL_CONNECTION_STATE VIRTUAL_CONNECTION_STATE;
|
|
|
|
struct rpc_virtual_connection
|
|
{
|
|
BYTE Cookie[16]; /* Virtual Connection Cookie */
|
|
VIRTUAL_CONNECTION_STATE State; /* Virtual Connection State */
|
|
RpcInChannel* DefaultInChannel; /* Default IN Channel */
|
|
RpcInChannel* NonDefaultInChannel; /* Non-Default IN Channel */
|
|
BYTE DefaultInChannelCookie[16]; /* Default IN Channel Cookie */
|
|
BYTE NonDefaultInChannelCookie[16]; /* Non-Default Default IN Channel Cookie */
|
|
RpcOutChannel* DefaultOutChannel; /* Default OUT Channel */
|
|
RpcOutChannel* NonDefaultOutChannel; /* Non-Default OUT Channel */
|
|
BYTE DefaultOutChannelCookie[16]; /* Default OUT Channel Cookie */
|
|
BYTE NonDefaultOutChannelCookie[16]; /* Non-Default Default OUT Channel Cookie */
|
|
BYTE AssociationGroupId[16]; /* AssociationGroupId */
|
|
};
|
|
typedef struct rpc_virtual_connection RpcVirtualConnection;
|
|
|
|
/* Virtual Connection Cookie Table */
|
|
|
|
#define RPC_UUID_FORMAT_STRING "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x"
|
|
#define RPC_UUID_FORMAT_ARGUMENTS(_rpc_uuid) \
|
|
_rpc_uuid[0], _rpc_uuid[1], _rpc_uuid[2], _rpc_uuid[3], _rpc_uuid[4], _rpc_uuid[5], _rpc_uuid[6], _rpc_uuid[7], \
|
|
_rpc_uuid[8], _rpc_uuid[9], _rpc_uuid[10], _rpc_uuid[11], _rpc_uuid[12], _rpc_uuid[13], _rpc_uuid[14], _rpc_uuid[15]
|
|
|
|
struct rpc_virtual_connection_cookie_entry
|
|
{
|
|
BYTE Cookie[16];
|
|
UINT32 ReferenceCount;
|
|
RpcVirtualConnection* Reference;
|
|
};
|
|
typedef struct rpc_virtual_connection_cookie_entry RpcVirtualConnectionCookieEntry;
|
|
|
|
struct rpc_virtual_connection_cookie_table
|
|
{
|
|
UINT32 Count;
|
|
UINT32 ArraySize;
|
|
RpcVirtualConnectionCookieEntry* Entries;
|
|
};
|
|
typedef struct rpc_virtual_connection_cookie_table RpcVirtualConnectionCookieTable;
|
|
|
|
struct rpc_client
|
|
{
|
|
HANDLE Thread;
|
|
HANDLE StopEvent;
|
|
|
|
HANDLE PduSentEvent;
|
|
HANDLE SendSemaphore;
|
|
BOOL SynchronousSend;
|
|
|
|
HANDLE PduReceivedEvent;
|
|
HANDLE ReceiveSemaphore;
|
|
BOOL SynchronousReceive;
|
|
};
|
|
typedef struct rpc_client RpcClient;
|
|
|
|
struct rdp_rpc
|
|
{
|
|
RPC_CLIENT_STATE State;
|
|
|
|
rdpTls* TlsIn;
|
|
rdpTls* TlsOut;
|
|
|
|
rdpNtlm* ntlm;
|
|
int send_seq_num;
|
|
|
|
RpcClient* client;
|
|
|
|
rdpNtlmHttp* NtlmHttpIn;
|
|
rdpNtlmHttp* NtlmHttpOut;
|
|
|
|
rdpSettings* settings;
|
|
rdpTransport* transport;
|
|
|
|
UINT32 call_id;
|
|
UINT32 pipe_call_id;
|
|
|
|
RPC_PDU* pdu;
|
|
|
|
BYTE* FragBuffer;
|
|
UINT32 FragBufferSize;
|
|
|
|
BYTE* StubBuffer;
|
|
UINT32 StubBufferSize;
|
|
UINT32 StubLength;
|
|
UINT32 StubOffset;
|
|
UINT32 StubFragCount;
|
|
|
|
BYTE rpc_vers;
|
|
BYTE rpc_vers_minor;
|
|
BYTE packed_drep[4];
|
|
|
|
UINT16 max_xmit_frag;
|
|
UINT16 max_recv_frag;
|
|
|
|
wQueue* SendQueue;
|
|
wQueue* ReceiveQueue;
|
|
|
|
UINT32 ReceiveWindow;
|
|
|
|
UINT32 ChannelLifetime;
|
|
UINT32 ChannelLifetimeSet;
|
|
|
|
UINT32 KeepAliveInterval;
|
|
UINT32 CurrentKeepAliveTime;
|
|
UINT32 CurrentKeepAliveInterval;
|
|
|
|
RpcVirtualConnection* VirtualConnection;
|
|
RpcVirtualConnectionCookieTable* VirtualConnectionCookieTable;
|
|
};
|
|
|
|
BOOL rpc_connect(rdpRpc* rpc);
|
|
|
|
void rpc_pdu_header_init(rdpRpc* rpc, rpcconn_hdr_t* header);
|
|
|
|
UINT32 rpc_offset_align(UINT32* offset, UINT32 alignment);
|
|
UINT32 rpc_offset_pad(UINT32* offset, UINT32 pad);
|
|
|
|
int rpc_out_write(rdpRpc* rpc, BYTE* data, int length);
|
|
int rpc_in_write(rdpRpc* rpc, BYTE* data, int length);
|
|
|
|
BOOL rpc_get_stub_data_info(rdpRpc* rpc, BYTE* header, UINT32* offset, UINT32* length);
|
|
int rpc_recv_pdu_header(rdpRpc* rpc, BYTE* header);
|
|
|
|
RPC_PDU* rpc_recv_pdu(rdpRpc* rpc);
|
|
int rpc_write(rdpRpc* rpc, BYTE* data, int length, UINT16 opnum);
|
|
|
|
int rpc_recv(rdpRpc* rpc, RPC_PDU* pdu);
|
|
|
|
rdpRpc* rpc_new(rdpTransport* transport);
|
|
void rpc_free(rdpRpc* rpc);
|
|
|
|
#ifdef WITH_DEBUG_TSG
|
|
#define WITH_DEBUG_RPC
|
|
#endif
|
|
|
|
#ifdef WITH_DEBUG_RPC
|
|
#define DEBUG_RPC(fmt, ...) DEBUG_CLASS(RPC, fmt, ## __VA_ARGS__)
|
|
#else
|
|
#define DEBUG_RPC(fmt, ...) DEBUG_NULL(fmt, ## __VA_ARGS__)
|
|
#endif
|
|
|
|
#endif /* FREERDP_CORE_RPC_H */
|