libfreerdp-core: improve options for connection cookie

This commit is contained in:
Marc-André Moreau 2012-10-25 18:38:51 -04:00
parent b7a289f843
commit 37b6f9dd8e
5 changed files with 81 additions and 9 deletions

View File

@ -301,7 +301,8 @@ struct rdp_settings
ALIGN64 BOOL send_preconnection_pdu; /* 72 */ ALIGN64 BOOL send_preconnection_pdu; /* 72 */
ALIGN64 UINT32 preconnection_id; /* 73 */ ALIGN64 UINT32 preconnection_id; /* 73 */
ALIGN64 char* preconnection_blob; /* 74 */ ALIGN64 char* preconnection_blob; /* 74 */
UINT64 paddingC[80 - 75]; /* 75 */ ALIGN64 char* computer_name; /* 75 */
UINT64 paddingC[80 - 76]; /* 76 */
/* User Interface Parameters */ /* User Interface Parameters */
ALIGN64 BOOL sw_gdi; /* 80 */ ALIGN64 BOOL sw_gdi; /* 80 */

View File

@ -62,7 +62,7 @@
*/ */
/** /**
* Establish RDP Connection based on the settings given in the 'rdp' paremeter. * Establish RDP Connection based on the settings given in the 'rdp' parameter.
* @msdn{cc240452} * @msdn{cc240452}
* @param rdp RDP module * @param rdp RDP module
* @return true if the connection succeeded. FALSE otherwise. * @return true if the connection succeeded. FALSE otherwise.
@ -74,7 +74,43 @@ BOOL rdp_client_connect(rdpRdp* rdp)
nego_init(rdp->nego); nego_init(rdp->nego);
nego_set_target(rdp->nego, settings->hostname, settings->port); nego_set_target(rdp->nego, settings->hostname, settings->port);
nego_set_cookie(rdp->nego, settings->username);
if (settings->ts_gateway)
{
char* user;
char* domain;
char* cookie;
int user_length;
int domain_length;
int cookie_length;
user = settings->username;
user_length = strlen(settings->username);
if (settings->domain)
domain = settings->domain;
else
domain = settings->computer_name;
domain_length = strlen(domain);
cookie_length = domain_length + 1 + user_length;
cookie = (char*) malloc(cookie_length + 1);
CopyMemory(cookie, domain, domain_length);
CharUpperBuffA(cookie, domain_length);
cookie[domain_length] = '\\';
CopyMemory(&cookie[domain_length + 1], user, user_length);
cookie[cookie_length] = '\0';
nego_set_cookie(rdp->nego, cookie);
//nego_set_cookie_max_length(rdp->nego, MSTSC_COOKIE_MAX_LENGTH);
}
else
{
nego_set_cookie(rdp->nego, settings->username);
}
nego_set_send_preconnection_pdu(rdp->nego, settings->send_preconnection_pdu); nego_set_send_preconnection_pdu(rdp->nego, settings->send_preconnection_pdu);
nego_set_preconnection_id(rdp->nego, settings->preconnection_id); nego_set_preconnection_id(rdp->nego, settings->preconnection_id);
nego_set_preconnection_blob(rdp->nego, settings->preconnection_blob); nego_set_preconnection_blob(rdp->nego, settings->preconnection_blob);

View File

@ -149,7 +149,7 @@ BOOL nego_connect(rdpNego* nego)
/* connect to selected security layer */ /* connect to selected security layer */
BOOL nego_security_connect(rdpNego* nego) BOOL nego_security_connect(rdpNego* nego)
{ {
if(!nego->tcp_connected) if (!nego->tcp_connected)
{ {
nego->security_connected = FALSE; nego->security_connected = FALSE;
} }
@ -160,7 +160,7 @@ BOOL nego_security_connect(rdpNego* nego)
DEBUG_NEGO("nego_security_connect with PROTOCOL_NLA"); DEBUG_NEGO("nego_security_connect with PROTOCOL_NLA");
nego->security_connected = transport_connect_nla(nego->transport); nego->security_connected = transport_connect_nla(nego->transport);
} }
else if (nego->selected_protocol == PROTOCOL_TLS ) else if (nego->selected_protocol == PROTOCOL_TLS)
{ {
DEBUG_NEGO("nego_security_connect with PROTOCOL_TLS"); DEBUG_NEGO("nego_security_connect with PROTOCOL_TLS");
nego->security_connected = transport_connect_tls(nego->transport); nego->security_connected = transport_connect_tls(nego->transport);
@ -175,6 +175,7 @@ BOOL nego_security_connect(rdpNego* nego)
DEBUG_NEGO("cannot connect security layer because no protocol has been selected yet."); DEBUG_NEGO("cannot connect security layer because no protocol has been selected yet.");
} }
} }
return nego->security_connected; return nego->security_connected;
} }
@ -562,6 +563,7 @@ BOOL nego_send_negotiation_request(rdpNego* nego)
STREAM* s; STREAM* s;
int length; int length;
BYTE *bm, *em; BYTE *bm, *em;
int cookie_length;
s = transport_send_stream_init(nego->transport, 256); s = transport_send_stream_init(nego->transport, 256);
length = TPDU_CONNECTION_REQUEST_LENGTH; length = TPDU_CONNECTION_REQUEST_LENGTH;
@ -575,7 +577,11 @@ BOOL nego_send_negotiation_request(rdpNego* nego)
} }
else if (nego->cookie != NULL) else if (nego->cookie != NULL)
{ {
int cookie_length = strlen(nego->cookie); cookie_length = strlen(nego->cookie);
if (cookie_length > nego->cookie_max_length)
cookie_length = nego->cookie_max_length;
stream_write(s, "Cookie: mstshash=", 17); stream_write(s, "Cookie: mstshash=", 17);
stream_write(s, (BYTE*) nego->cookie, cookie_length); stream_write(s, (BYTE*) nego->cookie, cookie_length);
stream_write_BYTE(s, 0x0D); /* CR */ stream_write_BYTE(s, 0x0D); /* CR */
@ -802,6 +808,7 @@ void nego_init(rdpNego* nego)
nego->requested_protocols = PROTOCOL_RDP; nego->requested_protocols = PROTOCOL_RDP;
nego->transport->recv_callback = nego_recv; nego->transport->recv_callback = nego_recv;
nego->transport->recv_extra = (void*) nego; nego->transport->recv_extra = (void*) nego;
nego->cookie_max_length = DEFAULT_COOKIE_MAX_LENGTH;
nego->flags = 0; nego->flags = 0;
} }
@ -919,6 +926,17 @@ void nego_set_cookie(rdpNego* nego, char* cookie)
nego->cookie = cookie; nego->cookie = cookie;
} }
/**
* Set cookie maximum length
* @param nego
* @param cookie_max_length
*/
void nego_set_cookie_max_length(rdpNego* nego, UINT32 cookie_max_length)
{
nego->cookie_max_length = cookie_max_length;
}
/** /**
* Enable / disable preconnection PDU. * Enable / disable preconnection PDU.
* @param nego * @param nego

View File

@ -65,14 +65,17 @@ enum RDP_NEG_MSG
TYPE_RDP_NEG_FAILURE = 0x3 TYPE_RDP_NEG_FAILURE = 0x3
}; };
#define EXTENDED_CLIENT_DATA_SUPPORTED 0x01 #define EXTENDED_CLIENT_DATA_SUPPORTED 0x01
#define PRECONNECTION_PDU_V1_SIZE 16 #define PRECONNECTION_PDU_V1_SIZE 16
#define PRECONNECTION_PDU_V2_MIN_SIZE (PRECONNECTION_PDU_V1_SIZE+2) #define PRECONNECTION_PDU_V2_MIN_SIZE (PRECONNECTION_PDU_V1_SIZE + 2)
#define PRECONNECTION_PDU_V1 1 #define PRECONNECTION_PDU_V1 1
#define PRECONNECTION_PDU_V2 2 #define PRECONNECTION_PDU_V2 2
#define MSTSC_COOKIE_MAX_LENGTH 9
#define DEFAULT_COOKIE_MAX_LENGTH 0xFF
struct rdp_nego struct rdp_nego
{ {
int port; int port;
@ -88,6 +91,7 @@ struct rdp_nego
NEGO_STATE state; NEGO_STATE state;
BOOL tcp_connected; BOOL tcp_connected;
BOOL security_connected; BOOL security_connected;
UINT32 cookie_max_length;
UINT32 selected_protocol; UINT32 selected_protocol;
UINT32 requested_protocols; UINT32 requested_protocols;
@ -128,6 +132,7 @@ void nego_enable_nla(rdpNego* nego, BOOL enable_nla);
void nego_enable_tls(rdpNego* nego, BOOL enable_tls); void nego_enable_tls(rdpNego* nego, BOOL enable_tls);
void nego_set_routing_token(rdpNego* nego, BYTE* RoutingToken, DWORD RoutingTokenLength); void nego_set_routing_token(rdpNego* nego, BYTE* RoutingToken, DWORD RoutingTokenLength);
void nego_set_cookie(rdpNego* nego, char* cookie); void nego_set_cookie(rdpNego* nego, char* cookie);
void nego_set_cookie_max_length(rdpNego* nego, UINT32 cookie_max_length);
void nego_set_send_preconnection_pdu(rdpNego* nego, BOOL send_pcpdu); void nego_set_send_preconnection_pdu(rdpNego* nego, BOOL send_pcpdu);
void nego_set_preconnection_id(rdpNego* nego, UINT32 id); void nego_set_preconnection_id(rdpNego* nego, UINT32 id);
void nego_set_preconnection_blob(rdpNego* nego, char* blob); void nego_set_preconnection_blob(rdpNego* nego, char* blob);

View File

@ -31,6 +31,7 @@
#endif #endif
#include <winpr/crt.h> #include <winpr/crt.h>
#include <winpr/sysinfo.h>
#include <winpr/registry.h> #include <winpr/registry.h>
#include <freerdp/settings.h> #include <freerdp/settings.h>
@ -184,6 +185,15 @@ void settings_load_hkey_local_machine(rdpSettings* settings)
settings_client_load_hkey_local_machine(settings); settings_client_load_hkey_local_machine(settings);
} }
void settings_get_computer_name(rdpSettings* settings)
{
DWORD nSize = 0;
GetComputerNameExA(ComputerNameNetBIOS, NULL, &nSize);
settings->computer_name = (char*) malloc(nSize);
GetComputerNameExA(ComputerNameNetBIOS, settings->computer_name, &nSize);
}
rdpSettings* settings_new(void* instance) rdpSettings* settings_new(void* instance)
{ {
rdpSettings* settings; rdpSettings* settings;
@ -235,6 +245,8 @@ rdpSettings* settings_new(void* instance)
settings->authentication_only = FALSE; settings->authentication_only = FALSE;
settings->from_stdin = FALSE; settings->from_stdin = FALSE;
settings_get_computer_name(settings);
settings->received_caps = xzalloc(32); settings->received_caps = xzalloc(32);
settings->order_support = xzalloc(32); settings->order_support = xzalloc(32);