Blame libfreerdp/core/window.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Windowing Alternate Secondary Orders
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 * Copyright 2011 Roman Barabanov <romanbarabanov@gmail.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
Packit 1fb8d4
#include "window.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("core.window")
Packit 1fb8d4
Packit Service 5a9772
static void update_free_window_icon_info(ICON_INFO* iconInfo);
Packit Service 5a9772
Packit 1fb8d4
BOOL rail_read_unicode_string(wStream* s, RAIL_UNICODE_STRING* unicode_string)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT16 new_len;
Packit 1fb8d4
	BYTE* new_str;
Packit 1fb8d4
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 2)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	Stream_Read_UINT16(s, new_len); /* cbString (2 bytes) */
Packit 1fb8d4
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < new_len)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!new_len)
Packit 1fb8d4
	{
Packit 1fb8d4
		free(unicode_string->string);
Packit 1fb8d4
		unicode_string->string = NULL;
Packit 1fb8d4
		unicode_string->length = 0;
Packit 1fb8d4
		return TRUE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	new_str = (BYTE*)realloc(unicode_string->string, new_len);
Packit 1fb8d4
Packit 1fb8d4
	if (!new_str)
Packit 1fb8d4
	{
Packit 1fb8d4
		free(unicode_string->string);
Packit 1fb8d4
		unicode_string->string = NULL;
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	unicode_string->string = new_str;
Packit 1fb8d4
	unicode_string->length = new_len;
Packit 1fb8d4
	Stream_Read(s, unicode_string->string, unicode_string->length);
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
BOOL utf8_string_to_rail_string(const char* string, RAIL_UNICODE_STRING* unicode_string)
Packit Service 5a9772
{
Packit Service 5a9772
	WCHAR* buffer = NULL;
Packit Service 5a9772
	int length = 0;
Packit Service 5a9772
	free(unicode_string->string);
Packit Service 5a9772
	unicode_string->string = NULL;
Packit Service 5a9772
	unicode_string->length = 0;
Packit Service 5a9772
Packit Service 5a9772
	if (!string || strlen(string) < 1)
Packit Service 5a9772
		return TRUE;
Packit Service 5a9772
Packit Service 5a9772
	length = ConvertToUnicode(CP_UTF8, 0, string, -1, &buffer, 0);
Packit Service 5a9772
Packit Service 5a9772
	if ((length < 0) || ((size_t)length * sizeof(WCHAR) > UINT16_MAX))
Packit Service 5a9772
	{
Packit Service 5a9772
		free(buffer);
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	unicode_string->string = (BYTE*)buffer;
Packit Service 5a9772
	unicode_string->length = (UINT16)length * sizeof(WCHAR);
Packit Service 5a9772
	return TRUE;
Packit Service 5a9772
}
Packit Service 5a9772
Packit 1fb8d4
/* See [MS-RDPERP] 2.2.1.2.3 Icon Info (TS_ICON_INFO) */
Packit 1fb8d4
static BOOL update_read_icon_info(wStream* s, ICON_INFO* iconInfo)
Packit 1fb8d4
{
Packit Service 5a9772
	BYTE* newBitMask = NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	Stream_Read_UINT16(s, iconInfo->cacheEntry); /* cacheEntry (2 bytes) */
Packit Service 5a9772
	Stream_Read_UINT8(s, iconInfo->cacheId);     /* cacheId (1 byte) */
Packit Service 5a9772
	Stream_Read_UINT8(s, iconInfo->bpp);         /* bpp (1 byte) */
Packit 1fb8d4
Packit 1fb8d4
	if ((iconInfo->bpp < 1) || (iconInfo->bpp > 32))
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_ERR(TAG, "invalid bpp value %" PRIu32 "", iconInfo->bpp);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	Stream_Read_UINT16(s, iconInfo->width);  /* width (2 bytes) */
Packit 1fb8d4
	Stream_Read_UINT16(s, iconInfo->height); /* height (2 bytes) */
Packit 1fb8d4
Packit 1fb8d4
	/* cbColorTable is only present when bpp is 1, 4 or 8 */
Packit 1fb8d4
	switch (iconInfo->bpp)
Packit 1fb8d4
	{
Packit 1fb8d4
		case 1:
Packit 1fb8d4
		case 4:
Packit 1fb8d4
		case 8:
Packit 1fb8d4
			if (Stream_GetRemainingLength(s) < 2)
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
Packit 1fb8d4
			Stream_Read_UINT16(s, iconInfo->cbColorTable); /* cbColorTable (2 bytes) */
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			iconInfo->cbColorTable = 0;
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit Service 5a9772
	Stream_Read_UINT16(s, iconInfo->cbBitsMask);  /* cbBitsMask (2 bytes) */
Packit 1fb8d4
	Stream_Read_UINT16(s, iconInfo->cbBitsColor); /* cbBitsColor (2 bytes) */
Packit 1fb8d4
Packit 1fb8d4
	/* bitsMask */
Packit Service 5a9772
	if (iconInfo->cbBitsMask > 0)
Packit Service 5a9772
	{
Packit Service 5a9772
		newBitMask = (BYTE*)realloc(iconInfo->bitsMask, iconInfo->cbBitsMask);
Packit 1fb8d4
Packit Service 5a9772
		if (!newBitMask)
Packit Service 5a9772
		{
Packit Service 5a9772
			free(iconInfo->bitsMask);
Packit Service 5a9772
			iconInfo->bitsMask = NULL;
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		iconInfo->bitsMask = newBitMask;
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < iconInfo->cbBitsMask)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
		Stream_Read(s, iconInfo->bitsMask, iconInfo->cbBitsMask);
Packit Service 5a9772
	}
Packit Service 5a9772
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		free(iconInfo->bitsMask);
Packit 1fb8d4
		iconInfo->bitsMask = NULL;
Packit Service 5a9772
		iconInfo->cbBitsMask = 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* colorTable */
Packit Service 5a9772
	if (iconInfo->cbColorTable > 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		BYTE* new_tab;
Packit Service 5a9772
		new_tab = (BYTE*)realloc(iconInfo->colorTable, iconInfo->cbColorTable);
Packit 1fb8d4
Packit 1fb8d4
		if (!new_tab)
Packit 1fb8d4
		{
Packit 1fb8d4
			free(iconInfo->colorTable);
Packit 1fb8d4
			iconInfo->colorTable = NULL;
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		iconInfo->colorTable = new_tab;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		free(iconInfo->colorTable);
Packit 1fb8d4
		iconInfo->colorTable = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (iconInfo->colorTable)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < iconInfo->cbColorTable)
Packit Service 5a9772
			return FALSE;
Packit 1fb8d4
		Stream_Read(s, iconInfo->colorTable, iconInfo->cbColorTable);
Packit Service 5a9772
	}
Packit 1fb8d4
Packit 1fb8d4
	/* bitsColor */
Packit Service 5a9772
	if (iconInfo->cbBitsColor > 0)
Packit Service 5a9772
	{
Packit Service 5a9772
		newBitMask = (BYTE*)realloc(iconInfo->bitsColor, iconInfo->cbBitsColor);
Packit 1fb8d4
Packit Service 5a9772
		if (!newBitMask)
Packit Service 5a9772
		{
Packit Service 5a9772
			free(iconInfo->bitsColor);
Packit Service 5a9772
			iconInfo->bitsColor = NULL;
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		iconInfo->bitsColor = newBitMask;
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < iconInfo->cbBitsColor)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
		Stream_Read(s, iconInfo->bitsColor, iconInfo->cbBitsColor);
Packit Service 5a9772
	}
Packit Service 5a9772
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		free(iconInfo->bitsColor);
Packit 1fb8d4
		iconInfo->bitsColor = NULL;
Packit Service 5a9772
		iconInfo->cbBitsColor = 0;
Packit 1fb8d4
	}
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_read_cached_icon_info(wStream* s, CACHED_ICON_INFO* cachedIconInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 3)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	Stream_Read_UINT16(s, cachedIconInfo->cacheEntry); /* cacheEntry (2 bytes) */
Packit Service 5a9772
	Stream_Read_UINT8(s, cachedIconInfo->cacheId);     /* cacheId (1 byte) */
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_read_notify_icon_infotip(wStream* s, NOTIFY_ICON_INFOTIP* notifyIconInfoTip)
Packit 1fb8d4
{
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit Service 5a9772
	Stream_Read_UINT32(s, notifyIconInfoTip->timeout);              /* timeout (4 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, notifyIconInfoTip->flags);                /* infoFlags (4 bytes) */
Packit 1fb8d4
	return rail_read_unicode_string(s, &notifyIconInfoTip->text) && /* infoTipText */
Packit Service 5a9772
	       rail_read_unicode_string(s, &notifyIconInfoTip->title);  /* title */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_read_window_state_order(wStream* s, WINDOW_ORDER_INFO* orderInfo,
Packit Service 5a9772
                                           WINDOW_STATE_ORDER* windowState)
Packit 1fb8d4
{
Packit Service 5a9772
	UINT32 i;
Packit Service 5a9772
	size_t size;
Packit 1fb8d4
	RECTANGLE_16* newRect;
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_OWNER)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT32(s, windowState->ownerWindowId); /* ownerWindowId (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_STYLE)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->style);         /* style (4 bytes) */
Packit 1fb8d4
		Stream_Read_UINT32(s, windowState->extendedStyle); /* extendedStyle (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_SHOW)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 1)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT8(s, windowState->showState); /* showState (1 byte) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_TITLE)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!rail_read_unicode_string(s, &windowState->titleInfo)) /* titleInfo */
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET)
Packit 1fb8d4
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_INT32(s, windowState->clientOffsetX); /* clientOffsetX (4 bytes) */
Packit Service 5a9772
		Stream_Read_INT32(s, windowState->clientOffsetY); /* clientOffsetY (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE)
Packit 1fb8d4
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->clientAreaWidth);  /* clientAreaWidth (4 bytes) */
Packit 1fb8d4
		Stream_Read_UINT32(s, windowState->clientAreaHeight); /* clientAreaHeight (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_X)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 8)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->resizeMarginLeft);
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->resizeMarginRight);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_Y)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 8)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->resizeMarginTop);
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->resizeMarginBottom);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RP_CONTENT)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 1)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT8(s, windowState->RPContent); /* RPContent (1 byte) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_ROOT_PARENT)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->rootParentHandle); /* rootParentHandle (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_INT32(s, windowState->windowOffsetX); /* windowOffsetX (4 bytes) */
Packit Service 5a9772
		Stream_Read_INT32(s, windowState->windowOffsetY); /* windowOffsetY (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_INT32(s, windowState->windowClientDeltaX); /* windowClientDeltaX (4 bytes) */
Packit Service 5a9772
		Stream_Read_INT32(s, windowState->windowClientDeltaY); /* windowClientDeltaY (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_UINT32(s, windowState->windowWidth);  /* windowWidth (4 bytes) */
Packit 1fb8d4
		Stream_Read_UINT32(s, windowState->windowHeight); /* windowHeight (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 2)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT16(s, windowState->numWindowRects); /* numWindowRects (2 bytes) */
Packit 1fb8d4
Packit 1fb8d4
		if (windowState->numWindowRects == 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		size = sizeof(RECTANGLE_16) * windowState->numWindowRects;
Packit 1fb8d4
		newRect = (RECTANGLE_16*)realloc(windowState->windowRects, size);
Packit 1fb8d4
Packit 1fb8d4
		if (!newRect)
Packit 1fb8d4
		{
Packit 1fb8d4
			free(windowState->windowRects);
Packit 1fb8d4
			windowState->windowRects = NULL;
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		windowState->windowRects = newRect;
Packit 1fb8d4
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 8 * windowState->numWindowRects)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		/* windowRects */
Packit Service 5a9772
		for (i = 0; i < windowState->numWindowRects; i++)
Packit 1fb8d4
		{
Packit Service 5a9772
			Stream_Read_UINT16(s, windowState->windowRects[i].left);   /* left (2 bytes) */
Packit Service 5a9772
			Stream_Read_UINT16(s, windowState->windowRects[i].top);    /* top (2 bytes) */
Packit Service 5a9772
			Stream_Read_UINT16(s, windowState->windowRects[i].right);  /* right (2 bytes) */
Packit 1fb8d4
			Stream_Read_UINT16(s, windowState->windowRects[i].bottom); /* bottom (2 bytes) */
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET)
Packit 1fb8d4
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT32(s, windowState->visibleOffsetX); /* visibleOffsetX (4 bytes) */
Packit 1fb8d4
		Stream_Read_UINT32(s, windowState->visibleOffsetY); /* visibleOffsetY (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 2)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT16(s, windowState->numVisibilityRects); /* numVisibilityRects (2 bytes) */
Packit 1fb8d4
Packit Service 5a9772
		if (windowState->numVisibilityRects != 0)
Packit 1fb8d4
		{
Packit Service 5a9772
			size = sizeof(RECTANGLE_16) * windowState->numVisibilityRects;
Packit Service 5a9772
			newRect = (RECTANGLE_16*)realloc(windowState->visibilityRects, size);
Packit Service 5a9772
Packit Service 5a9772
			if (!newRect)
Packit Service 5a9772
			{
Packit Service 5a9772
				free(windowState->visibilityRects);
Packit Service 5a9772
				windowState->visibilityRects = NULL;
Packit Service 5a9772
				return FALSE;
Packit Service 5a9772
			}
Packit Service 5a9772
Packit Service 5a9772
			windowState->visibilityRects = newRect;
Packit Service 5a9772
Packit Service 5a9772
			if (Stream_GetRemainingLength(s) < windowState->numVisibilityRects * 8)
Packit Service 5a9772
				return FALSE;
Packit Service 5a9772
Packit Service 5a9772
			/* visibilityRects */
Packit Service 5a9772
			for (i = 0; i < windowState->numVisibilityRects; i++)
Packit Service 5a9772
			{
Packit Service 5a9772
				Stream_Read_UINT16(s, windowState->visibilityRects[i].left);  /* left (2 bytes) */
Packit Service 5a9772
				Stream_Read_UINT16(s, windowState->visibilityRects[i].top);   /* top (2 bytes) */
Packit Service 5a9772
				Stream_Read_UINT16(s, windowState->visibilityRects[i].right); /* right (2 bytes) */
Packit Service 5a9772
				Stream_Read_UINT16(s,
Packit Service 5a9772
				                   windowState->visibilityRects[i].bottom); /* bottom (2 bytes) */
Packit Service 5a9772
			}
Packit 1fb8d4
		}
Packit Service 5a9772
	}
Packit 1fb8d4
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_OVERLAY_DESCRIPTION)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (!rail_read_unicode_string(s, &windowState->OverlayDescription))
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
	}
Packit 1fb8d4
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_ICON_OVERLAY_NULL)
Packit Service 5a9772
	{
Packit Service 5a9772
		/* no data to be read here */
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_TASKBAR_BUTTON)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 1)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_UINT8(s, windowState->TaskbarButton);
Packit Service 5a9772
	}
Packit 1fb8d4
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_ENFORCE_SERVER_ZORDER)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 1)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit Service 5a9772
		Stream_Read_UINT8(s, windowState->EnforceServerZOrder);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_APPBAR_STATE)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 1)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
Packit Service 5a9772
		Stream_Read_UINT8(s, windowState->AppBarState);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_APPBAR_EDGE)
Packit Service 5a9772
	{
Packit Service 5a9772
		if (Stream_GetRemainingLength(s) < 1)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
Packit Service 5a9772
		Stream_Read_UINT8(s, windowState->AppBarEdge);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_read_window_icon_order(wStream* s, WINDOW_ORDER_INFO* orderInfo,
Packit Service 5a9772
                                          WINDOW_ICON_ORDER* window_icon)
Packit 1fb8d4
{
Packit Service 5a9772
	WINPR_UNUSED(orderInfo);
Packit Service 5a9772
	window_icon->iconInfo = (ICON_INFO*)calloc(1, sizeof(ICON_INFO));
Packit 1fb8d4
Packit 1fb8d4
	if (!window_icon->iconInfo)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return update_read_icon_info(s, window_icon->iconInfo); /* iconInfo (ICON_INFO) */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_read_window_cached_icon_order(wStream* s, WINDOW_ORDER_INFO* orderInfo,
Packit Service 5a9772
                                                 WINDOW_CACHED_ICON_ORDER* window_cached_icon)
Packit 1fb8d4
{
Packit Service 5a9772
	WINPR_UNUSED(orderInfo);
Packit Service 5a9772
	return update_read_cached_icon_info(
Packit Service 5a9772
	    s, &window_cached_icon->cachedIcon); /* cachedIcon (CACHED_ICON_INFO) */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void update_read_window_delete_order(wStream* s, WINDOW_ORDER_INFO* orderInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	/* window deletion event */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL window_order_supported(const rdpSettings* settings, UINT32 fieldFlags)
Packit 1fb8d4
{
Packit 1fb8d4
	const UINT32 mask = (WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE | WINDOW_ORDER_FIELD_RP_CONTENT |
Packit 1fb8d4
	                     WINDOW_ORDER_FIELD_ROOT_PARENT);
Packit 1fb8d4
	BOOL dresult;
Packit 1fb8d4
Packit 1fb8d4
	if (!settings)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	/* See [MS-RDPERP] 2.2.1.1.2 Window List Capability Set */
Packit 1fb8d4
	dresult = settings->AllowUnanouncedOrdersFromServer;
Packit 1fb8d4
Packit 1fb8d4
	switch (settings->RemoteWndSupportLevel)
Packit 1fb8d4
	{
Packit 1fb8d4
		case WINDOW_LEVEL_SUPPORTED_EX:
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
Packit 1fb8d4
		case WINDOW_LEVEL_SUPPORTED:
Packit 1fb8d4
			return ((fieldFlags & mask) == 0) || dresult;
Packit 1fb8d4
Packit 1fb8d4
		case WINDOW_LEVEL_NOT_SUPPORTED:
Packit 1fb8d4
			return dresult;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return dresult;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
#define DUMP_APPEND(buffer, size, ...)            \
Packit Service 5a9772
	do                                            \
Packit Service 5a9772
	{                                             \
Packit Service 5a9772
		char* b = (buffer);                       \
Packit Service 5a9772
		size_t s = (size);                        \
Packit Service 5a9772
		size_t pos = strnlen(b, s);               \
Packit Service 5a9772
		_snprintf(&b[pos], s - pos, __VA_ARGS__); \
Packit Service 5a9772
	} while (0)
Packit Service 5a9772
Packit Service 5a9772
static void dump_window_state_order(wLog* log, const char* msg, const WINDOW_ORDER_INFO* order,
Packit Service 5a9772
                                    const WINDOW_STATE_ORDER* state)
Packit Service 5a9772
{
Packit Service 5a9772
	char buffer[3000] = { 0 };
Packit Service 5a9772
	const size_t bufferSize = sizeof(buffer) - 1;
Packit Service 5a9772
Packit Service 5a9772
	_snprintf(buffer, bufferSize, "%s windowId=0x%" PRIu32 "", msg, order->windowId);
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_OWNER)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " owner=0x%" PRIx32 "", state->ownerWindowId);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_STYLE)
Packit Service 5a9772
	{
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " [ex]style=<0x%" PRIx32 ", 0x%" PRIx32 "", state->style,
Packit Service 5a9772
		            state->extendedStyle);
Packit Service 5a9772
		if (state->style & WS_POPUP)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " popup");
Packit Service 5a9772
		if (state->style & WS_VISIBLE)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " visible");
Packit Service 5a9772
		if (state->style & WS_THICKFRAME)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " thickframe");
Packit Service 5a9772
		if (state->style & WS_BORDER)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " border");
Packit Service 5a9772
		if (state->style & WS_CAPTION)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " caption");
Packit Service 5a9772
Packit Service 5a9772
		if (state->extendedStyle & WS_EX_NOACTIVATE)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " noactivate");
Packit Service 5a9772
		if (state->extendedStyle & WS_EX_TOOLWINDOW)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " toolWindow");
Packit Service 5a9772
		if (state->extendedStyle & WS_EX_TOPMOST)
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, " topMost");
Packit Service 5a9772
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, ">");
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_SHOW)
Packit Service 5a9772
	{
Packit Service 5a9772
		const char* showStr;
Packit Service 5a9772
		switch (state->showState)
Packit Service 5a9772
		{
Packit Service 5a9772
			case 0:
Packit Service 5a9772
				showStr = "hidden";
Packit Service 5a9772
				break;
Packit Service 5a9772
			case 2:
Packit Service 5a9772
				showStr = "minimized";
Packit Service 5a9772
				break;
Packit Service 5a9772
			case 3:
Packit Service 5a9772
				showStr = "maximized";
Packit Service 5a9772
				break;
Packit Service 5a9772
			case 5:
Packit Service 5a9772
				showStr = "current";
Packit Service 5a9772
				break;
Packit Service 5a9772
			default:
Packit Service 5a9772
				showStr = "<unknown>";
Packit Service 5a9772
				break;
Packit Service 5a9772
		}
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " show=%s", showStr);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_TITLE)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " title");
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " clientOffset=(%" PRId32 ",%" PRId32 ")",
Packit Service 5a9772
		            state->clientOffsetX, state->clientOffsetY);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " clientAreaWidth=%" PRIu32 " clientAreaHeight=%" PRIu32 "",
Packit Service 5a9772
		            state->clientAreaWidth, state->clientAreaHeight);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_X)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize,
Packit Service 5a9772
		            " resizeMarginLeft=%" PRIu32 " resizeMarginRight=%" PRIu32 "",
Packit Service 5a9772
		            state->resizeMarginLeft, state->resizeMarginRight);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_Y)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize,
Packit Service 5a9772
		            " resizeMarginTop=%" PRIu32 " resizeMarginBottom=%" PRIu32 "",
Packit Service 5a9772
		            state->resizeMarginTop, state->resizeMarginBottom);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_RP_CONTENT)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " rpContent=0x%" PRIx32 "", state->RPContent);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_ROOT_PARENT)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " rootParent=0x%" PRIx32 "", state->rootParentHandle);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " windowOffset=(%" PRId32 ",%" PRId32 ")",
Packit Service 5a9772
		            state->windowOffsetX, state->windowOffsetY);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " windowClientDelta=(%" PRId32 ",%" PRId32 ")",
Packit Service 5a9772
		            state->windowClientDeltaX, state->windowClientDeltaY);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " windowWidth=%" PRIu32 " windowHeight=%" PRIu32 "",
Packit Service 5a9772
		            state->windowWidth, state->windowHeight);
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS)
Packit Service 5a9772
	{
Packit Service 5a9772
		UINT32 i;
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " windowRects=(");
Packit Service 5a9772
		for (i = 0; i < state->numWindowRects; i++)
Packit Service 5a9772
		{
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, "(%" PRIu16 ",%" PRIu16 ",%" PRIu16 ",%" PRIu16 ")",
Packit Service 5a9772
			            state->windowRects[i].left, state->windowRects[i].top,
Packit Service 5a9772
			            state->windowRects[i].right, state->windowRects[i].bottom);
Packit Service 5a9772
		}
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, ")");
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " visibleOffset=(%" PRId32 ",%" PRId32 ")",
Packit Service 5a9772
		            state->visibleOffsetX, state->visibleOffsetY);
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY)
Packit Service 5a9772
	{
Packit Service 5a9772
		UINT32 i;
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " visibilityRects=(");
Packit Service 5a9772
		for (i = 0; i < state->numVisibilityRects; i++)
Packit Service 5a9772
		{
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, "(%" PRIu16 ",%" PRIu16 ",%" PRIu16 ",%" PRIu16 ")",
Packit Service 5a9772
			            state->visibilityRects[i].left, state->visibilityRects[i].top,
Packit Service 5a9772
			            state->visibilityRects[i].right, state->visibilityRects[i].bottom);
Packit Service 5a9772
		}
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, ")");
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_OVERLAY_DESCRIPTION)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " overlayDescr");
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_ICON_OVERLAY_NULL)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " iconOverlayNull");
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_TASKBAR_BUTTON)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " taskBarButton=0x%" PRIx8 "", state->TaskbarButton);
Packit Service 5a9772
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_ENFORCE_SERVER_ZORDER)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " enforceServerZOrder=0x%" PRIx8 "",
Packit Service 5a9772
		            state->EnforceServerZOrder);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_APPBAR_STATE)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " appBarState=0x%" PRIx8 "", state->AppBarState);
Packit Service 5a9772
	if (order->fieldFlags & WINDOW_ORDER_FIELD_APPBAR_EDGE)
Packit Service 5a9772
	{
Packit Service 5a9772
		const char* appBarEdgeStr;
Packit Service 5a9772
		switch (state->AppBarEdge)
Packit Service 5a9772
		{
Packit Service 5a9772
			case 0:
Packit Service 5a9772
				appBarEdgeStr = "left";
Packit Service 5a9772
				break;
Packit Service 5a9772
			case 1:
Packit Service 5a9772
				appBarEdgeStr = "top";
Packit Service 5a9772
				break;
Packit Service 5a9772
			case 2:
Packit Service 5a9772
				appBarEdgeStr = "right";
Packit Service 5a9772
				break;
Packit Service 5a9772
			case 3:
Packit Service 5a9772
				appBarEdgeStr = "bottom";
Packit Service 5a9772
				break;
Packit Service 5a9772
			default:
Packit Service 5a9772
				appBarEdgeStr = "<unknown>";
Packit Service 5a9772
				break;
Packit Service 5a9772
		}
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " appBarEdge=%s", appBarEdgeStr);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	WLog_Print(log, WLOG_DEBUG, buffer);
Packit Service 5a9772
}
Packit Service 5a9772
Packit 1fb8d4
static BOOL update_recv_window_info_order(rdpUpdate* update, wStream* s,
Packit Service 5a9772
                                          WINDOW_ORDER_INFO* orderInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpContext* context = update->context;
Packit 1fb8d4
	rdpWindowUpdate* window = update->window;
Packit 1fb8d4
	BOOL result = TRUE;
Packit 1fb8d4
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	Stream_Read_UINT32(s, orderInfo->windowId); /* windowId (4 bytes) */
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_ICON)
Packit 1fb8d4
	{
Packit Service 5a9772
		WINDOW_ICON_ORDER window_icon = { 0 };
Packit Service 5a9772
		result = update_read_window_icon_order(s, orderInfo, &window_icon);
Packit Service 5a9772
Packit Service 5a9772
		if (result)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_Print(update->log, WLOG_DEBUG, "WindowIcon windowId=0x%" PRIx32 "",
Packit Service 5a9772
			           orderInfo->windowId);
Packit Service 5a9772
			IFCALLRET(window->WindowIcon, result, context, orderInfo, &window_icon);
Packit Service 5a9772
		}
Packit 1fb8d4
Packit Service 5a9772
		update_free_window_icon_info(window_icon.iconInfo);
Packit Service 5a9772
		free(window_icon.iconInfo);
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (orderInfo->fieldFlags & WINDOW_ORDER_CACHED_ICON)
Packit 1fb8d4
	{
Packit Service 5a9772
		WINDOW_CACHED_ICON_ORDER window_cached_icon = { 0 };
Packit Service 5a9772
		result = update_read_window_cached_icon_order(s, orderInfo, &window_cached_icon);
Packit 1fb8d4
Packit Service 5a9772
		if (result)
Packit Service 5a9772
		{
Packit Service 5a9772
			WLog_Print(update->log, WLOG_DEBUG, "WindowCachedIcon windowId=0x%" PRIx32 "",
Packit Service 5a9772
			           orderInfo->windowId);
Packit Service 5a9772
			IFCALLRET(window->WindowCachedIcon, result, context, orderInfo, &window_cached_icon);
Packit Service 5a9772
		}
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (orderInfo->fieldFlags & WINDOW_ORDER_STATE_DELETED)
Packit 1fb8d4
	{
Packit 1fb8d4
		update_read_window_delete_order(s, orderInfo);
Packit Service 5a9772
		WLog_Print(update->log, WLOG_DEBUG, "WindowDelete windowId=0x%" PRIx32 "",
Packit Service 5a9772
		           orderInfo->windowId);
Packit 1fb8d4
		IFCALLRET(window->WindowDelete, result, context, orderInfo);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit Service 5a9772
		WINDOW_STATE_ORDER windowState = { 0 };
Packit Service 5a9772
		result = update_read_window_state_order(s, orderInfo, &windowState);
Packit 1fb8d4
Packit Service 5a9772
		if (result)
Packit 1fb8d4
		{
Packit Service 5a9772
			if (orderInfo->fieldFlags & WINDOW_ORDER_STATE_NEW)
Packit Service 5a9772
			{
Packit Service 5a9772
				dump_window_state_order(update->log, "WindowCreate", orderInfo, &windowState);
Packit Service 5a9772
				IFCALLRET(window->WindowCreate, result, context, orderInfo, &windowState);
Packit Service 5a9772
			}
Packit Service 5a9772
			else
Packit Service 5a9772
			{
Packit Service 5a9772
				dump_window_state_order(update->log, "WindowUpdate", orderInfo, &windowState);
Packit Service 5a9772
				IFCALLRET(window->WindowUpdate, result, context, orderInfo, &windowState);
Packit Service 5a9772
			}
Packit Service 5a9772
Packit Service 5a9772
			update_free_window_state(&windowState);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return result;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static void update_notify_icon_state_order_free(NOTIFY_ICON_STATE_ORDER* notify)
Packit Service 5a9772
{
Packit Service 5a9772
	free(notify->toolTip.string);
Packit Service 5a9772
	free(notify->infoTip.text.string);
Packit Service 5a9772
	free(notify->infoTip.title.string);
Packit Service 5a9772
	update_free_window_icon_info(&notify->icon);
Packit Service 5a9772
	memset(notify, 0, sizeof(NOTIFY_ICON_STATE_ORDER));
Packit Service 5a9772
}
Packit Service 5a9772
Packit 1fb8d4
static BOOL update_read_notification_icon_state_order(wStream* s, WINDOW_ORDER_INFO* orderInfo,
Packit Service 5a9772
                                                      NOTIFY_ICON_STATE_ORDER* notify_icon_state)
Packit 1fb8d4
{
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_VERSION)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT32(s, notify_icon_state->version); /* version (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_TIP)
Packit 1fb8d4
	{
Packit Service 5a9772
		if (!rail_read_unicode_string(s,
Packit Service 5a9772
		                              &notify_icon_state->toolTip)) /* toolTip (UNICODE_STRING) */
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_INFO_TIP)
Packit 1fb8d4
	{
Packit Service 5a9772
		if (!update_read_notify_icon_infotip(
Packit Service 5a9772
		        s, &notify_icon_state->infoTip)) /* infoTip (NOTIFY_ICON_INFOTIP) */
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_STATE)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT32(s, notify_icon_state->state); /* state (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_ICON)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!update_read_icon_info(s, &notify_icon_state->icon)) /* icon (ICON_INFO) */
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_CACHED_ICON)
Packit 1fb8d4
	{
Packit Service 5a9772
		if (!update_read_cached_icon_info(
Packit Service 5a9772
		        s, &notify_icon_state->cachedIcon)) /* cachedIcon (CACHED_ICON_INFO) */
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void update_read_notification_icon_delete_order(wStream* s, WINDOW_ORDER_INFO* orderInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	/* notification icon deletion event */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_recv_notification_icon_info_order(rdpUpdate* update, wStream* s,
Packit Service 5a9772
                                                     WINDOW_ORDER_INFO* orderInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpContext* context = update->context;
Packit 1fb8d4
	rdpWindowUpdate* window = update->window;
Packit 1fb8d4
	BOOL result = TRUE;
Packit 1fb8d4
Packit 1fb8d4
	if (Stream_GetRemainingLength(s) < 8)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit Service 5a9772
	Stream_Read_UINT32(s, orderInfo->windowId);     /* windowId (4 bytes) */
Packit 1fb8d4
	Stream_Read_UINT32(s, orderInfo->notifyIconId); /* notifyIconId (4 bytes) */
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_STATE_DELETED)
Packit 1fb8d4
	{
Packit 1fb8d4
		update_read_notification_icon_delete_order(s, orderInfo);
Packit 1fb8d4
		WLog_Print(update->log, WLOG_DEBUG, "NotifyIconDelete");
Packit 1fb8d4
		IFCALLRET(window->NotifyIconDelete, result, context, orderInfo);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit Service 5a9772
		NOTIFY_ICON_STATE_ORDER notify_icon_state = { 0 };
Packit Service 5a9772
		result = update_read_notification_icon_state_order(s, orderInfo, &notify_icon_state);
Packit Service 5a9772
Packit Service 5a9772
		if (!result)
Packit Service 5a9772
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		if (orderInfo->fieldFlags & WINDOW_ORDER_STATE_NEW)
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(update->log, WLOG_DEBUG, "NotifyIconCreate");
Packit Service 5a9772
			IFCALLRET(window->NotifyIconCreate, result, context, orderInfo, &notify_icon_state);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(update->log, WLOG_DEBUG, "NotifyIconUpdate");
Packit Service 5a9772
			IFCALLRET(window->NotifyIconUpdate, result, context, orderInfo, &notify_icon_state);
Packit 1fb8d4
		}
Packit Service 5a9772
	fail:
Packit Service 5a9772
		update_notify_icon_state_order_free(&notify_icon_state);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return result;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL update_read_desktop_actively_monitored_order(wStream* s, WINDOW_ORDER_INFO* orderInfo,
Packit Service 5a9772
                                                         MONITORED_DESKTOP_ORDER* monitored_desktop)
Packit 1fb8d4
{
Packit 1fb8d4
	int i;
Packit 1fb8d4
	int size;
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 4)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT32(s, monitored_desktop->activeWindowId); /* activeWindowId (4 bytes) */
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ZORDER)
Packit 1fb8d4
	{
Packit 1fb8d4
		UINT32* newid;
Packit 1fb8d4
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 1)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		Stream_Read_UINT8(s, monitored_desktop->numWindowIds); /* numWindowIds (1 byte) */
Packit 1fb8d4
Packit 1fb8d4
		if (Stream_GetRemainingLength(s) < 4 * monitored_desktop->numWindowIds)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		if (monitored_desktop->numWindowIds > 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			size = sizeof(UINT32) * monitored_desktop->numWindowIds;
Packit 1fb8d4
			newid = (UINT32*)realloc(monitored_desktop->windowIds, size);
Packit 1fb8d4
Packit 1fb8d4
			if (!newid)
Packit 1fb8d4
			{
Packit 1fb8d4
				free(monitored_desktop->windowIds);
Packit 1fb8d4
				monitored_desktop->windowIds = NULL;
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			monitored_desktop->windowIds = newid;
Packit 1fb8d4
Packit 1fb8d4
			/* windowIds */
Packit 1fb8d4
			for (i = 0; i < (int)monitored_desktop->numWindowIds; i++)
Packit 1fb8d4
			{
Packit 1fb8d4
				Stream_Read_UINT32(s, monitored_desktop->windowIds[i]);
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void update_read_desktop_non_monitored_order(wStream* s, WINDOW_ORDER_INFO* orderInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	/* non-monitored desktop notification event */
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static void dump_monitored_desktop(wLog* log, const char* msg, const WINDOW_ORDER_INFO* orderInfo,
Packit Service 5a9772
                                   const MONITORED_DESKTOP_ORDER* monitored)
Packit Service 5a9772
{
Packit Service 5a9772
	char buffer[1000] = { 0 };
Packit Service 5a9772
	const size_t bufferSize = sizeof(buffer) - 1;
Packit Service 5a9772
Packit Service 5a9772
	DUMP_APPEND(buffer, bufferSize, "%s", msg);
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND)
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " activeWindowId=0x%" PRIx32 "", monitored->activeWindowId);
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ZORDER)
Packit Service 5a9772
	{
Packit Service 5a9772
		UINT32 i;
Packit Service 5a9772
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, " windows=(");
Packit Service 5a9772
		for (i = 0; i < monitored->numWindowIds; i++)
Packit Service 5a9772
		{
Packit Service 5a9772
			DUMP_APPEND(buffer, bufferSize, "0x%" PRIx32 ",", monitored->windowIds[i]);
Packit Service 5a9772
		}
Packit Service 5a9772
		DUMP_APPEND(buffer, bufferSize, ")");
Packit Service 5a9772
	}
Packit Service 5a9772
	WLog_Print(log, WLOG_DEBUG, buffer);
Packit Service 5a9772
}
Packit Service 5a9772
Packit 1fb8d4
static BOOL update_recv_desktop_info_order(rdpUpdate* update, wStream* s,
Packit Service 5a9772
                                           WINDOW_ORDER_INFO* orderInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpContext* context = update->context;
Packit 1fb8d4
	rdpWindowUpdate* window = update->window;
Packit 1fb8d4
	BOOL result = TRUE;
Packit 1fb8d4
Packit 1fb8d4
	if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_NONE)
Packit 1fb8d4
	{
Packit 1fb8d4
		update_read_desktop_non_monitored_order(s, orderInfo);
Packit Service 5a9772
		WLog_Print(update->log, WLOG_DEBUG, "NonMonitoredDesktop, windowId=0x%" PRIx32 "",
Packit Service 5a9772
		           orderInfo->windowId);
Packit 1fb8d4
		IFCALLRET(window->NonMonitoredDesktop, result, context, orderInfo);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit Service 5a9772
		MONITORED_DESKTOP_ORDER monitored_desktop = { 0 };
Packit Service 5a9772
		result = update_read_desktop_actively_monitored_order(s, orderInfo, &monitored_desktop);
Packit 1fb8d4
Packit Service 5a9772
		if (result)
Packit Service 5a9772
		{
Packit Service 5a9772
			dump_monitored_desktop(update->log, "ActivelyMonitoredDesktop", orderInfo,
Packit Service 5a9772
			                       &monitored_desktop);
Packit Service 5a9772
			IFCALLRET(window->MonitoredDesktop, result, context, orderInfo, &monitored_desktop);
Packit Service 5a9772
		}
Packit Service 5a9772
Packit Service 5a9772
		free(monitored_desktop.windowIds);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return result;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void update_free_window_icon_info(ICON_INFO* iconInfo)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!iconInfo)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	free(iconInfo->bitsColor);
Packit 1fb8d4
	iconInfo->bitsColor = NULL;
Packit 1fb8d4
	free(iconInfo->bitsMask);
Packit 1fb8d4
	iconInfo->bitsMask = NULL;
Packit 1fb8d4
	free(iconInfo->colorTable);
Packit 1fb8d4
	iconInfo->colorTable = NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL update_recv_altsec_window_order(rdpUpdate* update, wStream* s)
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL rc = TRUE;
Packit Service 5a9772
	size_t remaining;
Packit 1fb8d4
	UINT16 orderSize;
Packit Service 5a9772
	WINDOW_ORDER_INFO orderInfo = { 0 };
Packit Service 5a9772
	remaining = Stream_GetRemainingLength(s);
Packit 1fb8d4
Packit Service 5a9772
	if (remaining < 6)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(update->log, WLOG_ERROR, "Stream short");
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	Stream_Read_UINT16(s, orderSize);            /* orderSize (2 bytes) */
Packit Service 5a9772
	Stream_Read_UINT32(s, orderInfo.fieldFlags); /* FieldsPresentFlags (4 bytes) */
Packit 1fb8d4
Packit Service 5a9772
	if (remaining + 1 < orderSize)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_Print(update->log, WLOG_ERROR, "Stream short orderSize");
Packit 1fb8d4
		return FALSE;
Packit Service 5a9772
	}
Packit 1fb8d4
Packit Service 5a9772
	if (!window_order_supported(update->context->settings, orderInfo.fieldFlags))
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_INFO(TAG, "Window order %08" PRIx32 " not supported!", orderInfo.fieldFlags);
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (orderInfo.fieldFlags & WINDOW_ORDER_TYPE_WINDOW)
Packit Service 5a9772
		rc = update_recv_window_info_order(update, s, &orderInfo);
Packit Service 5a9772
	else if (orderInfo.fieldFlags & WINDOW_ORDER_TYPE_NOTIFY)
Packit Service 5a9772
		rc = update_recv_notification_icon_info_order(update, s, &orderInfo);
Packit Service 5a9772
	else if (orderInfo.fieldFlags & WINDOW_ORDER_TYPE_DESKTOP)
Packit Service 5a9772
		rc = update_recv_desktop_info_order(update, s, &orderInfo);
Packit 1fb8d4
Packit 1fb8d4
	if (!rc)
Packit Service 5a9772
		WLog_Print(update->log, WLOG_ERROR, "windoworder flags %08" PRIx32 " failed",
Packit Service 5a9772
		           orderInfo.fieldFlags);
Packit 1fb8d4
Packit 1fb8d4
	return rc;
Packit 1fb8d4
}