Blame channels/cliprdr/cliprdr_common.c

Packit Service 5a9772
/**
Packit Service 5a9772
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service 5a9772
 * Cliprdr common
Packit Service 5a9772
 *
Packit Service 5a9772
 * Copyright 2013 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service 5a9772
 * Copyright 2015 Thincast Technologies GmbH
Packit Service 5a9772
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
Packit Service 5a9772
 * Copyright 2019 Kobi Mizrachi <kmizrachi18@gmail.com>
Packit Service 5a9772
 *
Packit Service 5a9772
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service 5a9772
 * you may not use this file except in compliance with the License.
Packit Service 5a9772
 * You may obtain a copy of the License at
Packit Service 5a9772
 *
Packit Service 5a9772
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service 5a9772
 *
Packit Service 5a9772
 * Unless required by applicable law or agreed to in writing, software
Packit Service 5a9772
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service 5a9772
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service 5a9772
 * See the License for the specific language governing permissions and
Packit Service 5a9772
 * limitations under the License.
Packit Service 5a9772
 */
Packit Service 5a9772
Packit Service 5a9772
#include <winpr/crt.h>
Packit Service 5a9772
#include <winpr/stream.h>
Packit Service 5a9772
#include <freerdp/channels/log.h>
Packit Service 5a9772
Packit Service 5a9772
#define TAG CHANNELS_TAG("cliprdr.common")
Packit Service 5a9772
Packit Service 5a9772
#include "cliprdr_common.h"
Packit Service 5a9772
Packit Service 5a9772
static BOOL cliprdr_validate_file_contents_request(const CLIPRDR_FILE_CONTENTS_REQUEST* request)
Packit Service 5a9772
{
Packit Service 5a9772
	/*
Packit Service 5a9772
	 * [MS-RDPECLIP] 2.2.5.3 File Contents Request PDU (CLIPRDR_FILECONTENTS_REQUEST).
Packit Service 5a9772
	 *
Packit Service 5a9772
	 * A request for the size of the file identified by the lindex field. The size MUST be
Packit Service 5a9772
	 * returned as a 64-bit, unsigned integer. The cbRequested field MUST be set to
Packit Service 5a9772
	 * 0x00000008 and both the nPositionLow and nPositionHigh fields MUST be
Packit Service 5a9772
	 * set to 0x00000000.
Packit Service 5a9772
	 */
Packit Service 5a9772
Packit Service 5a9772
	if (request->dwFlags & FILECONTENTS_SIZE)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (request->cbRequested != sizeof(UINT64))
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "[%s]: cbRequested must be %" PRIu32 ", got %" PRIu32 "", __FUNCTION__,
Packit Service 5a9772
			         sizeof(UINT64), request->cbRequested);
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		if (request->nPositionHigh != 0 || request->nPositionLow != 0)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "[%s]: nPositionHigh and nPositionLow must be set to 0", __FUNCTION__);
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
		}
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	return TRUE;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
wStream* cliprdr_packet_new(UINT16 msgType, UINT16 msgFlags, UINT32 dataLen)
Packit Service 5a9772
{
Packit Service 5a9772
	wStream* s;
Packit Service 5a9772
	s = Stream_New(NULL, dataLen + 8);
Packit Service 5a9772
Packit Service 5a9772
	if (!s)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "Stream_New failed!");
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	Stream_Write_UINT16(s, msgType);
Packit Service 5a9772
	Stream_Write_UINT16(s, msgFlags);
Packit Service 5a9772
	/* Write actual length after the entire packet has been constructed. */
Packit Service 5a9772
	Stream_Seek(s, 4);
Packit Service 5a9772
	return s;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static void cliprdr_write_file_contents_request(wStream* s,
Packit Service 5a9772
                                                const CLIPRDR_FILE_CONTENTS_REQUEST* request)
Packit Service 5a9772
{
Packit Service 5a9772
	Stream_Write_UINT32(s, request->streamId);      /* streamId (4 bytes) */
Packit Service 5a9772
	Stream_Write_UINT32(s, request->listIndex);     /* listIndex (4 bytes) */
Packit Service 5a9772
	Stream_Write_UINT32(s, request->dwFlags);       /* dwFlags (4 bytes) */
Packit Service 5a9772
	Stream_Write_UINT32(s, request->nPositionLow);  /* nPositionLow (4 bytes) */
Packit Service 5a9772
	Stream_Write_UINT32(s, request->nPositionHigh); /* nPositionHigh (4 bytes) */
Packit Service 5a9772
	Stream_Write_UINT32(s, request->cbRequested);   /* cbRequested (4 bytes) */
Packit Service 5a9772
Packit Service 5a9772
	if (request->haveClipDataId)
Packit Service 5a9772
		Stream_Write_UINT32(s, request->clipDataId); /* clipDataId (4 bytes) */
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static INLINE void cliprdr_write_lock_unlock_clipdata(wStream* s, UINT32 clipDataId)
Packit Service 5a9772
{
Packit Service 5a9772
	Stream_Write_UINT32(s, clipDataId);
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static void cliprdr_write_lock_clipdata(wStream* s,
Packit Service 5a9772
                                        const CLIPRDR_LOCK_CLIPBOARD_DATA* lockClipboardData)
Packit Service 5a9772
{
Packit Service 5a9772
	cliprdr_write_lock_unlock_clipdata(s, lockClipboardData->clipDataId);
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static void cliprdr_write_unlock_clipdata(wStream* s,
Packit Service 5a9772
                                          const CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
Packit Service 5a9772
{
Packit Service 5a9772
	cliprdr_write_lock_unlock_clipdata(s, unlockClipboardData->clipDataId);
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static void cliprdr_write_file_contents_response(wStream* s,
Packit Service 5a9772
                                                 const CLIPRDR_FILE_CONTENTS_RESPONSE* response)
Packit Service 5a9772
{
Packit Service 5a9772
	Stream_Write_UINT32(s, response->streamId); /* streamId (4 bytes) */
Packit Service 5a9772
	Stream_Write(s, response->requestedData, response->cbRequested);
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
wStream* cliprdr_packet_lock_clipdata_new(const CLIPRDR_LOCK_CLIPBOARD_DATA* lockClipboardData)
Packit Service 5a9772
{
Packit Service 5a9772
	wStream* s;
Packit Service 5a9772
Packit Service 5a9772
	if (!lockClipboardData)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	s = cliprdr_packet_new(CB_LOCK_CLIPDATA, 0, 4);
Packit Service 5a9772
Packit Service 5a9772
	if (!s)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	cliprdr_write_lock_clipdata(s, lockClipboardData);
Packit Service 5a9772
	return s;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
wStream*
Packit Service 5a9772
cliprdr_packet_unlock_clipdata_new(const CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
Packit Service 5a9772
{
Packit Service 5a9772
	wStream* s;
Packit Service 5a9772
Packit Service 5a9772
	if (!unlockClipboardData)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	s = cliprdr_packet_new(CB_LOCK_CLIPDATA, 0, 4);
Packit Service 5a9772
Packit Service 5a9772
	if (!s)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	cliprdr_write_unlock_clipdata(s, unlockClipboardData);
Packit Service 5a9772
	return s;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
wStream* cliprdr_packet_file_contents_request_new(const CLIPRDR_FILE_CONTENTS_REQUEST* request)
Packit Service 5a9772
{
Packit Service 5a9772
	wStream* s;
Packit Service 5a9772
Packit Service 5a9772
	if (!request)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	s = cliprdr_packet_new(CB_FILECONTENTS_REQUEST, 0, 28);
Packit Service 5a9772
Packit Service 5a9772
	if (!s)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	cliprdr_write_file_contents_request(s, request);
Packit Service 5a9772
	return s;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
wStream* cliprdr_packet_file_contents_response_new(const CLIPRDR_FILE_CONTENTS_RESPONSE* response)
Packit Service 5a9772
{
Packit Service 5a9772
	wStream* s;
Packit Service 5a9772
Packit Service 5a9772
	if (!response)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	s = cliprdr_packet_new(CB_FILECONTENTS_RESPONSE, response->msgFlags, 4 + response->cbRequested);
Packit Service 5a9772
Packit Service 5a9772
	if (!s)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	cliprdr_write_file_contents_response(s, response);
Packit Service 5a9772
	return s;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
wStream* cliprdr_packet_format_list_new(const CLIPRDR_FORMAT_LIST* formatList,
Packit Service 5a9772
                                        BOOL useLongFormatNames)
Packit Service 5a9772
{
Packit Service 5a9772
	wStream* s;
Packit Service 5a9772
	UINT32 index;
Packit Service 5a9772
	int cchWideChar;
Packit Service 5a9772
	LPWSTR lpWideCharStr;
Packit Service 5a9772
	int formatNameSize;
Packit Service 5a9772
	char* szFormatName;
Packit Service 5a9772
	WCHAR* wszFormatName;
Packit Service 5a9772
	BOOL asciiNames = FALSE;
Packit Service 5a9772
	CLIPRDR_FORMAT* format;
Packit Service 5a9772
Packit Service 5a9772
	if (formatList->msgType != CB_FORMAT_LIST)
Packit Service 5a9772
		WLog_WARN(TAG, "[%s] called with invalid type %08" PRIx32, __FUNCTION__,
Packit Service 5a9772
		          formatList->msgType);
Packit Service 5a9772
Packit Service 5a9772
	if (!useLongFormatNames)
Packit Service 5a9772
	{
Packit Service 5a9772
		UINT32 length = formatList->numFormats * 36;
Packit Service 5a9772
		s = cliprdr_packet_new(CB_FORMAT_LIST, 0, length);
Packit Service 5a9772
Packit Service 5a9772
		if (!s)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "cliprdr_packet_new failed!");
Packit Service 5a9772
			return NULL;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		for (index = 0; index < formatList->numFormats; index++)
Packit Service 5a9772
		{
Packit Service 5a9772
			size_t formatNameLength = 0;
Packit Service 5a9772
			format = (CLIPRDR_FORMAT*)&(formatList->formats[index]);
Packit Service 5a9772
			Stream_Write_UINT32(s, format->formatId); /* formatId (4 bytes) */
Packit Service 5a9772
			formatNameSize = 0;
Packit Service 5a9772
Packit Service 5a9772
			szFormatName = format->formatName;
Packit Service 5a9772
Packit Service 5a9772
			if (asciiNames)
Packit Service 5a9772
			{
Packit Service 5a9772
				if (szFormatName)
Packit Service 5a9772
					formatNameLength = strnlen(szFormatName, 32);
Packit Service 5a9772
Packit Service 5a9772
				if (formatNameLength > 31)
Packit Service 5a9772
					formatNameLength = 31;
Packit Service 5a9772
Packit Service 5a9772
				Stream_Write(s, szFormatName, formatNameLength);
Packit Service 5a9772
				Stream_Zero(s, 32 - formatNameLength);
Packit Service 5a9772
			}
Packit Service 5a9772
			else
Packit Service 5a9772
			{
Packit Service 5a9772
				wszFormatName = NULL;
Packit Service 5a9772
Packit Service 5a9772
				if (szFormatName)
Packit Service 5a9772
					formatNameSize =
Packit Service 5a9772
					    ConvertToUnicode(CP_UTF8, 0, szFormatName, -1, &wszFormatName, 0);
Packit Service 5a9772
Packit Service 5a9772
				if (formatNameSize < 0)
Packit Service 5a9772
					return NULL;
Packit Service 5a9772
Packit Service 5a9772
				if (formatNameSize > 15)
Packit Service 5a9772
					formatNameSize = 15;
Packit Service 5a9772
Packit Service 5a9772
				/* size in bytes  instead of wchar */
Packit Service 5a9772
				formatNameSize *= 2;
Packit Service 5a9772
Packit Service 5a9772
				if (wszFormatName)
Packit Service 5a9772
					Stream_Write(s, wszFormatName, (size_t)formatNameSize);
Packit Service 5a9772
Packit Service 5a9772
				Stream_Zero(s, (size_t)(32 - formatNameSize));
Packit Service 5a9772
				free(wszFormatName);
Packit Service 5a9772
			}
Packit Service 5a9772
		}
Packit Service 5a9772
	}
Packit Service 5a9772
	else
Packit Service 5a9772
	{
Packit Service 5a9772
		UINT32 length = 0;
Packit Service 5a9772
		for (index = 0; index < formatList->numFormats; index++)
Packit Service 5a9772
		{
Packit Service 5a9772
			format = (CLIPRDR_FORMAT*)&(formatList->formats[index]);
Packit Service 5a9772
			length += 4;
Packit Service 5a9772
			formatNameSize = 2;
Packit Service 5a9772
Packit Service 5a9772
			if (format->formatName)
Packit Service 5a9772
				formatNameSize =
Packit Service 5a9772
				    MultiByteToWideChar(CP_UTF8, 0, format->formatName, -1, NULL, 0) * 2;
Packit Service 5a9772
Packit Service 5a9772
			if (formatNameSize < 0)
Packit Service 5a9772
				return NULL;
Packit Service 5a9772
Packit Service 5a9772
			length += (UINT32)formatNameSize;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		s = cliprdr_packet_new(CB_FORMAT_LIST, 0, length);
Packit Service 5a9772
Packit Service 5a9772
		if (!s)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "cliprdr_packet_new failed!");
Packit Service 5a9772
			return NULL;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		for (index = 0; index < formatList->numFormats; index++)
Packit Service 5a9772
		{
Packit Service 5a9772
			format = (CLIPRDR_FORMAT*)&(formatList->formats[index]);
Packit Service 5a9772
			Stream_Write_UINT32(s, format->formatId); /* formatId (4 bytes) */
Packit Service 5a9772
Packit Service 5a9772
			if (format->formatName)
Packit Service 5a9772
			{
Packit Service 5a9772
				const size_t cap = Stream_Capacity(s);
Packit Service 5a9772
				const size_t pos = Stream_GetPosition(s);
Packit Service 5a9772
				const size_t rem = cap - pos;
Packit Service 5a9772
				if ((cap < pos) || ((rem / 2) > INT_MAX))
Packit Service 5a9772
					return NULL;
Packit Service 5a9772
Packit Service 5a9772
				lpWideCharStr = (LPWSTR)Stream_Pointer(s);
Packit Service 5a9772
				cchWideChar = (int)(rem / 2);
Packit Service 5a9772
				formatNameSize = MultiByteToWideChar(CP_UTF8, 0, format->formatName, -1,
Packit Service 5a9772
				                                     lpWideCharStr, cchWideChar) *
Packit Service 5a9772
				                 2;
Packit Service 5a9772
				if (formatNameSize < 0)
Packit Service 5a9772
					return NULL;
Packit Service 5a9772
				Stream_Seek(s, (size_t)formatNameSize);
Packit Service 5a9772
			}
Packit Service 5a9772
			else
Packit Service 5a9772
			{
Packit Service 5a9772
				Stream_Write_UINT16(s, 0);
Packit Service 5a9772
			}
Packit Service 5a9772
		}
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	return s;
Packit Service 5a9772
}
Packit Service 5a9772
UINT cliprdr_read_unlock_clipdata(wStream* s, CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
Packit Service 5a9772
{
Packit Service 5a9772
	if (Stream_GetRemainingLength(s) < 4)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "not enough remaining data");
Packit Service 5a9772
		return ERROR_INVALID_DATA;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	Stream_Read_UINT32(s, unlockClipboardData->clipDataId); /* clipDataId (4 bytes) */
Packit Service 5a9772
	return CHANNEL_RC_OK;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
UINT cliprdr_read_format_data_request(wStream* s, CLIPRDR_FORMAT_DATA_REQUEST* request)
Packit Service 5a9772
{
Packit Service 5a9772
	if (Stream_GetRemainingLength(s) < 4)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "not enough data in stream!");
Packit Service 5a9772
		return ERROR_INVALID_DATA;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	Stream_Read_UINT32(s, request->requestedFormatId); /* requestedFormatId (4 bytes) */
Packit Service 5a9772
	return CHANNEL_RC_OK;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
UINT cliprdr_read_format_data_response(wStream* s, CLIPRDR_FORMAT_DATA_RESPONSE* response)
Packit Service 5a9772
{
Packit Service 5a9772
	response->requestedFormatData = NULL;
Packit Service 5a9772
Packit Service 5a9772
	if (Stream_GetRemainingLength(s) < response->dataLen)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "not enough data in stream!");
Packit Service 5a9772
		return ERROR_INVALID_DATA;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (response->dataLen)
Packit Service 5a9772
		response->requestedFormatData = Stream_Pointer(s);
Packit Service 5a9772
Packit Service 5a9772
	return CHANNEL_RC_OK;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
UINT cliprdr_read_file_contents_request(wStream* s, CLIPRDR_FILE_CONTENTS_REQUEST* request)
Packit Service 5a9772
{
Packit Service 5a9772
	if (Stream_GetRemainingLength(s) < 24)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "not enough remaining data");
Packit Service 5a9772
		return ERROR_INVALID_DATA;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	request->haveClipDataId = FALSE;
Packit Service 5a9772
	Stream_Read_UINT32(s, request->streamId);      /* streamId (4 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, request->listIndex);     /* listIndex (4 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, request->dwFlags);       /* dwFlags (4 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, request->nPositionLow);  /* nPositionLow (4 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, request->nPositionHigh); /* nPositionHigh (4 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, request->cbRequested);   /* cbRequested (4 bytes) */
Packit Service 5a9772
Packit Service 5a9772
	if (Stream_GetRemainingLength(s) >= 4)
Packit Service 5a9772
	{
Packit Service 5a9772
		Stream_Read_UINT32(s, request->clipDataId); /* clipDataId (4 bytes) */
Packit Service 5a9772
		request->haveClipDataId = TRUE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (!cliprdr_validate_file_contents_request(request))
Packit Service 5a9772
		return ERROR_BAD_ARGUMENTS;
Packit Service 5a9772
Packit Service 5a9772
	return CHANNEL_RC_OK;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
UINT cliprdr_read_file_contents_response(wStream* s, CLIPRDR_FILE_CONTENTS_RESPONSE* response)
Packit Service 5a9772
{
Packit Service 5a9772
	if (Stream_GetRemainingLength(s) < 4)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "not enough remaining data");
Packit Service 5a9772
		return ERROR_INVALID_DATA;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	Stream_Read_UINT32(s, response->streamId);   /* streamId (4 bytes) */
Packit Service 5a9772
	response->requestedData = Stream_Pointer(s); /* requestedFileContentsData */
Packit Service 5a9772
	response->cbRequested = response->dataLen - 4;
Packit Service 5a9772
	return CHANNEL_RC_OK;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
UINT cliprdr_read_format_list(wStream* s, CLIPRDR_FORMAT_LIST* formatList, BOOL useLongFormatNames)
Packit Service 5a9772
{
Packit Service 5a9772
	UINT32 index;
Packit Service 5a9772
	BOOL asciiNames;
Packit Service 5a9772
	int formatNameLength;
Packit Service 5a9772
	char* szFormatName;
Packit Service 5a9772
	WCHAR* wszFormatName;
Packit Service 5a9772
	wStream sub1, sub2;
Packit Service 5a9772
	CLIPRDR_FORMAT* formats = NULL;
Packit Service 5a9772
	UINT error = ERROR_INTERNAL_ERROR;
Packit Service 5a9772
Packit Service 5a9772
	asciiNames = (formatList->msgFlags & CB_ASCII_NAMES) ? TRUE : FALSE;
Packit Service 5a9772
Packit Service 5a9772
	index = 0;
Packit Service 5a9772
	/* empty format list */
Packit Service 5a9772
	formatList->formats = NULL;
Packit Service 5a9772
	formatList->numFormats = 0;
Packit Service 5a9772
Packit Service 5a9772
	Stream_StaticInit(&sub1, Stream_Pointer(s), formatList->dataLen);
Packit Service 5a9772
	if (!Stream_SafeSeek(s, formatList->dataLen))
Packit Service 5a9772
		return ERROR_INVALID_DATA;
Packit Service 5a9772
Packit Service 5a9772
	if (!formatList->dataLen)
Packit Service 5a9772
	{
Packit Service 5a9772
	}
Packit Service 5a9772
	else if (!useLongFormatNames)
Packit Service 5a9772
	{
Packit Service 5a9772
		const size_t cap = Stream_Capacity(&sub1;;
Packit Service 5a9772
		formatList->numFormats = (cap / 36);
Packit Service 5a9772
Packit Service 5a9772
		if ((formatList->numFormats * 36) != cap)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "Invalid short format list length: %" PRIuz "", cap);
Packit Service 5a9772
			return ERROR_INTERNAL_ERROR;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		if (formatList->numFormats)
Packit Service 5a9772
			formats = (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT));
Packit Service 5a9772
Packit Service 5a9772
		if (!formats)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "calloc failed!");
Packit Service 5a9772
			return CHANNEL_RC_NO_MEMORY;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		formatList->formats = formats;
Packit Service 5a9772
Packit Service 5a9772
		while (Stream_GetRemainingLength(&sub1) >= 4)
Packit Service 5a9772
		{
Packit Service 5a9772
			Stream_Read_UINT32(&sub1, formats[index].formatId); /* formatId (4 bytes) */
Packit Service 5a9772
Packit Service 5a9772
			formats[index].formatName = NULL;
Packit Service 5a9772
Packit Service 5a9772
			/* According to MS-RDPECLIP 2.2.3.1.1.1 formatName is "a 32-byte block containing
Packit Service 5a9772
			 * the *null-terminated* name assigned to the Clipboard Format: (32 ASCII 8 characters
Packit Service 5a9772
			 * or 16 Unicode characters)"
Packit Service 5a9772
			 * However, both Windows RDSH and mstsc violate this specs as seen in the following
Packit Service 5a9772
			 * example of a transferred short format name string: [R.i.c.h. .T.e.x.t. .F.o.r.m.a.t.]
Packit Service 5a9772
			 * These are 16 unicode charaters - *without* terminating null !
Packit Service 5a9772
			 */
Packit Service 5a9772
Packit Service 5a9772
			szFormatName = (char*)Stream_Pointer(&sub1;;
Packit Service 5a9772
			wszFormatName = (WCHAR*)Stream_Pointer(&sub1;;
Packit Service 5a9772
			if (!Stream_SafeSeek(&sub1, 32))
Packit Service 5a9772
				goto error_out;
Packit Service 5a9772
			if (asciiNames)
Packit Service 5a9772
			{
Packit Service 5a9772
				if (szFormatName[0])
Packit Service 5a9772
				{
Packit Service 5a9772
					/* ensure null termination */
Packit Service 5a9772
					formats[index].formatName = (char*)malloc(32 + 1);
Packit Service 5a9772
					if (!formats[index].formatName)
Packit Service 5a9772
					{
Packit Service 5a9772
						WLog_ERR(TAG, "malloc failed!");
Packit Service 5a9772
						error = CHANNEL_RC_NO_MEMORY;
Packit Service 5a9772
						goto error_out;
Packit Service 5a9772
					}
Packit Service 5a9772
					CopyMemory(formats[index].formatName, szFormatName, 32);
Packit Service 5a9772
					formats[index].formatName[32] = '\0';
Packit Service 5a9772
				}
Packit Service 5a9772
			}
Packit Service 5a9772
			else
Packit Service 5a9772
			{
Packit Service 5a9772
				if (wszFormatName[0])
Packit Service 5a9772
				{
Packit Service 5a9772
					/* ConvertFromUnicode always returns a null-terminated
Packit Service 5a9772
					 * string on success, even if the source string isn't.
Packit Service 5a9772
					 */
Packit Service 5a9772
					if (ConvertFromUnicode(CP_UTF8, 0, wszFormatName, 16,
Packit Service 5a9772
					                       &(formats[index].formatName), 0, NULL, NULL) < 1)
Packit Service 5a9772
					{
Packit Service 5a9772
						WLog_ERR(TAG, "failed to convert short clipboard format name");
Packit Service 5a9772
						error = ERROR_INTERNAL_ERROR;
Packit Service 5a9772
						goto error_out;
Packit Service 5a9772
					}
Packit Service 5a9772
				}
Packit Service 5a9772
			}
Packit Service 5a9772
Packit Service 5a9772
			index++;
Packit Service 5a9772
		}
Packit Service 5a9772
	}
Packit Service 5a9772
	else
Packit Service 5a9772
	{
Packit Service 5a9772
		sub2 = sub1;
Packit Service 5a9772
		while (Stream_GetRemainingLength(&sub1) > 0)
Packit Service 5a9772
		{
Packit Service 5a9772
			size_t rest;
Packit Service 5a9772
			if (!Stream_SafeSeek(&sub1, 4)) /* formatId (4 bytes) */
Packit Service 5a9772
				goto error_out;
Packit Service 5a9772
Packit Service 5a9772
			wszFormatName = (WCHAR*)Stream_Pointer(&sub1;;
Packit Service 5a9772
			rest = Stream_GetRemainingLength(&sub1;;
Packit Service 5a9772
			formatNameLength = _wcsnlen(wszFormatName, rest / sizeof(WCHAR));
Packit Service 5a9772
Packit Service 5a9772
			if (!Stream_SafeSeek(&sub1, (formatNameLength + 1) * sizeof(WCHAR)))
Packit Service 5a9772
				goto error_out;
Packit Service 5a9772
			formatList->numFormats++;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		if (formatList->numFormats)
Packit Service 5a9772
			formats = (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT));
Packit Service 5a9772
Packit Service 5a9772
		if (!formats)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_ERR(TAG, "calloc failed!");
Packit Service 5a9772
			return CHANNEL_RC_NO_MEMORY;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		formatList->formats = formats;
Packit Service 5a9772
Packit Service 5a9772
		while (Stream_GetRemainingLength(&sub2) >= 4)
Packit Service 5a9772
		{
Packit Service 5a9772
			size_t rest;
Packit Service 5a9772
			Stream_Read_UINT32(&sub2, formats[index].formatId); /* formatId (4 bytes) */
Packit Service 5a9772
Packit Service 5a9772
			formats[index].formatName = NULL;
Packit Service 5a9772
Packit Service 5a9772
			wszFormatName = (WCHAR*)Stream_Pointer(&sub2;;
Packit Service 5a9772
			rest = Stream_GetRemainingLength(&sub2;;
Packit Service 5a9772
			formatNameLength = _wcsnlen(wszFormatName, rest / sizeof(WCHAR));
Packit Service 5a9772
			if (!Stream_SafeSeek(&sub2, (formatNameLength + 1) * sizeof(WCHAR)))
Packit Service 5a9772
				goto error_out;
Packit Service 5a9772
Packit Service 5a9772
			if (formatNameLength)
Packit Service 5a9772
			{
Packit Service 5a9772
				if (ConvertFromUnicode(CP_UTF8, 0, wszFormatName, formatNameLength,
Packit Service 5a9772
				                       &(formats[index].formatName), 0, NULL, NULL) < 1)
Packit Service 5a9772
				{
Packit Service 5a9772
					WLog_ERR(TAG, "failed to convert long clipboard format name");
Packit Service 5a9772
					error = ERROR_INTERNAL_ERROR;
Packit Service 5a9772
					goto error_out;
Packit Service 5a9772
				}
Packit Service 5a9772
			}
Packit Service 5a9772
Packit Service 5a9772
			index++;
Packit Service 5a9772
		}
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	return CHANNEL_RC_OK;
Packit Service 5a9772
Packit Service 5a9772
error_out:
Packit Service 5a9772
	cliprdr_free_format_list(formatList);
Packit Service 5a9772
	return error;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
void cliprdr_free_format_list(CLIPRDR_FORMAT_LIST* formatList)
Packit Service 5a9772
{
Packit Service 5a9772
	UINT index = 0;
Packit Service 5a9772
Packit Service 5a9772
	if (formatList == NULL)
Packit Service 5a9772
		return;
Packit Service 5a9772
Packit Service 5a9772
	if (formatList->formats)
Packit Service 5a9772
	{
Packit Service 5a9772
		for (index = 0; index < formatList->numFormats; index++)
Packit Service 5a9772
		{
Packit Service 5a9772
			free(formatList->formats[index].formatName);
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		free(formatList->formats);
Packit Service 5a9772
		formatList->formats = NULL;
Packit Service 5a9772
		formatList->numFormats = 0;
Packit Service 5a9772
	}
Packit Service 5a9772
}