Files
FreeRDP/libfreerdp/core/tpkt.c

175 lines
4.1 KiB
C
Raw Normal View History

/**
2012-10-08 23:02:04 -04:00
* FreeRDP: A Remote Desktop Protocol Implementation
* Transport Packets (TPKTs)
*
* 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.
*/
2022-02-16 11:20:38 +01:00
#include <freerdp/config.h>
#include "tpdu.h"
#include "tpkt.h"
#include <winpr/wlog.h>
#define TAG FREERDP_TAG("core.tpkt")
/**
* TPKTs are defined in:
*
* http://tools.ietf.org/html/rfc1006/
* RFC 1006 - ISO Transport Service on top of the TCP
*
* http://www.itu.int/rec/T-REC-T.123/
* ITU-T T.123 (01/2007) - Network-specific data protocol stacks for multimedia conferencing
*
* TPKT Header
* ____________________ byte
* | |
* | 3 (version) | 1
* |____________________|
* | |
* | Reserved | 2
* |____________________|
* | |
* | Length (MSB) | 3
* |____________________|
* | |
* | Length (LSB) | 4
* |____________________|
* | |
* | X.224 TPDU | 5 - ?
* ....
*
* A TPKT header is of fixed length 4, and the following X.224 TPDU is at least three bytes long.
* Therefore, the minimum TPKT length is 7, and the maximum TPKT length is 65535. Because the TPKT
* length includes the TPKT header (4 bytes), the maximum X.224 TPDU length is 65531.
*/
/**
2022-12-09 14:35:03 +01:00
* Verify if a packet has valid TPKT header.
*
* @param s A stream to read from
*
* @return \b TRUE for success, \b FALSE otherwise
*/
int tpkt_verify_header(wStream* s)
{
BYTE version = 0;
if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
return -1;
2013-05-08 16:27:21 -04:00
Stream_Peek_UINT8(s, version);
2012-11-13 11:06:33 -05:00
if (version == 3)
return 1;
else
return 0;
}
/**
2022-12-09 14:35:03 +01:00
* Read a TPKT header.
*
* @param s A stream to read from
* @param length A pointer to the result, must not be nullptr
2022-12-09 14:35:03 +01:00
*
* @return \b TRUE for success, \b FALSE otherwise
*/
BOOL tpkt_read_header(wStream* s, UINT16* length)
{
BYTE version = 0;
if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
return FALSE;
2013-05-08 16:27:21 -04:00
Stream_Peek_UINT8(s, version);
if (version == 3)
{
UINT16 len = 0;
if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
return FALSE;
Stream_Seek(s, 2);
Stream_Read_UINT16_BE(s, len);
/* ITU-T Rec. T.123 8 Packet header to delimit data units in an octet stream */
if (len < 7)
{
WLog_ERR(TAG, "TPKT header too short, require minimum of 7 bytes, got %" PRId16, len);
return FALSE;
}
if (!Stream_CheckAndLogRequiredLength(TAG, s, len - 4))
{
WLog_ERR(TAG, "TPKT header length %" PRIu16 ", but received less", len);
return FALSE;
}
*length = len;
}
else
{
/* not a TPKT header */
*length = 0;
}
return TRUE;
}
2025-03-17 10:39:48 +01:00
BOOL tpkt_ensure_stream_consumed_(wLog* log, wStream* s, size_t length, const char* fkt)
{
if (length > UINT16_MAX)
{
WLog_Print(log, WLOG_ERROR, "[%s] length %" PRIuz " > %d", fkt, length, UINT16_MAX);
return FALSE;
}
size_t rem = Stream_GetRemainingLength(s);
if (rem > 0)
{
2025-03-17 10:39:48 +01:00
WLog_Print(log, WLOG_ERROR,
"[%s] Received invalid TPKT header length %" PRIuz ", %" PRIuz
2025-03-17 10:39:48 +01:00
" bytes too long!",
fkt, length, rem);
return FALSE;
}
return TRUE;
}
/**
2022-12-09 14:35:03 +01:00
* Write a TPKT header.
*
* @param s A stream to write to
* @param length The value to write
*
* @return \b TRUE for success, \b FALSE otherwise
*/
BOOL tpkt_write_header(wStream* s, size_t length)
{
2023-01-24 12:19:56 +01:00
if (!Stream_CheckAndLogRequiredCapacity(TAG, (s), 4))
return FALSE;
2019-11-06 15:24:51 +01:00
Stream_Write_UINT8(s, 3); /* version */
Stream_Write_UINT8(s, 0); /* reserved */
WINPR_ASSERT(length <= UINT16_MAX);
Stream_Write_UINT16_BE(s, (UINT16)length); /* length */
return TRUE;
}