4c4019203a
Commit 0357a38e31
modified the function
fastpath_send_update_pdu() to check if the desired update is possible
by checking the payload size against the computed maxLength and the
clients's advertised max request size.
If the check failed that commit added a workaround which simply
copied the payload to a slow path updade.
This workaround is totally flawed and causes protocol errors:
- the fast path update code is not checked and required data format
conversions are missing
- depending on the fast path update code rdp_send_data_pdu() would
have to be called with differend data pdu type values but the
workaround always uses DATA_PDU_TYPE_UPDATE
- the workaround does not check if the total size would exceed
the maximum possible size for a slow path update
The check if a fast path output is actually possible with the
passed parameters is basically a good idea.
However, if that check fails it would only indicate an error in
the server implementation who must not generate updates that
exceed the client's max request size.
Even though a slow-path conversion would be possible there is
much more involved than simply copying the payload stream.
In addition it is highly doubtful if there is a benefit at all.
Even the oldest rdesktop and windows ce clients do support fast
path and although some lack the multi-fragment update capability
we cannot really send larger updates using slow-path outputs.
For the reasons elucidated above, I have removed the workaround
but kept a modified version of the check if a fast-path output
is possible at all.
1072 lines
26 KiB
C
1072 lines
26 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* Fast Path
|
|
*
|
|
* Copyright 2011 Vic Lee
|
|
* Copyright 2014 Norbert Federa <norbert.federa@thincast.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.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <winpr/crt.h>
|
|
#include <winpr/stream.h>
|
|
|
|
#include <freerdp/api.h>
|
|
#include <freerdp/log.h>
|
|
#include <freerdp/crypto/per.h>
|
|
|
|
#include "orders.h"
|
|
#include "update.h"
|
|
#include "surface.h"
|
|
#include "fastpath.h"
|
|
#include "rdp.h"
|
|
|
|
#define TAG FREERDP_TAG("core.fastpath")
|
|
|
|
/**
|
|
* Fast-Path packet format is defined in [MS-RDPBCGR] 2.2.9.1.2, which revises
|
|
* server output packets from the first byte with the goal of improving
|
|
* bandwidth.
|
|
*
|
|
* Slow-Path packet always starts with TPKT header, which has the first
|
|
* byte 0x03, while Fast-Path packet starts with 2 zero bits in the first
|
|
* two less significant bits of the first byte.
|
|
*/
|
|
|
|
|
|
#ifdef WITH_DEBUG_RDP
|
|
static const char* const FASTPATH_UPDATETYPE_STRINGS[] =
|
|
{
|
|
"Orders", /* 0x0 */
|
|
"Bitmap", /* 0x1 */
|
|
"Palette", /* 0x2 */
|
|
"Synchronize", /* 0x3 */
|
|
"Surface Commands", /* 0x4 */
|
|
"System Pointer Hidden", /* 0x5 */
|
|
"System Pointer Default", /* 0x6 */
|
|
"???", /* 0x7 */
|
|
"Pointer Position", /* 0x8 */
|
|
"Color Pointer", /* 0x9 */
|
|
"Cached Pointer", /* 0xA */
|
|
"New Pointer", /* 0xB */
|
|
};
|
|
#endif
|
|
|
|
/*
|
|
* The fastpath header may be two or three bytes long.
|
|
* This function assumes that at least two bytes are available in the stream
|
|
* and doesn't touch third byte.
|
|
*/
|
|
UINT16 fastpath_header_length(wStream* s)
|
|
{
|
|
BYTE length1;
|
|
|
|
Stream_Seek_UINT8(s);
|
|
Stream_Read_UINT8(s, length1);
|
|
Stream_Rewind(s, 2);
|
|
|
|
return ((length1 & 0x80) != 0 ? 3 : 2);
|
|
}
|
|
|
|
/**
|
|
* Read a Fast-Path packet header.\n
|
|
* @param s stream
|
|
* @param encryptionFlags
|
|
* @return length
|
|
*/
|
|
UINT16 fastpath_read_header(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
BYTE header;
|
|
UINT16 length;
|
|
|
|
Stream_Read_UINT8(s, header);
|
|
|
|
if (fastpath)
|
|
{
|
|
fastpath->encryptionFlags = (header & 0xC0) >> 6;
|
|
fastpath->numberEvents = (header & 0x3C) >> 2;
|
|
}
|
|
|
|
per_read_length(s, &length);
|
|
|
|
return length;
|
|
}
|
|
|
|
void fastpath_read_update_header(wStream* s, BYTE* updateCode, BYTE* fragmentation, BYTE* compression)
|
|
{
|
|
BYTE updateHeader;
|
|
|
|
Stream_Read_UINT8(s, updateHeader);
|
|
*updateCode = updateHeader & 0x0F;
|
|
*fragmentation = (updateHeader >> 4) & 0x03;
|
|
*compression = (updateHeader >> 6) & 0x03;
|
|
}
|
|
|
|
void fastpath_write_update_header(wStream* s, FASTPATH_UPDATE_HEADER* fpUpdateHeader)
|
|
{
|
|
fpUpdateHeader->updateHeader = 0;
|
|
fpUpdateHeader->updateHeader |= fpUpdateHeader->updateCode & 0x0F;
|
|
fpUpdateHeader->updateHeader |= (fpUpdateHeader->fragmentation & 0x03) << 4;
|
|
fpUpdateHeader->updateHeader |= (fpUpdateHeader->compression & 0x03) << 6;
|
|
|
|
Stream_Write_UINT8(s, fpUpdateHeader->updateHeader);
|
|
|
|
if (fpUpdateHeader->compression)
|
|
Stream_Write_UINT8(s, fpUpdateHeader->compressionFlags);
|
|
|
|
Stream_Write_UINT16(s, fpUpdateHeader->size);
|
|
}
|
|
|
|
UINT32 fastpath_get_update_header_size(FASTPATH_UPDATE_HEADER* fpUpdateHeader)
|
|
{
|
|
return (fpUpdateHeader->compression) ? 4 : 3;
|
|
}
|
|
|
|
void fastpath_write_update_pdu_header(wStream* s, FASTPATH_UPDATE_PDU_HEADER* fpUpdatePduHeader, rdpRdp* rdp)
|
|
{
|
|
fpUpdatePduHeader->fpOutputHeader = 0;
|
|
fpUpdatePduHeader->fpOutputHeader |= (fpUpdatePduHeader->action & 0x03);
|
|
fpUpdatePduHeader->fpOutputHeader |= (fpUpdatePduHeader->secFlags & 0x03) << 6;
|
|
|
|
Stream_Write_UINT8(s, fpUpdatePduHeader->fpOutputHeader); /* fpOutputHeader (1 byte) */
|
|
|
|
Stream_Write_UINT8(s, 0x80 | (fpUpdatePduHeader->length >> 8)); /* length1 */
|
|
Stream_Write_UINT8(s, fpUpdatePduHeader->length & 0xFF); /* length2 */
|
|
|
|
if (fpUpdatePduHeader->secFlags)
|
|
{
|
|
if (rdp->settings->EncryptionMethods == ENCRYPTION_METHOD_FIPS)
|
|
Stream_Write(s, fpUpdatePduHeader->fipsInformation, 4);
|
|
Stream_Write(s, fpUpdatePduHeader->dataSignature, 8);
|
|
}
|
|
}
|
|
|
|
UINT32 fastpath_get_update_pdu_header_size(FASTPATH_UPDATE_PDU_HEADER* fpUpdatePduHeader, rdpRdp* rdp)
|
|
{
|
|
UINT32 size = 3; /* fpUpdatePduHeader + length1 + length2 */
|
|
|
|
if (fpUpdatePduHeader->secFlags)
|
|
{
|
|
size += 8; /* dataSignature */
|
|
|
|
if (rdp->settings->EncryptionMethods == ENCRYPTION_METHOD_FIPS)
|
|
size += 4; /* fipsInformation */
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
BOOL fastpath_read_header_rdp(rdpFastPath* fastpath, wStream* s, UINT16 *length)
|
|
{
|
|
BYTE header;
|
|
|
|
Stream_Read_UINT8(s, header);
|
|
|
|
if (fastpath)
|
|
{
|
|
fastpath->encryptionFlags = (header & 0xC0) >> 6;
|
|
fastpath->numberEvents = (header & 0x3C) >> 2;
|
|
}
|
|
|
|
if (!per_read_length(s, length))
|
|
return FALSE;
|
|
|
|
*length = *length - Stream_GetPosition(s);
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_orders(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
rdpUpdate* update = fastpath->rdp->update;
|
|
UINT16 numberOrders;
|
|
|
|
Stream_Read_UINT16(s, numberOrders); /* numberOrders (2 bytes) */
|
|
|
|
while (numberOrders > 0)
|
|
{
|
|
if (!update_recv_order(update, s))
|
|
return FALSE;
|
|
|
|
numberOrders--;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_update_common(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
UINT16 updateType;
|
|
rdpUpdate* update = fastpath->rdp->update;
|
|
rdpContext* context = update->context;
|
|
|
|
if (Stream_GetRemainingLength(s) < 2)
|
|
return FALSE;
|
|
|
|
Stream_Read_UINT16(s, updateType); /* updateType (2 bytes) */
|
|
|
|
switch (updateType)
|
|
{
|
|
case UPDATE_TYPE_BITMAP:
|
|
if (!update_read_bitmap_update(update, s, &update->bitmap_update))
|
|
return FALSE;
|
|
IFCALL(update->BitmapUpdate, context, &update->bitmap_update);
|
|
break;
|
|
|
|
case UPDATE_TYPE_PALETTE:
|
|
if (!update_read_palette(update, s, &update->palette_update))
|
|
return FALSE;
|
|
IFCALL(update->Palette, context, &update->palette_update);
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_update_synchronize(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
/* server 2008 can send invalid synchronize packet with missing padding,
|
|
so don't return FALSE even if the packet is invalid */
|
|
Stream_SafeSeek(s, 2); /* size (2 bytes), MUST be set to zero */
|
|
return TRUE;
|
|
}
|
|
|
|
static int fastpath_recv_update(rdpFastPath* fastpath, BYTE updateCode, UINT32 size, wStream* s)
|
|
{
|
|
int status = 0;
|
|
rdpUpdate* update = fastpath->rdp->update;
|
|
rdpContext* context = fastpath->rdp->update->context;
|
|
rdpPointerUpdate* pointer = update->pointer;
|
|
|
|
#ifdef WITH_DEBUG_RDP
|
|
DEBUG_RDP("recv Fast-Path %s Update (0x%X), length:%d",
|
|
updateCode < ARRAYSIZE(FASTPATH_UPDATETYPE_STRINGS) ? FASTPATH_UPDATETYPE_STRINGS[updateCode] : "???", updateCode, size);
|
|
#endif
|
|
|
|
switch (updateCode)
|
|
{
|
|
case FASTPATH_UPDATETYPE_ORDERS:
|
|
if (!fastpath_recv_orders(fastpath, s))
|
|
return -1;
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_BITMAP:
|
|
case FASTPATH_UPDATETYPE_PALETTE:
|
|
if (!fastpath_recv_update_common(fastpath, s))
|
|
return -1;
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_SYNCHRONIZE:
|
|
if (!fastpath_recv_update_synchronize(fastpath, s))
|
|
WLog_ERR(TAG, "fastpath_recv_update_synchronize failure but we continue");
|
|
else
|
|
IFCALL(update->Synchronize, context);
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_SURFCMDS:
|
|
status = update_recv_surfcmds(update, size, s);
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_PTR_NULL:
|
|
pointer->pointer_system.type = SYSPTR_NULL;
|
|
IFCALL(pointer->PointerSystem, context, &pointer->pointer_system);
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_PTR_DEFAULT:
|
|
update->pointer->pointer_system.type = SYSPTR_DEFAULT;
|
|
IFCALL(pointer->PointerSystem, context, &pointer->pointer_system);
|
|
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_PTR_POSITION:
|
|
if (!update_read_pointer_position(s, &pointer->pointer_position))
|
|
return -1;
|
|
IFCALL(pointer->PointerPosition, context, &pointer->pointer_position);
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_COLOR:
|
|
if (!update_read_pointer_color(s, &pointer->pointer_color, 24))
|
|
return -1;
|
|
IFCALL(pointer->PointerColor, context, &pointer->pointer_color);
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_CACHED:
|
|
if (!update_read_pointer_cached(s, &pointer->pointer_cached))
|
|
return -1;
|
|
IFCALL(pointer->PointerCached, context, &pointer->pointer_cached);
|
|
break;
|
|
|
|
case FASTPATH_UPDATETYPE_POINTER:
|
|
if (!update_read_pointer_new(s, &pointer->pointer_new))
|
|
return -1;
|
|
IFCALL(pointer->PointerNew, context, &pointer->pointer_new);
|
|
break;
|
|
|
|
default:
|
|
WLog_ERR(TAG, "unknown updateCode 0x%X", updateCode);
|
|
break;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
const char* fastpath_get_fragmentation_string(BYTE fragmentation)
|
|
{
|
|
if (fragmentation == FASTPATH_FRAGMENT_SINGLE)
|
|
return "FASTPATH_FRAGMENT_SINGLE";
|
|
else if (fragmentation == FASTPATH_FRAGMENT_LAST)
|
|
return "FASTPATH_FRAGMENT_LAST";
|
|
else if (fragmentation == FASTPATH_FRAGMENT_FIRST)
|
|
return "FASTPATH_FRAGMENT_FIRST";
|
|
else if (fragmentation == FASTPATH_FRAGMENT_NEXT)
|
|
return "FASTPATH_FRAGMENT_NEXT";
|
|
|
|
return "FASTPATH_FRAGMENT_UNKNOWN";
|
|
}
|
|
|
|
static int fastpath_recv_update_data(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
int status;
|
|
UINT16 size;
|
|
rdpRdp* rdp;
|
|
int next_pos;
|
|
wStream* cs;
|
|
int bulkStatus;
|
|
UINT32 totalSize;
|
|
BYTE updateCode;
|
|
BYTE fragmentation;
|
|
BYTE compression;
|
|
BYTE compressionFlags;
|
|
UINT32 DstSize = 0;
|
|
BYTE* pDstData = NULL;
|
|
rdpTransport* transport;
|
|
|
|
status = 0;
|
|
rdp = fastpath->rdp;
|
|
transport = fastpath->rdp->transport;
|
|
|
|
fastpath_read_update_header(s, &updateCode, &fragmentation, &compression);
|
|
|
|
if (compression == FASTPATH_OUTPUT_COMPRESSION_USED)
|
|
Stream_Read_UINT8(s, compressionFlags);
|
|
else
|
|
compressionFlags = 0;
|
|
|
|
Stream_Read_UINT16(s, size);
|
|
|
|
if (Stream_GetRemainingLength(s) < size)
|
|
return -1;
|
|
|
|
cs = s;
|
|
next_pos = Stream_GetPosition(s) + size;
|
|
|
|
bulkStatus = bulk_decompress(rdp->bulk, Stream_Pointer(s), size, &pDstData, &DstSize, compressionFlags);
|
|
|
|
if (bulkStatus < 0)
|
|
{
|
|
WLog_ERR(TAG, "bulk_decompress() failed");
|
|
return -1;
|
|
}
|
|
|
|
if (bulkStatus > 0)
|
|
{
|
|
/* data was compressed, copy from decompression buffer */
|
|
|
|
size = DstSize;
|
|
cs = StreamPool_Take(transport->ReceivePool, DstSize);
|
|
|
|
Stream_SetPosition(cs, 0);
|
|
Stream_Write(cs, pDstData, DstSize);
|
|
Stream_SealLength(cs);
|
|
Stream_SetPosition(cs, 0);
|
|
}
|
|
|
|
if (fragmentation == FASTPATH_FRAGMENT_SINGLE)
|
|
{
|
|
if (fastpath->fragmentation != -1)
|
|
{
|
|
WLog_ERR(TAG, "Unexpected FASTPATH_FRAGMENT_SINGLE");
|
|
return -1;
|
|
}
|
|
|
|
totalSize = size;
|
|
status = fastpath_recv_update(fastpath, updateCode, totalSize, cs);
|
|
|
|
if (status < 0)
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
if (fragmentation == FASTPATH_FRAGMENT_FIRST)
|
|
{
|
|
if (fastpath->fragmentation != -1)
|
|
{
|
|
WLog_ERR(TAG, "Unexpected FASTPATH_FRAGMENT_FIRST");
|
|
return -1;
|
|
}
|
|
|
|
fastpath->fragmentation = FASTPATH_FRAGMENT_FIRST;
|
|
|
|
totalSize = size;
|
|
|
|
if (totalSize > transport->settings->MultifragMaxRequestSize)
|
|
{
|
|
WLog_ERR(TAG, "Total size (%d) exceeds MultifragMaxRequestSize (%d)",
|
|
totalSize, transport->settings->MultifragMaxRequestSize);
|
|
return -1;
|
|
}
|
|
|
|
fastpath->updateData = StreamPool_Take(transport->ReceivePool, size);
|
|
Stream_SetPosition(fastpath->updateData, 0);
|
|
|
|
Stream_Copy(fastpath->updateData, cs, size);
|
|
}
|
|
else if (fragmentation == FASTPATH_FRAGMENT_NEXT)
|
|
{
|
|
if ((fastpath->fragmentation != FASTPATH_FRAGMENT_FIRST) &&
|
|
(fastpath->fragmentation != FASTPATH_FRAGMENT_NEXT))
|
|
{
|
|
WLog_ERR(TAG, "Unexpected FASTPATH_FRAGMENT_NEXT");
|
|
return -1;
|
|
}
|
|
|
|
fastpath->fragmentation = FASTPATH_FRAGMENT_NEXT;
|
|
|
|
totalSize = Stream_GetPosition(fastpath->updateData) + size;
|
|
|
|
if (totalSize > transport->settings->MultifragMaxRequestSize)
|
|
{
|
|
WLog_ERR(TAG, "Total size (%d) exceeds MultifragMaxRequestSize (%d)",
|
|
totalSize, transport->settings->MultifragMaxRequestSize);
|
|
return -1;
|
|
}
|
|
|
|
Stream_EnsureCapacity(fastpath->updateData, totalSize);
|
|
|
|
Stream_Copy(fastpath->updateData, cs, size);
|
|
}
|
|
else if (fragmentation == FASTPATH_FRAGMENT_LAST)
|
|
{
|
|
if ((fastpath->fragmentation != FASTPATH_FRAGMENT_FIRST) &&
|
|
(fastpath->fragmentation != FASTPATH_FRAGMENT_NEXT))
|
|
{
|
|
WLog_ERR(TAG, "Unexpected FASTPATH_FRAGMENT_LAST");
|
|
return -1;
|
|
}
|
|
|
|
fastpath->fragmentation = -1;
|
|
|
|
totalSize = Stream_GetPosition(fastpath->updateData) + size;
|
|
|
|
if (totalSize > transport->settings->MultifragMaxRequestSize)
|
|
{
|
|
WLog_ERR(TAG, "Total size (%d) exceeds MultifragMaxRequestSize (%d)",
|
|
totalSize, transport->settings->MultifragMaxRequestSize);
|
|
return -1;
|
|
}
|
|
|
|
Stream_EnsureCapacity(fastpath->updateData, totalSize);
|
|
|
|
Stream_Copy(fastpath->updateData, cs, size);
|
|
|
|
Stream_SealLength(fastpath->updateData);
|
|
Stream_SetPosition(fastpath->updateData, 0);
|
|
|
|
status = fastpath_recv_update(fastpath, updateCode, totalSize, fastpath->updateData);
|
|
|
|
Stream_Release(fastpath->updateData);
|
|
|
|
if (status < 0)
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
Stream_SetPosition(s, next_pos);
|
|
|
|
if (cs != s)
|
|
Stream_Release(cs);
|
|
|
|
return status;
|
|
}
|
|
|
|
int fastpath_recv_updates(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
int status = 0;
|
|
rdpUpdate* update = fastpath->rdp->update;
|
|
|
|
IFCALL(update->BeginPaint, update->context);
|
|
|
|
while (Stream_GetRemainingLength(s) >= 3)
|
|
{
|
|
if (fastpath_recv_update_data(fastpath, s) < 0)
|
|
return -1;
|
|
}
|
|
|
|
IFCALL(update->EndPaint, update->context);
|
|
|
|
return status;
|
|
}
|
|
|
|
static BOOL fastpath_read_input_event_header(wStream* s, BYTE* eventFlags, BYTE* eventCode)
|
|
{
|
|
BYTE eventHeader;
|
|
|
|
if (Stream_GetRemainingLength(s) < 1)
|
|
return FALSE;
|
|
|
|
Stream_Read_UINT8(s, eventHeader); /* eventHeader (1 byte) */
|
|
|
|
*eventFlags = (eventHeader & 0x1F);
|
|
*eventCode = (eventHeader >> 5);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_input_event_scancode(rdpFastPath* fastpath, wStream* s, BYTE eventFlags)
|
|
{
|
|
UINT16 flags;
|
|
UINT16 code;
|
|
|
|
if (Stream_GetRemainingLength(s) < 1)
|
|
return FALSE;
|
|
|
|
Stream_Read_UINT8(s, code); /* keyCode (1 byte) */
|
|
|
|
flags = 0;
|
|
|
|
if ((eventFlags & FASTPATH_INPUT_KBDFLAGS_RELEASE))
|
|
flags |= KBD_FLAGS_RELEASE;
|
|
else
|
|
flags |= KBD_FLAGS_DOWN;
|
|
|
|
if ((eventFlags & FASTPATH_INPUT_KBDFLAGS_EXTENDED))
|
|
flags |= KBD_FLAGS_EXTENDED;
|
|
|
|
IFCALL(fastpath->rdp->input->KeyboardEvent, fastpath->rdp->input, flags, code);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_input_event_mouse(rdpFastPath* fastpath, wStream* s, BYTE eventFlags)
|
|
{
|
|
UINT16 pointerFlags;
|
|
UINT16 xPos;
|
|
UINT16 yPos;
|
|
|
|
if (Stream_GetRemainingLength(s) < 6)
|
|
return FALSE;
|
|
|
|
Stream_Read_UINT16(s, pointerFlags); /* pointerFlags (2 bytes) */
|
|
Stream_Read_UINT16(s, xPos); /* xPos (2 bytes) */
|
|
Stream_Read_UINT16(s, yPos); /* yPos (2 bytes) */
|
|
|
|
IFCALL(fastpath->rdp->input->MouseEvent, fastpath->rdp->input, pointerFlags, xPos, yPos);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_input_event_mousex(rdpFastPath* fastpath, wStream* s, BYTE eventFlags)
|
|
{
|
|
UINT16 pointerFlags;
|
|
UINT16 xPos;
|
|
UINT16 yPos;
|
|
|
|
if (Stream_GetRemainingLength(s) < 6)
|
|
return FALSE;
|
|
|
|
Stream_Read_UINT16(s, pointerFlags); /* pointerFlags (2 bytes) */
|
|
Stream_Read_UINT16(s, xPos); /* xPos (2 bytes) */
|
|
Stream_Read_UINT16(s, yPos); /* yPos (2 bytes) */
|
|
|
|
IFCALL(fastpath->rdp->input->ExtendedMouseEvent, fastpath->rdp->input, pointerFlags, xPos, yPos);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_input_event_sync(rdpFastPath* fastpath, wStream* s, BYTE eventFlags)
|
|
{
|
|
IFCALL(fastpath->rdp->input->SynchronizeEvent, fastpath->rdp->input, eventFlags);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_input_event_unicode(rdpFastPath* fastpath, wStream* s, BYTE eventFlags)
|
|
{
|
|
UINT16 unicodeCode;
|
|
UINT16 flags;
|
|
|
|
if (Stream_GetRemainingLength(s) < 2)
|
|
return FALSE;
|
|
|
|
Stream_Read_UINT16(s, unicodeCode); /* unicodeCode (2 bytes) */
|
|
|
|
flags = 0;
|
|
|
|
if ((eventFlags & FASTPATH_INPUT_KBDFLAGS_RELEASE))
|
|
flags |= KBD_FLAGS_RELEASE;
|
|
else
|
|
flags |= KBD_FLAGS_DOWN;
|
|
|
|
IFCALL(fastpath->rdp->input->UnicodeKeyboardEvent, fastpath->rdp->input, flags, unicodeCode);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL fastpath_recv_input_event(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
BYTE eventFlags;
|
|
BYTE eventCode;
|
|
|
|
if (!fastpath_read_input_event_header(s, &eventFlags, &eventCode))
|
|
return FALSE;
|
|
|
|
switch (eventCode)
|
|
{
|
|
case FASTPATH_INPUT_EVENT_SCANCODE:
|
|
if (!fastpath_recv_input_event_scancode(fastpath, s, eventFlags))
|
|
return FALSE;
|
|
break;
|
|
|
|
case FASTPATH_INPUT_EVENT_MOUSE:
|
|
if (!fastpath_recv_input_event_mouse(fastpath, s, eventFlags))
|
|
return FALSE;
|
|
break;
|
|
|
|
case FASTPATH_INPUT_EVENT_MOUSEX:
|
|
if (!fastpath_recv_input_event_mousex(fastpath, s, eventFlags))
|
|
return FALSE;
|
|
break;
|
|
|
|
case FASTPATH_INPUT_EVENT_SYNC:
|
|
if (!fastpath_recv_input_event_sync(fastpath, s, eventFlags))
|
|
return FALSE;
|
|
break;
|
|
|
|
case FASTPATH_INPUT_EVENT_UNICODE:
|
|
if (!fastpath_recv_input_event_unicode(fastpath, s, eventFlags))
|
|
return FALSE;
|
|
break;
|
|
|
|
default:
|
|
WLog_ERR(TAG, "Unknown eventCode %d", eventCode);
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
int fastpath_recv_inputs(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
BYTE i;
|
|
|
|
if (fastpath->numberEvents == 0)
|
|
{
|
|
/**
|
|
* If numberEvents is not provided in fpInputHeader, it will be provided
|
|
* as one additional byte here.
|
|
*/
|
|
|
|
if (Stream_GetRemainingLength(s) < 1)
|
|
return -1;
|
|
|
|
Stream_Read_UINT8(s, fastpath->numberEvents); /* eventHeader (1 byte) */
|
|
}
|
|
|
|
for (i = 0; i < fastpath->numberEvents; i++)
|
|
{
|
|
if (!fastpath_recv_input_event(fastpath, s))
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static UINT32 fastpath_get_sec_bytes(rdpRdp* rdp)
|
|
{
|
|
UINT32 sec_bytes;
|
|
|
|
sec_bytes = 0;
|
|
|
|
if (rdp->do_crypt)
|
|
{
|
|
sec_bytes = 8;
|
|
|
|
if (rdp->settings->EncryptionMethods == ENCRYPTION_METHOD_FIPS)
|
|
sec_bytes += 4;
|
|
}
|
|
|
|
return sec_bytes;
|
|
}
|
|
|
|
wStream* fastpath_input_pdu_init_header(rdpFastPath* fastpath)
|
|
{
|
|
rdpRdp *rdp;
|
|
wStream* s;
|
|
|
|
rdp = fastpath->rdp;
|
|
|
|
s = transport_send_stream_init(rdp->transport, 256);
|
|
|
|
Stream_Seek(s, 3); /* fpInputHeader, length1 and length2 */
|
|
|
|
if (rdp->do_crypt)
|
|
{
|
|
rdp->sec_flags |= SEC_ENCRYPT;
|
|
|
|
if (rdp->do_secure_checksum)
|
|
rdp->sec_flags |= SEC_SECURE_CHECKSUM;
|
|
}
|
|
|
|
Stream_Seek(s, fastpath_get_sec_bytes(rdp));
|
|
|
|
return s;
|
|
}
|
|
|
|
wStream* fastpath_input_pdu_init(rdpFastPath* fastpath, BYTE eventFlags, BYTE eventCode)
|
|
{
|
|
rdpRdp *rdp;
|
|
wStream* s;
|
|
|
|
rdp = fastpath->rdp;
|
|
|
|
s = fastpath_input_pdu_init_header(fastpath);
|
|
Stream_Write_UINT8(s, eventFlags | (eventCode << 5)); /* eventHeader (1 byte) */
|
|
|
|
return s;
|
|
}
|
|
|
|
BOOL fastpath_send_multiple_input_pdu(rdpFastPath* fastpath, wStream* s, int iNumEvents)
|
|
{
|
|
rdpRdp* rdp;
|
|
UINT16 length;
|
|
BYTE eventHeader;
|
|
|
|
/*
|
|
* A maximum of 15 events are allowed per request
|
|
* if the optional numEvents field isn't used
|
|
* see MS-RDPBCGR 2.2.8.1.2 for details
|
|
*/
|
|
if (iNumEvents > 15)
|
|
return FALSE;
|
|
|
|
rdp = fastpath->rdp;
|
|
|
|
length = Stream_GetPosition(s);
|
|
|
|
if (length >= (2 << 14))
|
|
{
|
|
WLog_ERR(TAG, "Maximum FastPath PDU length is 32767");
|
|
return FALSE;
|
|
}
|
|
|
|
eventHeader = FASTPATH_INPUT_ACTION_FASTPATH;
|
|
eventHeader |= (iNumEvents << 2); /* numberEvents */
|
|
|
|
if (rdp->sec_flags & SEC_ENCRYPT)
|
|
eventHeader |= (FASTPATH_INPUT_ENCRYPTED << 6);
|
|
if (rdp->sec_flags & SEC_SECURE_CHECKSUM)
|
|
eventHeader |= (FASTPATH_INPUT_SECURE_CHECKSUM << 6);
|
|
|
|
Stream_SetPosition(s, 0);
|
|
Stream_Write_UINT8(s, eventHeader);
|
|
|
|
/* Write length later, RDP encryption might add a padding */
|
|
Stream_Seek(s, 2);
|
|
|
|
if (rdp->sec_flags & SEC_ENCRYPT)
|
|
{
|
|
int sec_bytes = fastpath_get_sec_bytes(fastpath->rdp);
|
|
BYTE* fpInputEvents = Stream_Pointer(s) + sec_bytes;
|
|
UINT16 fpInputEvents_length = length - 3 - sec_bytes;
|
|
|
|
if (rdp->settings->EncryptionMethods == ENCRYPTION_METHOD_FIPS)
|
|
{
|
|
BYTE pad;
|
|
|
|
if ((pad = 8 - (fpInputEvents_length % 8)) == 8)
|
|
pad = 0;
|
|
|
|
Stream_Write_UINT16(s, 0x10); /* length */
|
|
Stream_Write_UINT8(s, 0x1); /* TSFIPS_VERSION 1*/
|
|
Stream_Write_UINT8(s, pad); /* padding */
|
|
|
|
security_hmac_signature(fpInputEvents, fpInputEvents_length, Stream_Pointer(s), rdp);
|
|
|
|
if (pad)
|
|
memset(fpInputEvents + fpInputEvents_length, 0, pad);
|
|
|
|
security_fips_encrypt(fpInputEvents, fpInputEvents_length + pad, rdp);
|
|
|
|
length += pad;
|
|
}
|
|
else
|
|
{
|
|
if (rdp->sec_flags & SEC_SECURE_CHECKSUM)
|
|
security_salted_mac_signature(rdp, fpInputEvents, fpInputEvents_length, TRUE, Stream_Pointer(s));
|
|
else
|
|
security_mac_signature(rdp, fpInputEvents, fpInputEvents_length, Stream_Pointer(s));
|
|
|
|
security_encrypt(fpInputEvents, fpInputEvents_length, rdp);
|
|
}
|
|
}
|
|
|
|
rdp->sec_flags = 0;
|
|
|
|
/*
|
|
* We always encode length in two bytes, even though we could use
|
|
* only one byte if length <= 0x7F. It is just easier that way,
|
|
* because we can leave room for fixed-length header, store all
|
|
* the data first and then store the header.
|
|
*/
|
|
Stream_SetPosition(s, 1);
|
|
Stream_Write_UINT16_BE(s, 0x8000 | length);
|
|
|
|
Stream_SetPosition(s, length);
|
|
Stream_SealLength(s);
|
|
|
|
if (transport_write(fastpath->rdp->transport, s) < 0)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL fastpath_send_input_pdu(rdpFastPath* fastpath, wStream* s)
|
|
{
|
|
return fastpath_send_multiple_input_pdu(fastpath, s, 1);
|
|
}
|
|
|
|
wStream* fastpath_update_pdu_init(rdpFastPath* fastpath)
|
|
{
|
|
wStream* s;
|
|
s = transport_send_stream_init(fastpath->rdp->transport, FASTPATH_MAX_PACKET_SIZE);
|
|
return s;
|
|
}
|
|
|
|
wStream* fastpath_update_pdu_init_new(rdpFastPath* fastpath)
|
|
{
|
|
wStream* s;
|
|
s = Stream_New(NULL, FASTPATH_MAX_PACKET_SIZE);
|
|
return s;
|
|
}
|
|
|
|
BOOL fastpath_send_update_pdu(rdpFastPath* fastpath, BYTE updateCode, wStream* s, BOOL skipCompression)
|
|
{
|
|
int fragment;
|
|
UINT16 maxLength;
|
|
UINT32 totalLength;
|
|
BOOL status = TRUE;
|
|
wStream* fs = NULL;
|
|
rdpSettings* settings;
|
|
rdpRdp* rdp = fastpath->rdp;
|
|
UINT32 fpHeaderSize = 6;
|
|
UINT32 fpUpdatePduHeaderSize;
|
|
UINT32 fpUpdateHeaderSize;
|
|
UINT32 CompressionMaxSize;
|
|
FASTPATH_UPDATE_PDU_HEADER fpUpdatePduHeader = { 0 };
|
|
FASTPATH_UPDATE_HEADER fpUpdateHeader = { 0 };
|
|
|
|
fs = fastpath->fs;
|
|
settings = rdp->settings;
|
|
|
|
maxLength = FASTPATH_MAX_PACKET_SIZE - 20;
|
|
|
|
if (settings->CompressionEnabled && !skipCompression)
|
|
{
|
|
CompressionMaxSize = bulk_compression_max_size(rdp->bulk);
|
|
maxLength = (maxLength < CompressionMaxSize) ? maxLength : CompressionMaxSize;
|
|
maxLength -= 20;
|
|
}
|
|
|
|
totalLength = Stream_GetPosition(s);
|
|
Stream_SetPosition(s, 0);
|
|
|
|
/* check if fast path output is possible */
|
|
if (!settings->FastPathOutput)
|
|
{
|
|
WLog_ERR(TAG, "client does not support fast path output");
|
|
return FALSE;
|
|
}
|
|
|
|
/* check if the client's fast path pdu buffer is large enough */
|
|
if (totalLength > settings->MultifragMaxRequestSize)
|
|
{
|
|
WLog_ERR(TAG, "fast path update size (%u) exceeds the client's maximum request size (%u)",
|
|
totalLength, settings->MultifragMaxRequestSize);
|
|
return FALSE;
|
|
}
|
|
|
|
if (rdp->do_crypt)
|
|
{
|
|
rdp->sec_flags |= SEC_ENCRYPT;
|
|
|
|
if (rdp->do_secure_checksum)
|
|
rdp->sec_flags |= SEC_SECURE_CHECKSUM;
|
|
}
|
|
|
|
for (fragment = 0; (totalLength > 0) || (fragment == 0); fragment++)
|
|
{
|
|
BYTE* pSrcData;
|
|
UINT32 SrcSize;
|
|
UINT32 DstSize = 0;
|
|
BYTE* pDstData = NULL;
|
|
UINT32 compressionFlags = 0;
|
|
BYTE pad = 0;
|
|
BYTE* pSignature = NULL;
|
|
|
|
fpUpdatePduHeader.action = 0;
|
|
fpUpdatePduHeader.secFlags = 0;
|
|
|
|
fpUpdateHeader.compression = 0;
|
|
fpUpdateHeader.compressionFlags = 0;
|
|
fpUpdateHeader.updateCode = updateCode;
|
|
fpUpdateHeader.size = (totalLength > maxLength) ? maxLength : totalLength;
|
|
|
|
pSrcData = pDstData = Stream_Pointer(s);
|
|
SrcSize = DstSize = fpUpdateHeader.size;
|
|
|
|
if (rdp->sec_flags & SEC_ENCRYPT)
|
|
fpUpdatePduHeader.secFlags |= FASTPATH_OUTPUT_ENCRYPTED;
|
|
if (rdp->sec_flags & SEC_SECURE_CHECKSUM)
|
|
fpUpdatePduHeader.secFlags |= FASTPATH_OUTPUT_SECURE_CHECKSUM;
|
|
|
|
if (settings->CompressionEnabled && !skipCompression)
|
|
{
|
|
if (bulk_compress(rdp->bulk, pSrcData, SrcSize, &pDstData, &DstSize, &compressionFlags) >= 0)
|
|
{
|
|
if (compressionFlags)
|
|
{
|
|
fpUpdateHeader.compressionFlags = compressionFlags;
|
|
fpUpdateHeader.compression = FASTPATH_OUTPUT_COMPRESSION_USED;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!fpUpdateHeader.compression)
|
|
{
|
|
pDstData = Stream_Pointer(s);
|
|
DstSize = fpUpdateHeader.size;
|
|
}
|
|
|
|
fpUpdateHeader.size = DstSize;
|
|
totalLength -= SrcSize;
|
|
|
|
if (totalLength == 0)
|
|
fpUpdateHeader.fragmentation = (fragment == 0) ? FASTPATH_FRAGMENT_SINGLE : FASTPATH_FRAGMENT_LAST;
|
|
else
|
|
fpUpdateHeader.fragmentation = (fragment == 0) ? FASTPATH_FRAGMENT_FIRST : FASTPATH_FRAGMENT_NEXT;
|
|
|
|
fpUpdateHeaderSize = fastpath_get_update_header_size(&fpUpdateHeader);
|
|
fpUpdatePduHeaderSize = fastpath_get_update_pdu_header_size(&fpUpdatePduHeader, rdp);
|
|
fpHeaderSize = fpUpdateHeaderSize + fpUpdatePduHeaderSize;
|
|
|
|
if (rdp->sec_flags & SEC_ENCRYPT)
|
|
{
|
|
pSignature = Stream_Buffer(fs) + 3;
|
|
|
|
if (rdp->settings->EncryptionMethods == ENCRYPTION_METHOD_FIPS)
|
|
{
|
|
pSignature += 4;
|
|
|
|
if ((pad = 8 - ((DstSize + fpUpdateHeaderSize) % 8)) == 8)
|
|
pad = 0;
|
|
|
|
fpUpdatePduHeader.fipsInformation[0] = 0x10;
|
|
fpUpdatePduHeader.fipsInformation[1] = 0x00;
|
|
fpUpdatePduHeader.fipsInformation[2] = 0x01;
|
|
fpUpdatePduHeader.fipsInformation[3] = pad;
|
|
}
|
|
}
|
|
|
|
fpUpdatePduHeader.length = fpUpdateHeader.size + fpHeaderSize + pad;
|
|
|
|
Stream_SetPosition(fs, 0);
|
|
fastpath_write_update_pdu_header(fs, &fpUpdatePduHeader, rdp);
|
|
fastpath_write_update_header(fs, &fpUpdateHeader);
|
|
Stream_Write(fs, pDstData, DstSize);
|
|
|
|
if (pad)
|
|
Stream_Zero(fs, pad);
|
|
|
|
if (rdp->sec_flags & SEC_ENCRYPT)
|
|
{
|
|
UINT32 dataSize = fpUpdateHeaderSize + DstSize + pad;
|
|
BYTE *data = Stream_Pointer(fs) - dataSize;
|
|
|
|
if (rdp->settings->EncryptionMethods == ENCRYPTION_METHOD_FIPS)
|
|
{
|
|
security_hmac_signature(data, dataSize - pad, pSignature, rdp);
|
|
security_fips_encrypt(data, dataSize, rdp);
|
|
}
|
|
else
|
|
{
|
|
if (rdp->sec_flags & SEC_SECURE_CHECKSUM)
|
|
security_salted_mac_signature(rdp, data, dataSize, TRUE, pSignature);
|
|
else
|
|
security_mac_signature(rdp, data, dataSize, pSignature);
|
|
|
|
security_encrypt(data, dataSize, rdp);
|
|
}
|
|
}
|
|
|
|
Stream_SealLength(fs);
|
|
|
|
if (transport_write(rdp->transport, fs) < 0)
|
|
{
|
|
status = FALSE;
|
|
break;
|
|
}
|
|
|
|
Stream_Seek(s, SrcSize);
|
|
}
|
|
|
|
rdp->sec_flags = 0;
|
|
|
|
return status;
|
|
}
|
|
|
|
rdpFastPath* fastpath_new(rdpRdp* rdp)
|
|
{
|
|
rdpFastPath* fastpath;
|
|
|
|
fastpath = (rdpFastPath*) calloc(1, sizeof(rdpFastPath));
|
|
if (!fastpath)
|
|
return NULL;
|
|
|
|
fastpath->rdp = rdp;
|
|
fastpath->fragmentation = -1;
|
|
fastpath->fs = Stream_New(NULL, FASTPATH_MAX_PACKET_SIZE);
|
|
if (!fastpath->fs)
|
|
goto out_free;
|
|
|
|
return fastpath;
|
|
|
|
out_free:
|
|
free(fastpath);
|
|
return NULL;
|
|
}
|
|
|
|
void fastpath_free(rdpFastPath* fastpath)
|
|
{
|
|
if (fastpath)
|
|
{
|
|
Stream_Free(fastpath->fs, TRUE);
|
|
free(fastpath);
|
|
}
|
|
}
|