/** * FreeRDP: A Remote Desktop Protocol Implementation * RPC over HTTP * * Copyright 2012 Fujitsu Technology Solutions GmbH * Copyright 2012 Dmitrij Jasnov * Copyright 2012 Marc-Andre Moreau * * 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 typedef struct rdp_rpc rdpRpc; typedef struct rdp_ntlm_http rdpNtlmHttp; #include "tcp.h" #include "rts.h" #include "http.h" #include "transport.h" #include #include #include #include #include #include #include #include #include #include #include #include /** * CAE Specification * DCE 1.1: Remote Procedure Call * Document Number: C706 * http://pubs.opengroup.org/onlinepubs/9629399/ */ #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 struct _rpc_pdu_header { DEFINE_RPC_COMMON_FIELDS(); }; typedef struct _rpc_pdu_header RPC_PDU_HEADER; 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 { /* 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 UID */ /* 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; struct rdp_ntlm { CtxtHandle context; ULONG cbMaxToken; ULONG fContextReq; ULONG pfContextAttr; TimeStamp expiration; PSecBuffer pBuffer; SecBuffer inputBuffer; SecBuffer outputBuffer; BOOL haveContext; BOOL haveInputBuffer; LPTSTR ServicePrincipalName; SecBufferDesc inputBufferDesc; SecBufferDesc outputBufferDesc; CredHandle credentials; BOOL confidentiality; SecPkgInfo* pPackageInfo; SecurityFunctionTable* table; SEC_WINNT_AUTH_IDENTITY identity; SecPkgContext_Sizes ContextSizes; }; typedef struct rdp_ntlm rdpNtlm; enum _TSG_CHANNEL { TSG_CHANNEL_IN, TSG_CHANNEL_OUT }; typedef enum _TSG_CHANNEL TSG_CHANNEL; struct rdp_ntlm_http { rdpNtlm* ntlm; HttpContext* context; }; /* Ping Originator */ struct rpc_ping_originator { UINT32 ConnectionTimeout; UINT32 LastPacketSentTimestamp; UINT32 KeepAliveInterval; }; typedef struct rpc_ping_originator RpcPingOriginator; /* Client In Channel */ struct rpc_in_channel { /* Sending Channel */ UINT32 PlugState; void* SendQueue; UINT32 BytesSent; UINT32 SenderAvailableWindow; UINT32 PeerReceiveWindow; /* Ping Originator */ RpcPingOriginator PingOriginator; }; typedef struct rpc_in_channel RpcInChannel; /* Client Out Channel */ struct rpc_out_channel { /* Receiving Channel */ 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; struct rdp_rpc { rdpTls* tls_in; rdpTls* tls_out; rdpNtlm* ntlm; int send_seq_num; rdpNtlmHttp* ntlm_http_in; rdpNtlmHttp* ntlm_http_out; rdpSettings* settings; rdpTransport* transport; UINT32 call_id; UINT32 pipe_call_id; BYTE* buffer; UINT32 length; BYTE rpc_vers; BYTE rpc_vers_minor; BYTE packed_drep[4]; UINT32 ReceiveWindow; RpcVirtualConnection* VirtualConnection; }; BOOL ntlm_authenticate(rdpNtlm* ntlm); BOOL ntlm_client_init(rdpNtlm* ntlm, BOOL confidentiality, char* user, char* domain, char* password); void ntlm_client_uninit(rdpNtlm* ntlm); rdpNtlm* ntlm_new(); void ntlm_free(rdpNtlm* ntlm); BOOL rpc_connect(rdpRpc* rpc); BOOL rpc_ntlm_http_out_connect(rdpRpc* rpc); BOOL rpc_ntlm_http_in_connect(rdpRpc* rpc); void rpc_pdu_header_init(rdpRpc* rpc, RPC_PDU_HEADER* 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); int rpc_recv_pdu(rdpRpc* rpc); int rpc_out_read(rdpRpc* rpc, BYTE* data, int length); int rpc_tsg_write(rdpRpc* rpc, BYTE* data, int length, UINT16 opnum); int rpc_read(rdpRpc* rpc, BYTE* data, int length); 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 */