codec/rfx: simplification, segfault/malloc fixes

rfx_process_message_sync:
- simplified the check if the header messages got processed

rfx_process_message_tileset:
- ObjectPool_Take result was not checked
- fail if TS_RFX_TILE block type is not CBT_TILE
- CreateThreadpoolWork result was not checked
- post decoding loop code segfaulted in error case

rfx_decoder_tile_new:
- missing malloc check

rfx_message_free:
- segfault protection

rfx_write_message_tileset:
- segfault protection
This commit is contained in:
Norbert Federa 2015-04-24 17:54:49 +02:00
parent 84577b1ca7
commit 5926bbcf48
2 changed files with 77 additions and 60 deletions

View File

@ -116,6 +116,13 @@ enum _RFX_STATE
}; };
typedef enum _RFX_STATE RFX_STATE; typedef enum _RFX_STATE RFX_STATE;
#define _RFX_DECODED_SYNC 0x00000001
#define _RFX_DECODED_CONTEXT 0x00000002
#define _RFX_DECODED_VERSIONS 0x00000004
#define _RFX_DECODED_CHANNELS 0x00000008
#define _RFX_DECODED_HEADERS 0x0000000F
struct _RFX_CONTEXT struct _RFX_CONTEXT
{ {
RFX_STATE state; RFX_STATE state;
@ -143,6 +150,9 @@ struct _RFX_CONTEXT
BYTE quantIdxCb; BYTE quantIdxCb;
BYTE quantIdxCr; BYTE quantIdxCr;
/* decoded header blocks */
UINT32 decodedHeaderBlocks;
/* routines */ /* routines */
void (*quantization_decode)(INT16* buffer, const UINT32* quantization_values); void (*quantization_decode)(INT16* buffer, const UINT32* quantization_values);
void (*quantization_encode)(INT16* buffer, const UINT32* quantization_values); void (*quantization_encode)(INT16* buffer, const UINT32* quantization_values);

View File

@ -3,6 +3,7 @@
* RemoteFX Codec Library * RemoteFX Codec Library
* *
* Copyright 2011 Vic Lee * Copyright 2011 Vic Lee
* Copyright 2015 Thincast Technologies GmbH
* Copyright 2015 Norbert Federa <norbert.federa@thincast.com> * Copyright 2015 Norbert Federa <norbert.federa@thincast.com>
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
@ -166,7 +167,11 @@ RFX_TILE* rfx_decoder_tile_new()
if (!(tile = (RFX_TILE*) calloc(1, sizeof(RFX_TILE)))) if (!(tile = (RFX_TILE*) calloc(1, sizeof(RFX_TILE))))
return NULL; return NULL;
tile->data = (BYTE*) malloc(4096 * 4); /* 64x64 * 4 */ if (!(tile->data = (BYTE*) malloc(4 * 64 * 64))) {
free(tile);
return NULL;
}
tile->allocated = TRUE; tile->allocated = TRUE;
return tile; return tile;
@ -421,7 +426,7 @@ static BOOL rfx_process_message_sync(RFX_CONTEXT* context, wStream* s)
{ {
UINT32 magic; UINT32 magic;
context->version = 0; context->decodedHeaderBlocks &= ~_RFX_DECODED_SYNC;
/* RFX_SYNC */ /* RFX_SYNC */
if (Stream_GetRemainingLength(s) < 6) if (Stream_GetRemainingLength(s) < 6)
@ -441,12 +446,13 @@ static BOOL rfx_process_message_sync(RFX_CONTEXT* context, wStream* s)
if (context->version != WF_VERSION_1_0) if (context->version != WF_VERSION_1_0)
{ {
WLog_ERR(TAG, "unknown version number 0x%X", context->version); WLog_ERR(TAG, "invalid version number 0x%04X", context->version);
return FALSE; return FALSE;
} }
WLog_Print(context->priv->log, WLOG_DEBUG, "version 0x%X", context->version); WLog_Print(context->priv->log, WLOG_DEBUG, "version 0x%X", context->version);
context->decodedHeaderBlocks |= _RFX_DECODED_SYNC;
return TRUE; return TRUE;
} }
@ -454,34 +460,39 @@ static BOOL rfx_process_message_codec_versions(RFX_CONTEXT* context, wStream* s)
{ {
BYTE numCodecs; BYTE numCodecs;
context->codec_id = 0; context->decodedHeaderBlocks &= ~_RFX_DECODED_VERSIONS;
context->codec_version = 0;
if (Stream_GetRemainingLength(s) < 1) if (Stream_GetRemainingLength(s) < 4)
{ {
WLog_ERR(TAG, "RfxCodecVersion packet too small"); WLog_ERR(TAG, "%s: packet too small for reading codec versions", __FUNCTION__);
return FALSE; return FALSE;
} }
Stream_Read_UINT8(s, numCodecs); /* numCodecs (1 byte), must be set to 0x01 */ Stream_Read_UINT8(s, numCodecs); /* numCodecs (1 byte), must be set to 0x01 */
Stream_Read_UINT8(s, context->codec_id); /* codecId (1 byte), must be set to 0x01 */
Stream_Read_UINT16(s, context->codec_version); /* version (2 bytes), must be set to WF_VERSION_1_0 (0x0100) */
if (numCodecs != 1) if (numCodecs != 1)
{ {
WLog_ERR(TAG, "numCodecs: %d, expected:1", numCodecs); WLog_ERR(TAG, "%s: numCodes is 0x%02X (must be 0x01)", __FUNCTION__, numCodecs);
return FALSE; return FALSE;
} }
if (Stream_GetRemainingLength(s) < (size_t) (2 * numCodecs)) if (context->codec_id != 0x01)
{ {
WLog_ERR(TAG, "RfxCodecVersion packet too small for numCodecs=%d", numCodecs); WLog_ERR(TAG, "%s: invalid codec id (0x%02X)", __FUNCTION__, context->codec_id);
return FALSE; return FALSE;
} }
/* RFX_CODEC_VERSIONT */ if (context->codec_version != WF_VERSION_1_0)
Stream_Read_UINT8(s, context->codec_id); /* codecId (1 byte) */ {
Stream_Read_UINT16(s, context->codec_version); /* version (2 bytes) */ WLog_ERR(TAG, "%s: invalid codec version (0x%04X)", __FUNCTION__, context->codec_version);
return FALSE;
}
WLog_Print(context->priv->log, WLOG_DEBUG, "id %d version 0x%X.", context->codec_id, context->codec_version); WLog_Print(context->priv->log, WLOG_DEBUG, "id %d version 0x%X.", context->codec_id, context->codec_version);
context->decodedHeaderBlocks |= _RFX_DECODED_VERSIONS;
return TRUE; return TRUE;
} }
@ -490,8 +501,7 @@ static BOOL rfx_process_message_channels(RFX_CONTEXT* context, wStream* s)
BYTE channelId; BYTE channelId;
BYTE numChannels; BYTE numChannels;
context->width = 0; context->decodedHeaderBlocks &= ~_RFX_DECODED_CHANNELS;
context->height = 0;
if (Stream_GetRemainingLength(s) < 1) if (Stream_GetRemainingLength(s) < 1)
{ {
@ -527,12 +537,19 @@ static BOOL rfx_process_message_channels(RFX_CONTEXT* context, wStream* s)
Stream_Read_UINT16(s, context->width); /* width (2 bytes) */ Stream_Read_UINT16(s, context->width); /* width (2 bytes) */
Stream_Read_UINT16(s, context->height); /* height (2 bytes) */ Stream_Read_UINT16(s, context->height); /* height (2 bytes) */
if (!context->width || !context->height)
{
WLog_ERR(TAG, "%s: invalid channel with/height: %ux%u", __FUNCTION__, context->width, context->height);
return FALSE;
}
/* Now, only the first monitor can be used, therefore the other channels will be ignored. */ /* Now, only the first monitor can be used, therefore the other channels will be ignored. */
Stream_Seek(s, 5 * (numChannels - 1)); Stream_Seek(s, 5 * (numChannels - 1));
WLog_Print(context->priv->log, WLOG_DEBUG, "numChannels %d id %d, %dx%d.", WLog_Print(context->priv->log, WLOG_DEBUG, "numChannels %d id %d, %dx%d.",
numChannels, channelId, context->width, context->height); numChannels, channelId, context->width, context->height);
context->decodedHeaderBlocks |= _RFX_DECODED_CHANNELS;
return TRUE; return TRUE;
} }
@ -542,6 +559,8 @@ static BOOL rfx_process_message_context(RFX_CONTEXT* context, wStream* s)
UINT16 tileSize; UINT16 tileSize;
UINT16 properties; UINT16 properties;
context->decodedHeaderBlocks &= ~_RFX_DECODED_CONTEXT;
if (Stream_GetRemainingLength(s) < 5) if (Stream_GetRemainingLength(s) < 5)
{ {
WLog_ERR(TAG, "RfxMessageContext packet too small"); WLog_ERR(TAG, "RfxMessageContext packet too small");
@ -584,6 +603,7 @@ static BOOL rfx_process_message_context(RFX_CONTEXT* context, wStream* s)
return FALSE; return FALSE;
} }
context->decodedHeaderBlocks |= _RFX_DECODED_CONTEXT;
return TRUE; return TRUE;
} }
@ -701,8 +721,8 @@ static BOOL rfx_process_message_region(RFX_CONTEXT* context, RFX_MESSAGE* messag
if (regionType != CBT_REGION) if (regionType != CBT_REGION)
{ {
WLog_ERR(TAG, "%s: invalid region type 0x%04X", __FUNCTION__, CBT_REGION); WLog_ERR(TAG, "%s: invalid region type 0x%04X", __FUNCTION__, regionType);
return FALSE; return TRUE;
} }
if (numTileSets != 0x0001) if (numTileSets != 0x0001)
@ -826,7 +846,10 @@ static BOOL rfx_process_message_tileset(RFX_CONTEXT* context, RFX_MESSAGE* messa
} }
if (!(message->tiles = (RFX_TILE**) calloc(message->numTiles, sizeof(RFX_TILE*)))) if (!(message->tiles = (RFX_TILE**) calloc(message->numTiles, sizeof(RFX_TILE*))))
{
message->numTiles = 0;
return FALSE; return FALSE;
}
if (context->priv->UseThreads) if (context->priv->UseThreads)
{ {
@ -851,7 +874,14 @@ static BOOL rfx_process_message_tileset(RFX_CONTEXT* context, RFX_MESSAGE* messa
rc = TRUE; rc = TRUE;
for (i = 0; i < message->numTiles; i++) for (i = 0; i < message->numTiles; i++)
{ {
tile = message->tiles[i] = (RFX_TILE*) ObjectPool_Take(context->priv->TilePool); if (!(tile = (RFX_TILE*) ObjectPool_Take(context->priv->TilePool)))
{
WLog_ERR(TAG, "RfxMessageTileSet failed to get tile from object pool");
rc = FALSE;
break;
}
message->tiles[i] = tile;
/* RFX_TILE */ /* RFX_TILE */
if (Stream_GetRemainingLength(s) < 6) if (Stream_GetRemainingLength(s) < 6)
@ -877,6 +907,7 @@ static BOOL rfx_process_message_tileset(RFX_CONTEXT* context, RFX_MESSAGE* messa
if (blockType != CBT_TILE) if (blockType != CBT_TILE)
{ {
WLog_ERR(TAG, "unknown block type 0x%X, expected CBT_TILE (0xCAC3).", blockType); WLog_ERR(TAG, "unknown block type 0x%X, expected CBT_TILE (0xCAC3).", blockType);
rc = FALSE;
break; break;
} }
@ -906,8 +937,13 @@ static BOOL rfx_process_message_tileset(RFX_CONTEXT* context, RFX_MESSAGE* messa
params[i].context = context; params[i].context = context;
params[i].tile = message->tiles[i]; params[i].tile = message->tiles[i];
work_objects[i] = CreateThreadpoolWork((PTP_WORK_CALLBACK) rfx_process_message_tile_work_callback, if (!(work_objects[i] = CreateThreadpoolWork((PTP_WORK_CALLBACK) rfx_process_message_tile_work_callback,
(void*) &params[i], &context->priv->ThreadPoolEnv); (void*) &params[i], &context->priv->ThreadPoolEnv)))
{
WLog_ERR(TAG, "CreateThreadpoolWork failed.");
rc = FALSE;
break;
}
SubmitThreadpoolWork(work_objects[i]); SubmitThreadpoolWork(work_objects[i]);
close_cnt = i + 1; close_cnt = i + 1;
@ -927,14 +963,15 @@ static BOOL rfx_process_message_tileset(RFX_CONTEXT* context, RFX_MESSAGE* messa
WaitForThreadpoolWorkCallbacks(work_objects[i], FALSE); WaitForThreadpoolWorkCallbacks(work_objects[i], FALSE);
CloseThreadpoolWork(work_objects[i]); CloseThreadpoolWork(work_objects[i]);
} }
}
free(work_objects); free(work_objects);
free(params); free(params);
}
for (i = 0; i < message->numTiles; i++) for (i = 0; i < message->numTiles; i++)
{ {
tile = message->tiles[i]; if (!(tile = message->tiles[i]))
continue;
tile->YLen = tile->CbLen = tile->CrLen = 0; tile->YLen = tile->CbLen = tile->CrLen = 0;
tile->YData = tile->CbData = tile->CrData = NULL; tile->YData = tile->CbData = tile->CrData = NULL;
} }
@ -982,44 +1019,12 @@ RFX_MESSAGE* rfx_process_message(RFX_CONTEXT* context, BYTE* data, UINT32 length
pos = Stream_GetPosition(s) - 6 + blockLen; pos = Stream_GetPosition(s) - 6 + blockLen;
if (blockType != WBT_SYNC && context->version != WF_VERSION_1_0) if (blockType > WBT_CONTEXT && context->decodedHeaderBlocks != _RFX_DECODED_HEADERS)
{ {
/* The TS_RFX_SYNC message MUST be the first message in any encoded stream. */ WLog_ERR(TAG, "%s: incomplete header blocks processing", __FUNCTION__);
WLog_ERR(TAG, "%s: TS_RFX_SYNC was not processed", __FUNCTION__);
goto fail; goto fail;
} }
if (blockType > WBT_CONTEXT)
{
if (context->mode == 0)
{
/* Unknown rlgr mode.
* A valid TS_RFX_CONTEXT message should have been processed by now
*/
WLog_ERR(TAG, "%s: TS_RFX_CONTEXT was not processed", __FUNCTION__);
goto fail;
}
if (context->codec_version != WF_VERSION_1_0)
{
/* Unknown codec version.
* A valid TS_RFX_CODEC_VERSIONS message should have been processed by now
*/
WLog_ERR(TAG, "%s: TS_RFX_CODEC_VERSIONS was not processed", __FUNCTION__);
goto fail;
}
if (!context->width || !context->height)
{
/* Empty plane.
* A valid TS_RFX_CODEC_CHANNELS message should have been processed by now
* which would have resulted in a non-zero width and height in the context
*/
WLog_ERR(TAG, "%s: TS_RFX_CODEC_CHANNELS was not processed", __FUNCTION__);
goto fail;
}
}
if (blockType >= WBT_CONTEXT && blockType <= WBT_EXTENSION) if (blockType >= WBT_CONTEXT && blockType <= WBT_EXTENSION)
{ {
/* RFX_CODEC_CHANNELT */ /* RFX_CODEC_CHANNELT */
@ -1162,7 +1167,8 @@ void rfx_message_free(RFX_CONTEXT* context, RFX_MESSAGE* message)
{ {
for (i = 0; i < message->numTiles; i++) for (i = 0; i < message->numTiles; i++)
{ {
tile = message->tiles[i]; if (!(tile = message->tiles[i]))
continue;
if (tile->YCbCrData) if (tile->YCbCrData)
{ {
@ -1687,7 +1693,8 @@ static BOOL rfx_write_message_tileset(RFX_CONTEXT* context, wStream* s, RFX_MESS
for (i = 0; i < message->numTiles; i++) for (i = 0; i < message->numTiles; i++)
{ {
tile = message->tiles[i]; if (!(tile = message->tiles[i]))
return FALSE;
if (!rfx_write_tile(context, s, tile)) if (!rfx_write_tile(context, s, tile))
return FALSE; return FALSE;
} }