2011-07-25 21:52:48 +04:00
|
|
|
/**
|
|
|
|
* FreeRDP: A Remote Desktop Protocol Client
|
|
|
|
* Activation Sequence
|
|
|
|
*
|
|
|
|
* Copyright 2011 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "activation.h"
|
|
|
|
|
2011-11-30 05:15:50 +04:00
|
|
|
/*
|
|
|
|
static const char* const CTRLACTION_STRINGS[] =
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
"",
|
|
|
|
"CTRLACTION_REQUEST_CONTROL",
|
|
|
|
"CTRLACTION_GRANTED_CONTROL",
|
|
|
|
"CTRLACTION_DETACH",
|
|
|
|
"CTRLACTION_COOPERATE"
|
|
|
|
};
|
2011-11-30 05:15:50 +04:00
|
|
|
*/
|
2011-07-25 21:52:48 +04:00
|
|
|
|
2011-08-21 17:48:42 +04:00
|
|
|
void rdp_write_synchronize_pdu(STREAM* s, rdpSettings* settings)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
stream_write_uint16(s, SYNCMSGTYPE_SYNC); /* messageType (2 bytes) */
|
|
|
|
stream_write_uint16(s, settings->pdu_source); /* targetUser (2 bytes) */
|
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_server_synchronize_pdu(rdpRdp* rdp, STREAM* s)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
2011-08-21 17:48:42 +04:00
|
|
|
boolean rdp_send_server_synchronize_pdu(rdpRdp* rdp)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
2011-08-21 17:48:42 +04:00
|
|
|
rdp_write_synchronize_pdu(s, rdp->settings);
|
2011-07-25 21:52:48 +04:00
|
|
|
|
|
|
|
rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_SYNCHRONIZE, rdp->mcs->user_id);
|
2011-08-21 17:48:42 +04:00
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 17:48:42 +04:00
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_client_synchronize_pdu(STREAM* s)
|
2011-08-21 18:52:37 +04:00
|
|
|
{
|
|
|
|
uint16 messageType;
|
|
|
|
|
|
|
|
if (stream_get_left(s) < 4)
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 18:52:37 +04:00
|
|
|
|
|
|
|
stream_read_uint16(s, messageType); /* messageType (2 bytes) */
|
|
|
|
if (messageType != SYNCMSGTYPE_SYNC)
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 18:52:37 +04:00
|
|
|
/* targetUser (2 bytes) */
|
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 18:52:37 +04:00
|
|
|
}
|
|
|
|
|
2011-08-21 17:48:42 +04:00
|
|
|
boolean rdp_send_client_synchronize_pdu(rdpRdp* rdp)
|
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
rdp_write_synchronize_pdu(s, rdp->settings);
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_SYNCHRONIZE, rdp->mcs->user_id);
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_control_pdu(STREAM* s, uint16* action)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
2011-08-21 18:52:37 +04:00
|
|
|
if (stream_get_left(s) < 8)
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 18:52:37 +04:00
|
|
|
|
2011-07-25 21:52:48 +04:00
|
|
|
stream_read_uint16(s, *action); /* action (2 bytes) */
|
|
|
|
stream_seek_uint16(s); /* grantId (2 bytes) */
|
|
|
|
stream_seek_uint32(s); /* controlId (4 bytes) */
|
2011-08-21 18:52:37 +04:00
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void rdp_write_client_control_pdu(STREAM* s, uint16 action)
|
|
|
|
{
|
|
|
|
stream_write_uint16(s, action); /* action (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* grantId (2 bytes) */
|
|
|
|
stream_write_uint32(s, 0); /* controlId (4 bytes) */
|
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_server_control_pdu(rdpRdp* rdp, STREAM* s)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
uint16 action;
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
rdp_recv_control_pdu(s, &action);
|
2011-07-25 21:52:48 +04:00
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
2011-08-21 17:48:42 +04:00
|
|
|
boolean rdp_send_server_control_cooperate_pdu(rdpRdp* rdp)
|
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
stream_write_uint16(s, CTRLACTION_COOPERATE); /* action (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* grantId (2 bytes) */
|
|
|
|
stream_write_uint32(s, 0); /* controlId (4 bytes) */
|
|
|
|
|
|
|
|
rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_CONTROL, rdp->mcs->user_id);
|
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 17:48:42 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
boolean rdp_send_server_control_granted_pdu(rdpRdp* rdp)
|
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
stream_write_uint16(s, CTRLACTION_GRANTED_CONTROL); /* action (2 bytes) */
|
|
|
|
stream_write_uint16(s, rdp->mcs->user_id); /* grantId (2 bytes) */
|
|
|
|
stream_write_uint32(s, 0x03EA); /* controlId (4 bytes) */
|
|
|
|
|
|
|
|
rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_CONTROL, rdp->mcs->user_id);
|
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 17:48:42 +04:00
|
|
|
}
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
boolean rdp_send_client_control_pdu(rdpRdp* rdp, uint16 action)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
rdp_write_client_control_pdu(s, action);
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_CONTROL, rdp->mcs->user_id);
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void rdp_write_persistent_list_entry(STREAM* s, uint32 key1, uint32 key2)
|
|
|
|
{
|
|
|
|
stream_write_uint32(s, key1); /* key1 (4 bytes) */
|
|
|
|
stream_write_uint32(s, key2); /* key2 (4 bytes) */
|
|
|
|
}
|
|
|
|
|
|
|
|
void rdp_write_client_persistent_key_list_pdu(STREAM* s, rdpSettings* settings)
|
|
|
|
{
|
|
|
|
stream_write_uint16(s, 0); /* numEntriesCache0 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* numEntriesCache1 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* numEntriesCache2 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* numEntriesCache3 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* numEntriesCache4 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalEntriesCache0 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalEntriesCache1 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalEntriesCache2 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalEntriesCache3 (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalEntriesCache4 (2 bytes) */
|
2011-10-12 01:53:50 +04:00
|
|
|
stream_write_uint8(s, PERSIST_FIRST_PDU | PERSIST_LAST_PDU); /* bBitMask (1 byte) */
|
2011-07-25 21:52:48 +04:00
|
|
|
stream_write_uint8(s, 0); /* pad1 (1 byte) */
|
|
|
|
stream_write_uint16(s, 0); /* pad3 (2 bytes) */
|
|
|
|
|
|
|
|
/* entries */
|
|
|
|
}
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
boolean rdp_send_client_persistent_key_list_pdu(rdpRdp* rdp)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
rdp_write_client_persistent_key_list_pdu(s, rdp->settings);
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_BITMAP_CACHE_PERSISTENT_LIST, rdp->mcs->user_id);
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_client_font_list_pdu(STREAM* s)
|
2011-08-21 19:20:37 +04:00
|
|
|
{
|
|
|
|
if (stream_get_left(s) < 8)
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 19:20:37 +04:00
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 19:20:37 +04:00
|
|
|
}
|
|
|
|
|
2011-07-25 21:52:48 +04:00
|
|
|
void rdp_write_client_font_list_pdu(STREAM* s, uint16 flags)
|
|
|
|
{
|
|
|
|
stream_write_uint16(s, 0); /* numberFonts (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalNumFonts (2 bytes) */
|
|
|
|
stream_write_uint16(s, flags); /* listFlags (2 bytes) */
|
|
|
|
stream_write_uint16(s, 50); /* entrySize (2 bytes) */
|
|
|
|
}
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
boolean rdp_send_client_font_list_pdu(rdpRdp* rdp, uint16 flags)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
rdp_write_client_font_list_pdu(s, flags);
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_FONT_LIST, rdp->mcs->user_id);
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_server_font_map_pdu(rdpRdp* rdp, STREAM* s)
|
2011-07-25 21:52:48 +04:00
|
|
|
{
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-07-25 21:52:48 +04:00
|
|
|
}
|
2011-07-29 01:44:09 +04:00
|
|
|
|
2011-08-21 19:20:37 +04:00
|
|
|
boolean rdp_send_server_font_map_pdu(rdpRdp* rdp)
|
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_data_pdu_init(rdp);
|
|
|
|
|
|
|
|
stream_write_uint16(s, 0); /* numberEntries (2 bytes) */
|
|
|
|
stream_write_uint16(s, 0); /* totalNumEntries (2 bytes) */
|
|
|
|
stream_write_uint16(s, FONTLIST_FIRST | FONTLIST_LAST); /* mapFlags (2 bytes) */
|
|
|
|
stream_write_uint16(s, 4); /* entrySize (2 bytes) */
|
|
|
|
|
2011-09-06 13:19:16 +04:00
|
|
|
return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_FONT_MAP, rdp->mcs->user_id);
|
2011-08-21 19:20:37 +04:00
|
|
|
}
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
boolean rdp_recv_deactivate_all(rdpRdp* rdp, STREAM* s)
|
2011-07-29 01:44:09 +04:00
|
|
|
{
|
|
|
|
uint16 lengthSourceDescriptor;
|
|
|
|
|
|
|
|
stream_read_uint32(s, rdp->settings->share_id); /* shareId (4 bytes) */
|
|
|
|
stream_read_uint16(s, lengthSourceDescriptor); /* lengthSourceDescriptor (2 bytes) */
|
|
|
|
stream_seek(s, lengthSourceDescriptor); /* sourceDescriptor (should be 0x00) */
|
|
|
|
|
2011-08-22 11:03:58 +04:00
|
|
|
rdp->state = CONNECTION_STATE_CAPABILITY;
|
|
|
|
|
2011-09-06 17:45:52 +04:00
|
|
|
while (rdp->state != CONNECTION_STATE_ACTIVE)
|
|
|
|
{
|
|
|
|
if (rdp_check_fds(rdp) < 0)
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-09-06 17:45:52 +04:00
|
|
|
}
|
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-07-29 01:44:09 +04:00
|
|
|
}
|
|
|
|
|
2011-09-06 13:19:16 +04:00
|
|
|
boolean rdp_send_deactivate_all(rdpRdp* rdp)
|
|
|
|
{
|
|
|
|
STREAM* s;
|
|
|
|
|
|
|
|
s = rdp_pdu_init(rdp);
|
|
|
|
|
|
|
|
stream_write_uint32(s, rdp->settings->share_id); /* shareId (4 bytes) */
|
|
|
|
stream_write_uint16(s, 1); /* lengthSourceDescriptor (2 bytes) */
|
|
|
|
stream_write_uint8(s, 0); /* sourceDescriptor (should be 0x00) */
|
|
|
|
|
|
|
|
return rdp_send_pdu(rdp, s, PDU_TYPE_DEACTIVATE_ALL, rdp->mcs->user_id);
|
|
|
|
}
|
|
|
|
|
2011-08-21 19:20:37 +04:00
|
|
|
boolean rdp_server_accept_client_control_pdu(rdpRdp* rdp, STREAM* s)
|
|
|
|
{
|
|
|
|
uint16 action;
|
|
|
|
|
2011-08-23 08:58:10 +04:00
|
|
|
if (!rdp_recv_control_pdu(s, &action))
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 19:20:37 +04:00
|
|
|
if (action == CTRLACTION_REQUEST_CONTROL)
|
|
|
|
{
|
|
|
|
if (!rdp_send_server_control_granted_pdu(rdp))
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 19:20:37 +04:00
|
|
|
}
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 19:20:37 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
boolean rdp_server_accept_client_font_list_pdu(rdpRdp* rdp, STREAM* s)
|
|
|
|
{
|
2011-08-23 08:58:10 +04:00
|
|
|
if (!rdp_recv_client_font_list_pdu(s))
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 19:20:37 +04:00
|
|
|
if (!rdp_send_server_font_map_pdu(rdp))
|
2011-11-19 21:19:16 +04:00
|
|
|
return false;
|
2011-08-21 19:20:37 +04:00
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
return true;
|
2011-08-21 19:20:37 +04:00
|
|
|
}
|
|
|
|
|