Blame libfreerdp/codec/planar.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * RDP6 Planar Codec
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2013 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service fa4841
 * Copyright 2016 Armin Novak <armin.novak@thincast.com>
Packit Service fa4841
 * Copyright 2016 Thincast Technologies GmbH
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_CONFIG_H
Packit Service b1ea74
#	include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <winpr/crt.h>
Packit Service fa4841
#include <winpr/print.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/primitives.h>
Packit Service fa4841
#include <freerdp/log.h>
Packit Service fa4841
#include <freerdp/codec/bitmap.h>
Packit Service fa4841
#include <freerdp/codec/planar.h>
Packit Service fa4841
Packit Service fa4841
#define TAG FREERDP_TAG("codec")
Packit Service fa4841
Packit Service b1ea74
static INLINE BOOL freerdp_bitmap_planar_compress_plane_rle(const BYTE* plane, UINT32 width,
Packit Service b1ea74
                                                            UINT32 height, BYTE* outPlane,
Packit Service b1ea74
                                                            UINT32* dstSize);
Packit Service b1ea74
static INLINE BYTE* freerdp_bitmap_planar_delta_encode_plane(const BYTE* inPlane, UINT32 width,
Packit Service b1ea74
                                                             UINT32 height, BYTE* outPlane);
Packit Service fa4841
Packit Service b1ea74
static INLINE INT32 planar_skip_plane_rle(const BYTE* pSrcData, UINT32 SrcSize, UINT32 nWidth,
Packit Service b1ea74
                                          UINT32 nHeight)
Packit Service fa4841
{
Packit Service b1ea74
	UINT32 used = 0;
Packit Service fa4841
	UINT32 x, y;
Packit Service fa4841
	BYTE controlByte;
Packit Service fa4841
Packit Service fa4841
	for (y = 0; y < nHeight; y++)
Packit Service fa4841
	{
Packit Service fa4841
		for (x = 0; x < nWidth;)
Packit Service fa4841
		{
Packit Service fa4841
			int cRawBytes;
Packit Service fa4841
			int nRunLength;
Packit Service fa4841
Packit Service b1ea74
			if (used >= SrcSize)
Packit Service fa4841
				return -1;
Packit Service fa4841
Packit Service b1ea74
			controlByte = pSrcData[used++];
Packit Service fa4841
			nRunLength = PLANAR_CONTROL_BYTE_RUN_LENGTH(controlByte);
Packit Service fa4841
			cRawBytes = PLANAR_CONTROL_BYTE_RAW_BYTES(controlByte);
Packit Service fa4841
Packit Service fa4841
			if (nRunLength == 1)
Packit Service fa4841
			{
Packit Service fa4841
				nRunLength = cRawBytes + 16;
Packit Service fa4841
				cRawBytes = 0;
Packit Service fa4841
			}
Packit Service fa4841
			else if (nRunLength == 2)
Packit Service fa4841
			{
Packit Service fa4841
				nRunLength = cRawBytes + 32;
Packit Service fa4841
				cRawBytes = 0;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service b1ea74
			used += cRawBytes;
Packit Service fa4841
			x += cRawBytes;
Packit Service fa4841
			x += nRunLength;
Packit Service fa4841
Packit Service fa4841
			if (x > nWidth)
Packit Service fa4841
				return -1;
Packit Service fa4841
Packit Service b1ea74
			if (used > SrcSize)
Packit Service fa4841
				return -1;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	if (used > INT32_MAX)
Packit Service b1ea74
		return -1;
Packit Service b1ea74
	return (INT32)used;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static INLINE INT32 planar_decompress_plane_rle_only(const BYTE* pSrcData, UINT32 SrcSize,
Packit Service b1ea74
                                                     BYTE* pDstData, UINT32 nWidth, UINT32 nHeight)
Packit Service b1ea74
{
Packit Service b1ea74
	INT32 x, y;
Packit Service b1ea74
	UINT32 pixel;
Packit Service b1ea74
	UINT32 cRawBytes;
Packit Service b1ea74
	UINT32 nRunLength;
Packit Service b1ea74
	INT32 deltaValue;
Packit Service b1ea74
	BYTE controlByte;
Packit Service b1ea74
	BYTE* currentScanline;
Packit Service b1ea74
	BYTE* previousScanline;
Packit Service b1ea74
	const BYTE* srcp = pSrcData;
Packit Service b1ea74
Packit Service b1ea74
	if ((nHeight > INT32_MAX) || (nWidth > INT32_MAX))
Packit Service b1ea74
		return -1;
Packit Service b1ea74
Packit Service b1ea74
	previousScanline = NULL;
Packit Service b1ea74
Packit Service b1ea74
	for (y = 0; y < (INT32)nHeight; y++)
Packit Service b1ea74
	{
Packit Service b1ea74
		BYTE* dstp = &pDstData[((y) * (INT32)nWidth)];
Packit Service b1ea74
		pixel = 0;
Packit Service b1ea74
		currentScanline = dstp;
Packit Service b1ea74
Packit Service b1ea74
		for (x = 0; x < (INT32)nWidth;)
Packit Service b1ea74
		{
Packit Service b1ea74
			controlByte = *srcp;
Packit Service b1ea74
			srcp++;
Packit Service b1ea74
Packit Service b1ea74
			if ((srcp - pSrcData) > SrcSize)
Packit Service b1ea74
			{
Packit Service b1ea74
				WLog_ERR(TAG, "error reading input buffer");
Packit Service b1ea74
				return -1;
Packit Service b1ea74
			}
Packit Service b1ea74
Packit Service b1ea74
			nRunLength = PLANAR_CONTROL_BYTE_RUN_LENGTH(controlByte);
Packit Service b1ea74
			cRawBytes = PLANAR_CONTROL_BYTE_RAW_BYTES(controlByte);
Packit Service b1ea74
Packit Service b1ea74
			if (nRunLength == 1)
Packit Service b1ea74
			{
Packit Service b1ea74
				nRunLength = cRawBytes + 16;
Packit Service b1ea74
				cRawBytes = 0;
Packit Service b1ea74
			}
Packit Service b1ea74
			else if (nRunLength == 2)
Packit Service b1ea74
			{
Packit Service b1ea74
				nRunLength = cRawBytes + 32;
Packit Service b1ea74
				cRawBytes = 0;
Packit Service b1ea74
			}
Packit Service b1ea74
Packit Service b1ea74
			if (((dstp + (cRawBytes + nRunLength)) - currentScanline) > nWidth)
Packit Service b1ea74
			{
Packit Service b1ea74
				WLog_ERR(TAG, "too many pixels in scanline");
Packit Service b1ea74
				return -1;
Packit Service b1ea74
			}
Packit Service b1ea74
Packit Service b1ea74
			if (!previousScanline)
Packit Service b1ea74
			{
Packit Service b1ea74
				/* first scanline, absolute values */
Packit Service b1ea74
				while (cRawBytes > 0)
Packit Service b1ea74
				{
Packit Service b1ea74
					pixel = *srcp;
Packit Service b1ea74
					srcp++;
Packit Service b1ea74
					*dstp = pixel;
Packit Service b1ea74
					dstp++;
Packit Service b1ea74
					x++;
Packit Service b1ea74
					cRawBytes--;
Packit Service b1ea74
				}
Packit Service b1ea74
Packit Service b1ea74
				while (nRunLength > 0)
Packit Service b1ea74
				{
Packit Service b1ea74
					*dstp = pixel;
Packit Service b1ea74
					dstp++;
Packit Service b1ea74
					x++;
Packit Service b1ea74
					nRunLength--;
Packit Service b1ea74
				}
Packit Service b1ea74
			}
Packit Service b1ea74
			else
Packit Service b1ea74
			{
Packit Service b1ea74
				/* delta values relative to previous scanline */
Packit Service b1ea74
				while (cRawBytes > 0)
Packit Service b1ea74
				{
Packit Service b1ea74
					deltaValue = *srcp;
Packit Service b1ea74
					srcp++;
Packit Service b1ea74
Packit Service b1ea74
					if (deltaValue & 1)
Packit Service b1ea74
					{
Packit Service b1ea74
						deltaValue = deltaValue >> 1;
Packit Service b1ea74
						deltaValue = deltaValue + 1;
Packit Service b1ea74
						pixel = -deltaValue;
Packit Service b1ea74
					}
Packit Service b1ea74
					else
Packit Service b1ea74
					{
Packit Service b1ea74
						deltaValue = deltaValue >> 1;
Packit Service b1ea74
						pixel = deltaValue;
Packit Service b1ea74
					}
Packit Service b1ea74
Packit Service b1ea74
					deltaValue = previousScanline[x] + pixel;
Packit Service b1ea74
					*dstp = deltaValue;
Packit Service b1ea74
					dstp++;
Packit Service b1ea74
					x++;
Packit Service b1ea74
					cRawBytes--;
Packit Service b1ea74
				}
Packit Service b1ea74
Packit Service b1ea74
				while (nRunLength > 0)
Packit Service b1ea74
				{
Packit Service b1ea74
					deltaValue = previousScanline[x] + pixel;
Packit Service b1ea74
					*dstp = deltaValue;
Packit Service b1ea74
					dstp++;
Packit Service b1ea74
					x++;
Packit Service b1ea74
					nRunLength--;
Packit Service b1ea74
				}
Packit Service b1ea74
			}
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		previousScanline = currentScanline;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return (INT32)(srcp - pSrcData);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static INLINE INT32 planar_decompress_plane_rle(const BYTE* pSrcData, UINT32 SrcSize,
Packit Service b1ea74
                                                BYTE* pDstData, INT32 nDstStep, UINT32 nXDst,
Packit Service b1ea74
                                                UINT32 nYDst, UINT32 nWidth, UINT32 nHeight,
Packit Service b1ea74
                                                UINT32 nChannel, BOOL vFlip)
Packit Service fa4841
{
Packit Service b1ea74
	INT32 x, y;
Packit Service fa4841
	UINT32 pixel;
Packit Service fa4841
	UINT32 cRawBytes;
Packit Service fa4841
	UINT32 nRunLength;
Packit Service fa4841
	INT32 deltaValue;
Packit Service fa4841
	INT32 beg, end, inc;
Packit Service fa4841
	BYTE controlByte;
Packit Service fa4841
	BYTE* currentScanline;
Packit Service fa4841
	BYTE* previousScanline;
Packit Service fa4841
	const BYTE* srcp = pSrcData;
Packit Service b1ea74
Packit Service b1ea74
	if ((nHeight > INT32_MAX) || (nWidth > INT32_MAX) || (nDstStep > INT32_MAX))
Packit Service b1ea74
		return -1;
Packit Service b1ea74
Packit Service fa4841
	previousScanline = NULL;
Packit Service fa4841
Packit Service fa4841
	if (vFlip)
Packit Service fa4841
	{
Packit Service b1ea74
		beg = (INT32)nHeight - 1;
Packit Service fa4841
		end = -1;
Packit Service fa4841
		inc = -1;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		beg = 0;
Packit Service b1ea74
		end = (INT32)nHeight;
Packit Service fa4841
		inc = 1;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	for (y = beg; y != end; y += inc)
Packit Service fa4841
	{
Packit Service b1ea74
		BYTE* dstp = &pDstData[((nYDst + y) * (INT32)nDstStep) + (nXDst * 4) + nChannel];
Packit Service fa4841
		pixel = 0;
Packit Service fa4841
		currentScanline = dstp;
Packit Service fa4841
Packit Service b1ea74
		for (x = 0; x < (INT32)nWidth;)
Packit Service fa4841
		{
Packit Service fa4841
			controlByte = *srcp;
Packit Service fa4841
			srcp++;
Packit Service fa4841
Packit Service fa4841
			if ((srcp - pSrcData) > SrcSize)
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "error reading input buffer");
Packit Service fa4841
				return -1;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			nRunLength = PLANAR_CONTROL_BYTE_RUN_LENGTH(controlByte);
Packit Service fa4841
			cRawBytes = PLANAR_CONTROL_BYTE_RAW_BYTES(controlByte);
Packit Service fa4841
Packit Service fa4841
			if (nRunLength == 1)
Packit Service fa4841
			{
Packit Service fa4841
				nRunLength = cRawBytes + 16;
Packit Service fa4841
				cRawBytes = 0;
Packit Service fa4841
			}
Packit Service fa4841
			else if (nRunLength == 2)
Packit Service fa4841
			{
Packit Service fa4841
				nRunLength = cRawBytes + 32;
Packit Service fa4841
				cRawBytes = 0;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			if (((dstp + (cRawBytes + nRunLength)) - currentScanline) > nWidth * 4)
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "too many pixels in scanline");
Packit Service fa4841
				return -1;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			if (!previousScanline)
Packit Service fa4841
			{
Packit Service fa4841
				/* first scanline, absolute values */
Packit Service fa4841
				while (cRawBytes > 0)
Packit Service fa4841
				{
Packit Service fa4841
					pixel = *srcp;
Packit Service fa4841
					srcp++;
Packit Service fa4841
					*dstp = pixel;
Packit Service fa4841
					dstp += 4;
Packit Service fa4841
					x++;
Packit Service fa4841
					cRawBytes--;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				while (nRunLength > 0)
Packit Service fa4841
				{
Packit Service fa4841
					*dstp = pixel;
Packit Service fa4841
					dstp += 4;
Packit Service fa4841
					x++;
Packit Service fa4841
					nRunLength--;
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				/* delta values relative to previous scanline */
Packit Service fa4841
				while (cRawBytes > 0)
Packit Service fa4841
				{
Packit Service fa4841
					deltaValue = *srcp;
Packit Service fa4841
					srcp++;
Packit Service fa4841
Packit Service fa4841
					if (deltaValue & 1)
Packit Service fa4841
					{
Packit Service fa4841
						deltaValue = deltaValue >> 1;
Packit Service fa4841
						deltaValue = deltaValue + 1;
Packit Service fa4841
						pixel = -deltaValue;
Packit Service fa4841
					}
Packit Service fa4841
					else
Packit Service fa4841
					{
Packit Service fa4841
						deltaValue = deltaValue >> 1;
Packit Service fa4841
						pixel = deltaValue;
Packit Service fa4841
					}
Packit Service fa4841
Packit Service fa4841
					deltaValue = previousScanline[x * 4] + pixel;
Packit Service fa4841
					*dstp = deltaValue;
Packit Service fa4841
					dstp += 4;
Packit Service fa4841
					x++;
Packit Service fa4841
					cRawBytes--;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				while (nRunLength > 0)
Packit Service fa4841
				{
Packit Service fa4841
					deltaValue = previousScanline[x * 4] + pixel;
Packit Service fa4841
					*dstp = deltaValue;
Packit Service fa4841
					dstp += 4;
Packit Service fa4841
					x++;
Packit Service fa4841
					nRunLength--;
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		previousScanline = currentScanline;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return (INT32)(srcp - pSrcData);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static INLINE INT32 planar_set_plane(BYTE bValue, BYTE* pDstData, INT32 nDstStep, UINT32 nXDst,
Packit Service b1ea74
                                     UINT32 nYDst, UINT32 nWidth, UINT32 nHeight, UINT32 nChannel,
Packit Service b1ea74
                                     BOOL vFlip)
Packit Service b1ea74
{
Packit Service b1ea74
	INT32 x, y;
Packit Service b1ea74
	INT32 beg, end, inc;
Packit Service b1ea74
Packit Service b1ea74
	if ((nHeight > INT32_MAX) || (nWidth > INT32_MAX) || (nDstStep > INT32_MAX))
Packit Service b1ea74
		return -1;
Packit Service b1ea74
Packit Service b1ea74
	if (vFlip)
Packit Service b1ea74
	{
Packit Service b1ea74
		beg = (INT32)nHeight - 1;
Packit Service b1ea74
		end = -1;
Packit Service b1ea74
		inc = -1;
Packit Service b1ea74
	}
Packit Service b1ea74
	else
Packit Service b1ea74
	{
Packit Service b1ea74
		beg = 0;
Packit Service b1ea74
		end = (INT32)nHeight;
Packit Service b1ea74
		inc = 1;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	for (y = beg; y != end; y += inc)
Packit Service b1ea74
	{
Packit Service b1ea74
		BYTE* dstp = &pDstData[((nYDst + y) * (INT32)nDstStep) + (nXDst * 4) + nChannel];
Packit Service b1ea74
Packit Service b1ea74
		for (x = 0; x < (INT32)nWidth; ++x)
Packit Service b1ea74
		{
Packit Service b1ea74
			*dstp = bValue;
Packit Service b1ea74
			dstp += 4;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return 0;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service fa4841
static INLINE BOOL writeLine(BYTE** ppRgba, UINT32 DstFormat, UINT32 width, const BYTE** ppR,
Packit Service fa4841
                             const BYTE** ppG, const BYTE** ppB, const BYTE** ppA)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 x;
Packit Service fa4841
Packit Service fa4841
	if (!ppRgba || !ppR || !ppG || !ppB)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	switch (DstFormat)
Packit Service fa4841
	{
Packit Service b1ea74
	case PIXEL_FORMAT_BGRA32:
Packit Service b1ea74
		for (x = 0; x < width; x++)
Packit Service b1ea74
		{
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppB)++;
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppG)++;
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppR)++;
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppA)++;
Packit Service b1ea74
		}
Packit Service fa4841
Packit Service b1ea74
		return TRUE;
Packit Service fa4841
Packit Service b1ea74
	case PIXEL_FORMAT_BGRX32:
Packit Service b1ea74
		for (x = 0; x < width; x++)
Packit Service b1ea74
		{
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppB)++;
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppG)++;
Packit Service b1ea74
			*(*ppRgba)++ = *(*ppR)++;
Packit Service b1ea74
			*(*ppRgba)++ = 0xFF;
Packit Service b1ea74
		}
Packit Service fa4841
Packit Service b1ea74
		return TRUE;
Packit Service bb5c11
Packit Service b1ea74
	default:
Packit Service b1ea74
		if (ppA)
Packit Service b1ea74
		{
Packit Service b1ea74
			for (x = 0; x < width; x++)
Packit Service bb5c11
			{
Packit Service b1ea74
				BYTE alpha = *(*ppA)++;
Packit Service b1ea74
				UINT32 color = FreeRDPGetColor(DstFormat, *(*ppR)++, *(*ppG)++, *(*ppB)++, alpha);
Packit Service b1ea74
				WriteColor(*ppRgba, DstFormat, color);
Packit Service b1ea74
				*ppRgba += GetBytesPerPixel(DstFormat);
Packit Service bb5c11
			}
Packit Service b1ea74
		}
Packit Service b1ea74
		else
Packit Service b1ea74
		{
Packit Service b1ea74
			const BYTE alpha = 0xFF;
Packit Service bb5c11
Packit Service b1ea74
			for (x = 0; x < width; x++)
Packit Service b1ea74
			{
Packit Service b1ea74
				UINT32 color = FreeRDPGetColor(DstFormat, *(*ppR)++, *(*ppG)++, *(*ppB)++, alpha);
Packit Service b1ea74
				WriteColor(*ppRgba, DstFormat, color);
Packit Service b1ea74
				*ppRgba += GetBytesPerPixel(DstFormat);
Packit Service fa4841
			}
Packit Service b1ea74
		}
Packit Service fa4841
Packit Service b1ea74
		return TRUE;
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static INLINE BOOL planar_decompress_planes_raw(const BYTE* pSrcData[4], BYTE* pDstData,
Packit Service b1ea74
                                                UINT32 DstFormat, UINT32 nDstStep, UINT32 nXDst,
Packit Service b1ea74
                                                UINT32 nYDst, UINT32 nWidth, UINT32 nHeight,
Packit Service b1ea74
                                                BOOL vFlip)
Packit Service fa4841
{
Packit Service fa4841
	INT32 y;
Packit Service fa4841
	INT32 beg, end, inc;
Packit Service fa4841
	const BYTE* pR = pSrcData[0];
Packit Service fa4841
	const BYTE* pG = pSrcData[1];
Packit Service fa4841
	const BYTE* pB = pSrcData[2];
Packit Service fa4841
	const BYTE* pA = pSrcData[3];
Packit Service fa4841
Packit Service fa4841
	if (vFlip)
Packit Service fa4841
	{
Packit Service fa4841
		beg = nHeight - 1;
Packit Service fa4841
		end = -1;
Packit Service fa4841
		inc = -1;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		beg = 0;
Packit Service fa4841
		end = nHeight;
Packit Service fa4841
		inc = 1;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	for (y = beg; y != end; y += inc)
Packit Service fa4841
	{
Packit Service b1ea74
		BYTE* pRGB = &pDstData[((nYDst + y) * nDstStep) + (nXDst * GetBytesPerPixel(DstFormat))];
Packit Service fa4841
Packit Service fa4841
		if (!writeLine(&pRGB, DstFormat, nWidth, &pR, &pG, &pB, &pA))
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static BOOL planar_subsample_expand(const BYTE* plane, size_t planeLength, UINT32 nWidth,
Packit Service b1ea74
                                    UINT32 nHeight, UINT32 nPlaneWidth, UINT32 nPlaneHeight,
Packit Service b1ea74
                                    BYTE* deltaPlane)
Packit Service b1ea74
{
Packit Service b1ea74
	size_t pos = 0;
Packit Service b1ea74
	UINT32 y;
Packit Service b1ea74
	if (!plane || !deltaPlane)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	if (nWidth > nPlaneWidth * 2)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	if (nHeight > nPlaneHeight * 2)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	for (y = 0; y < nHeight; y++)
Packit Service b1ea74
	{
Packit Service b1ea74
		const BYTE* src = plane + y / 2 * nPlaneWidth;
Packit Service b1ea74
		UINT32 x;
Packit Service b1ea74
Packit Service b1ea74
		for (x = 0; x < nWidth; x++)
Packit Service b1ea74
		{
Packit Service b1ea74
			deltaPlane[pos++] = src[x / 2];
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
BOOL planar_decompress(BITMAP_PLANAR_CONTEXT* planar, const BYTE* pSrcData, UINT32 SrcSize,
Packit Service b1ea74
                       UINT32 nSrcWidth, UINT32 nSrcHeight, BYTE* pDstData, UINT32 DstFormat,
Packit Service b1ea74
                       UINT32 nDstStep, UINT32 nXDst, UINT32 nYDst, UINT32 nDstWidth,
Packit Service b1ea74
                       UINT32 nDstHeight, BOOL vFlip)
Packit Service fa4841
{
Packit Service fa4841
	BOOL cs;
Packit Service fa4841
	BOOL rle;
Packit Service fa4841
	UINT32 cll;
Packit Service fa4841
	BOOL alpha;
Packit Service fa4841
	BOOL useAlpha = FALSE;
Packit Service fa4841
	INT32 status;
Packit Service fa4841
	const BYTE* srcp;
Packit Service fa4841
	UINT32 subSize;
Packit Service fa4841
	UINT32 subWidth;
Packit Service fa4841
	UINT32 subHeight;
Packit Service fa4841
	UINT32 planeSize;
Packit Service fa4841
	INT32 rleSizes[4] = { 0, 0, 0, 0 };
Packit Service fa4841
	UINT32 rawSizes[4];
Packit Service fa4841
	UINT32 rawWidths[4];
Packit Service fa4841
	UINT32 rawHeights[4];
Packit Service fa4841
	BYTE FormatHeader;
Packit Service b1ea74
	const BYTE* planes[4] = { 0 };
Packit Service fa4841
	const UINT32 w = MIN(nSrcWidth, nDstWidth);
Packit Service fa4841
	const UINT32 h = MIN(nSrcHeight, nDstHeight);
Packit Service fa4841
	const primitives_t* prims = primitives_get();
Packit Service fa4841
Packit Service fa4841
	if (nDstStep <= 0)
Packit Service fa4841
		nDstStep = nDstWidth * GetBytesPerPixel(DstFormat);
Packit Service fa4841
Packit Service fa4841
	srcp = pSrcData;
Packit Service fa4841
Packit Service fa4841
	if (!pDstData)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "Invalid argument pDstData=NULL");
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	FormatHeader = *srcp++;
Packit Service fa4841
	cll = (FormatHeader & PLANAR_FORMAT_HEADER_CLL_MASK);
Packit Service fa4841
	cs = (FormatHeader & PLANAR_FORMAT_HEADER_CS) ? TRUE : FALSE;
Packit Service fa4841
	rle = (FormatHeader & PLANAR_FORMAT_HEADER_RLE) ? TRUE : FALSE;
Packit Service fa4841
	alpha = (FormatHeader & PLANAR_FORMAT_HEADER_NA) ? FALSE : TRUE;
Packit Service fa4841
Packit Service fa4841
	if (alpha)
Packit Service fa4841
		useAlpha = ColorHasAlpha(DstFormat);
Packit Service fa4841
Packit Service b1ea74
	// WLog_INFO(TAG, "CLL: %"PRIu32" CS: %"PRIu8" RLE: %"PRIu8" ALPHA: %"PRIu8"", cll, cs, rle,
Packit Service b1ea74
	// alpha);
Packit Service fa4841
Packit Service fa4841
	if (!cll && cs)
Packit Service b1ea74
	{
Packit Service b1ea74
		WLog_ERR(TAG, "Chroma subsampling requires YCoCg and does not work with RGB data");
Packit Service fa4841
		return FALSE; /* Chroma subsampling requires YCoCg */
Packit Service b1ea74
	}
Packit Service fa4841
Packit Service fa4841
	subWidth = (nSrcWidth / 2) + (nSrcWidth % 2);
Packit Service fa4841
	subHeight = (nSrcHeight / 2) + (nSrcHeight % 2);
Packit Service fa4841
	planeSize = nSrcWidth * nSrcHeight;
Packit Service fa4841
	subSize = subWidth * subHeight;
Packit Service fa4841
Packit Service fa4841
	if (!cs)
Packit Service fa4841
	{
Packit Service fa4841
		rawSizes[0] = planeSize; /* LumaOrRedPlane */
Packit Service fa4841
		rawWidths[0] = nSrcWidth;
Packit Service fa4841
		rawHeights[0] = nSrcHeight;
Packit Service fa4841
		rawSizes[1] = planeSize; /* OrangeChromaOrGreenPlane */
Packit Service fa4841
		rawWidths[1] = nSrcWidth;
Packit Service fa4841
		rawHeights[1] = nSrcHeight;
Packit Service fa4841
		rawSizes[2] = planeSize; /* GreenChromaOrBluePlane */
Packit Service fa4841
		rawWidths[2] = nSrcWidth;
Packit Service fa4841
		rawHeights[2] = nSrcHeight;
Packit Service fa4841
		rawSizes[3] = planeSize; /* AlphaPlane */
Packit Service fa4841
		rawWidths[3] = nSrcWidth;
Packit Service fa4841
		rawHeights[3] = nSrcHeight;
Packit Service fa4841
	}
Packit Service fa4841
	else /* Chroma Subsampling */
Packit Service fa4841
	{
Packit Service fa4841
		rawSizes[0] = planeSize; /* LumaOrRedPlane */
Packit Service fa4841
		rawWidths[0] = nSrcWidth;
Packit Service fa4841
		rawHeights[0] = nSrcHeight;
Packit Service fa4841
		rawSizes[1] = subSize; /* OrangeChromaOrGreenPlane */
Packit Service fa4841
		rawWidths[1] = subWidth;
Packit Service fa4841
		rawHeights[1] = subHeight;
Packit Service fa4841
		rawSizes[2] = subSize; /* GreenChromaOrBluePlane */
Packit Service fa4841
		rawWidths[2] = subWidth;
Packit Service fa4841
		rawHeights[2] = subHeight;
Packit Service fa4841
		rawSizes[3] = planeSize; /* AlphaPlane */
Packit Service fa4841
		rawWidths[3] = nSrcWidth;
Packit Service fa4841
		rawHeights[3] = nSrcHeight;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!rle) /* RAW */
Packit Service fa4841
	{
Packit Service b1ea74
		UINT32 base = planeSize * 3;
Packit Service b1ea74
		if (cs)
Packit Service b1ea74
			base = planeSize + planeSize / 2;
Packit Service b1ea74
Packit Service fa4841
		if (alpha)
Packit Service fa4841
		{
Packit Service b1ea74
			if ((SrcSize - (srcp - pSrcData)) < (planeSize + base))
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			planes[3] = srcp;                    /* AlphaPlane */
Packit Service fa4841
			planes[0] = planes[3] + rawSizes[3]; /* LumaOrRedPlane */
Packit Service fa4841
			planes[1] = planes[0] + rawSizes[0]; /* OrangeChromaOrGreenPlane */
Packit Service fa4841
			planes[2] = planes[1] + rawSizes[1]; /* GreenChromaOrBluePlane */
Packit Service fa4841
Packit Service fa4841
			if ((planes[2] + rawSizes[2]) > &pSrcData[SrcSize])
Packit Service fa4841
				return FALSE;
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service b1ea74
			if ((SrcSize - (srcp - pSrcData)) < base)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			planes[0] = srcp;                    /* LumaOrRedPlane */
Packit Service fa4841
			planes[1] = planes[0] + rawSizes[0]; /* OrangeChromaOrGreenPlane */
Packit Service fa4841
			planes[2] = planes[1] + rawSizes[1]; /* GreenChromaOrBluePlane */
Packit Service fa4841
Packit Service fa4841
			if ((planes[2] + rawSizes[2]) > &pSrcData[SrcSize])
Packit Service fa4841
				return FALSE;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else /* RLE */
Packit Service fa4841
	{
Packit Service fa4841
		if (alpha)
Packit Service fa4841
		{
Packit Service fa4841
			planes[3] = srcp;
Packit Service fa4841
			rleSizes[3] = planar_skip_plane_rle(planes[3], SrcSize - (planes[3] - pSrcData),
Packit Service fa4841
			                                    rawWidths[3], rawHeights[3]); /* AlphaPlane */
Packit Service fa4841
Packit Service fa4841
			if (rleSizes[3] < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			planes[0] = planes[3] + rleSizes[3];
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
			planes[0] = srcp;
Packit Service fa4841
Packit Service fa4841
		rleSizes[0] = planar_skip_plane_rle(planes[0], SrcSize - (planes[0] - pSrcData),
Packit Service fa4841
		                                    rawWidths[0], rawHeights[0]); /* RedPlane */
Packit Service fa4841
Packit Service fa4841
		if (rleSizes[0] < 0)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
Packit Service fa4841
		planes[1] = planes[0] + rleSizes[0];
Packit Service fa4841
		rleSizes[1] = planar_skip_plane_rle(planes[1], SrcSize - (planes[1] - pSrcData),
Packit Service fa4841
		                                    rawWidths[1], rawHeights[1]); /* GreenPlane */
Packit Service fa4841
Packit Service fa4841
		if (rleSizes[1] < 1)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
Packit Service fa4841
		planes[2] = planes[1] + rleSizes[1];
Packit Service fa4841
		rleSizes[2] = planar_skip_plane_rle(planes[2], SrcSize - (planes[2] - pSrcData),
Packit Service fa4841
		                                    rawWidths[2], rawHeights[2]); /* BluePlane */
Packit Service fa4841
Packit Service fa4841
		if (rleSizes[2] < 1)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!cll) /* RGB */
Packit Service fa4841
	{
Packit Service fa4841
		UINT32 TempFormat;
Packit Service fa4841
		BYTE* pTempData = pDstData;
Packit Service fa4841
		UINT32 nTempStep = nDstStep;
Packit Service fa4841
Packit Service fa4841
		if (useAlpha)
Packit Service fa4841
			TempFormat = PIXEL_FORMAT_BGRA32;
Packit Service fa4841
		else
Packit Service fa4841
			TempFormat = PIXEL_FORMAT_BGRX32;
Packit Service fa4841
Packit Service b1ea74
		if ((TempFormat != DstFormat) || (nSrcWidth != nDstWidth) || (nSrcHeight != nDstHeight))
Packit Service fa4841
		{
Packit Service fa4841
			pTempData = planar->pTempData;
Packit Service fa4841
			nTempStep = planar->nTempStep;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (!rle) /* RAW */
Packit Service fa4841
		{
Packit Service b1ea74
			if (!planar_decompress_planes_raw(planes, pTempData, TempFormat, nTempStep, nXDst,
Packit Service b1ea74
			                                  nYDst, nSrcWidth, nSrcHeight, vFlip))
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			if (alpha)
Packit Service fa4841
				srcp += rawSizes[0] + rawSizes[1] + rawSizes[2] + rawSizes[3];
Packit Service fa4841
			else /* NoAlpha */
Packit Service fa4841
				srcp += rawSizes[0] + rawSizes[1] + rawSizes[2];
Packit Service fa4841
Packit Service fa4841
			if ((SrcSize - (srcp - pSrcData)) == 1)
Packit Service fa4841
				srcp++; /* pad */
Packit Service fa4841
		}
Packit Service fa4841
		else /* RLE */
Packit Service fa4841
		{
Packit Service b1ea74
			status =
Packit Service b1ea74
			    planar_decompress_plane_rle(planes[0], rleSizes[0], pTempData, nTempStep, nXDst,
Packit Service b1ea74
			                                nYDst, nSrcWidth, nSrcHeight, 2, vFlip); /* RedPlane */
Packit Service fa4841
Packit Service fa4841
			if (status < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			status = planar_decompress_plane_rle(planes[1], rleSizes[1], pTempData, nTempStep,
Packit Service b1ea74
			                                     nXDst, nYDst, nSrcWidth, nSrcHeight, 1,
Packit Service fa4841
			                                     vFlip); /* GreenPlane */
Packit Service fa4841
Packit Service fa4841
			if (status < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			status =
Packit Service b1ea74
			    planar_decompress_plane_rle(planes[2], rleSizes[2], pTempData, nTempStep, nXDst,
Packit Service b1ea74
			                                nYDst, nSrcWidth, nSrcHeight, 0, vFlip); /* BluePlane */
Packit Service fa4841
Packit Service fa4841
			if (status < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			srcp += rleSizes[0] + rleSizes[1] + rleSizes[2];
Packit Service fa4841
Packit Service fa4841
			if (useAlpha)
Packit Service fa4841
			{
Packit Service b1ea74
				status = planar_decompress_plane_rle(planes[3], rleSizes[3], pTempData, nTempStep,
Packit Service b1ea74
				                                     nXDst, nYDst, nSrcWidth, nSrcHeight, 3,
Packit Service fa4841
				                                     vFlip); /* AlphaPlane */
Packit Service bb5c11
			}
Packit Service b1ea74
			else
Packit Service b1ea74
				status = planar_set_plane(0xFF, pTempData, nTempStep, nXDst, nYDst, nSrcWidth,
Packit Service b1ea74
				                          nSrcHeight, 3, vFlip);
Packit Service b1ea74
Packit Service b1ea74
			if (status < 0)
Packit Service b1ea74
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			if (alpha)
Packit Service fa4841
				srcp += rleSizes[3];
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (pTempData != pDstData)
Packit Service fa4841
		{
Packit Service b1ea74
			if (!freerdp_image_copy(pDstData, DstFormat, nDstStep, nXDst, nYDst, w, h, pTempData,
Packit Service fa4841
			                        TempFormat, nTempStep, nXDst, nYDst, NULL, FREERDP_FLIP_NONE))
Packit Service fa4841
				return FALSE;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else /* YCoCg */
Packit Service fa4841
	{
Packit Service fa4841
		UINT32 TempFormat;
Packit Service fa4841
		BYTE* pTempData = planar->pTempData;
Packit Service fa4841
		UINT32 nTempStep = planar->nTempStep;
Packit Service fa4841
Packit Service fa4841
		if (useAlpha)
Packit Service fa4841
			TempFormat = PIXEL_FORMAT_BGRA32;
Packit Service fa4841
		else
Packit Service fa4841
			TempFormat = PIXEL_FORMAT_BGRX32;
Packit Service fa4841
Packit Service fa4841
		if (!pTempData)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
Packit Service b1ea74
		if (rle) /* RLE encoded data. Decode and handle it like raw data. */
Packit Service fa4841
		{
Packit Service b1ea74
			BYTE* rleBuffer[4] = { 0 };
Packit Service fa4841
Packit Service b1ea74
			rleBuffer[3] = planar->rlePlanesBuffer;  /* AlphaPlane */
Packit Service b1ea74
			rleBuffer[0] = rleBuffer[3] + planeSize; /* LumaOrRedPlane */
Packit Service b1ea74
			rleBuffer[1] = rleBuffer[0] + planeSize; /* OrangeChromaOrGreenPlane */
Packit Service b1ea74
			rleBuffer[2] = rleBuffer[1] + planeSize; /* GreenChromaOrBluePlane */
Packit Service fa4841
			if (useAlpha)
Packit Service fa4841
			{
Packit Service b1ea74
				status =
Packit Service b1ea74
				    planar_decompress_plane_rle_only(planes[3], rleSizes[3], rleBuffer[3],
Packit Service b1ea74
				                                     rawWidths[3], rawHeights[3]); /* AlphaPlane */
Packit Service fa4841
Packit Service fa4841
				if (status < 0)
Packit Service fa4841
					return FALSE;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			if (alpha)
Packit Service fa4841
				srcp += rleSizes[3];
Packit Service fa4841
Packit Service b1ea74
			status = planar_decompress_plane_rle_only(planes[0], rleSizes[0], rleBuffer[0],
Packit Service b1ea74
			                                          rawWidths[0], rawHeights[0]); /* LumaPlane */
Packit Service fa4841
Packit Service fa4841
			if (status < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			status =
Packit Service b1ea74
			    planar_decompress_plane_rle_only(planes[1], rleSizes[1], rleBuffer[1], rawWidths[1],
Packit Service b1ea74
			                                     rawHeights[1]); /* OrangeChromaPlane */
Packit Service fa4841
Packit Service fa4841
			if (status < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			status =
Packit Service b1ea74
			    planar_decompress_plane_rle_only(planes[2], rleSizes[2], rleBuffer[2], rawWidths[2],
Packit Service b1ea74
			                                     rawHeights[2]); /* GreenChromaPlane */
Packit Service fa4841
Packit Service fa4841
			if (status < 0)
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service b1ea74
			planes[0] = rleBuffer[0];
Packit Service b1ea74
			planes[1] = rleBuffer[1];
Packit Service b1ea74
			planes[2] = rleBuffer[2];
Packit Service b1ea74
			planes[3] = rleBuffer[3];
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		/* RAW */
Packit Service b1ea74
		{
Packit Service b1ea74
			if (cs)
Packit Service b1ea74
			{ /* Chroma subsampling for Co and Cg:
Packit Service b1ea74
			   * Each pixel contains the value that should be expanded to
Packit Service b1ea74
			   * [2x,2y;2x+1,2y;2x+1,2y+1;2x;2y+1] */
Packit Service b1ea74
				if (!planar_subsample_expand(planes[1], rawSizes[1], nSrcWidth, nSrcHeight,
Packit Service b1ea74
				                             rawWidths[1], rawHeights[1], planar->deltaPlanes[0]))
Packit Service b1ea74
					return FALSE;
Packit Service b1ea74
Packit Service b1ea74
				planes[1] = planar->deltaPlanes[0];
Packit Service b1ea74
				rawSizes[1] = planeSize; /* OrangeChromaOrGreenPlane */
Packit Service b1ea74
				rawWidths[1] = nSrcWidth;
Packit Service b1ea74
				rawHeights[1] = nSrcHeight;
Packit Service b1ea74
Packit Service b1ea74
				if (!planar_subsample_expand(planes[2], rawSizes[2], nSrcWidth, nSrcHeight,
Packit Service b1ea74
				                             rawWidths[2], rawHeights[2], planar->deltaPlanes[1]))
Packit Service b1ea74
					return FALSE;
Packit Service b1ea74
Packit Service b1ea74
				planes[2] = planar->deltaPlanes[1];
Packit Service b1ea74
				rawSizes[2] = planeSize; /* GreenChromaOrBluePlane */
Packit Service b1ea74
				rawWidths[2] = nSrcWidth;
Packit Service b1ea74
				rawHeights[2] = nSrcHeight;
Packit Service b1ea74
			}
Packit Service b1ea74
Packit Service b1ea74
			if (!planar_decompress_planes_raw(planes, pTempData, TempFormat, nTempStep, nXDst,
Packit Service b1ea74
			                                  nYDst, nSrcWidth, nSrcHeight, vFlip))
Packit Service b1ea74
				return FALSE;
Packit Service b1ea74
Packit Service b1ea74
			if (alpha)
Packit Service b1ea74
				srcp += rawSizes[0] + rawSizes[1] + rawSizes[2] + rawSizes[3];
Packit Service b1ea74
			else /* NoAlpha */
Packit Service b1ea74
				srcp += rawSizes[0] + rawSizes[1] + rawSizes[2];
Packit Service b1ea74
Packit Service b1ea74
			if ((SrcSize - (srcp - pSrcData)) == 1)
Packit Service b1ea74
				srcp++; /* pad */
Packit Service fa4841
		}
Packit Service fa4841
Packit Service b1ea74
		if (prims->YCoCgToRGB_8u_AC4R(pTempData, nTempStep, pDstData, DstFormat, nDstStep, w, h,
Packit Service b1ea74
		                              cll, useAlpha) != PRIMITIVES_SUCCESS)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static INLINE BOOL freerdp_split_color_planes(const BYTE* data, UINT32 format, UINT32 width,
Packit Service b1ea74
                                              UINT32 height, UINT32 scanline, BYTE* planes[4])
Packit Service fa4841
{
Packit Service fa4841
	INT32 i, j, k;
Packit Service b1ea74
	if ((width > INT32_MAX) || (height > INT32_MAX) || (scanline > INT32_MAX))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service fa4841
	k = 0;
Packit Service fa4841
Packit Service fa4841
	if (scanline == 0)
Packit Service fa4841
		scanline = width * GetBytesPerPixel(format);
Packit Service fa4841
Packit Service b1ea74
	for (i = (INT32)height - 1; i >= 0; i--)
Packit Service fa4841
	{
Packit Service b1ea74
		const BYTE* pixel = &data[(INT32)scanline * i];
Packit Service fa4841
Packit Service b1ea74
		for (j = 0; j < (INT32)width; j++)
Packit Service fa4841
		{
Packit Service fa4841
			const UINT32 color = ReadColor(pixel, format);
Packit Service fa4841
			pixel += GetBytesPerPixel(format);
Packit Service b1ea74
			SplitColor(color, format, &planes[1][k], &planes[2][k], &planes[3][k], &planes[0][k],
Packit Service b1ea74
			           NULL);
Packit Service fa4841
			k++;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static INLINE UINT32 freerdp_bitmap_planar_write_rle_bytes(const BYTE* pInBuffer, UINT32 cRawBytes,
Packit Service b1ea74
                                                           UINT32 nRunLength, BYTE* pOutBuffer,
Packit Service b1ea74
                                                           UINT32 outBufferSize)
Packit Service fa4841
{
Packit Service fa4841
	const BYTE* pInput;
Packit Service fa4841
	BYTE* pOutput;
Packit Service fa4841
	BYTE controlByte;
Packit Service fa4841
	UINT32 nBytesToWrite;
Packit Service fa4841
	pInput = pInBuffer;
Packit Service fa4841
	pOutput = pOutBuffer;
Packit Service fa4841
Packit Service fa4841
	if (!cRawBytes && !nRunLength)
Packit Service fa4841
		return 0;
Packit Service fa4841
Packit Service fa4841
	if (nRunLength < 3)
Packit Service fa4841
	{
Packit Service fa4841
		cRawBytes += nRunLength;
Packit Service fa4841
		nRunLength = 0;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	while (cRawBytes)
Packit Service fa4841
	{
Packit Service fa4841
		if (cRawBytes < 16)
Packit Service fa4841
		{
Packit Service fa4841
			if (nRunLength > 15)
Packit Service fa4841
			{
Packit Service fa4841
				if (nRunLength < 18)
Packit Service fa4841
				{
Packit Service fa4841
					controlByte = PLANAR_CONTROL_BYTE(13, cRawBytes);
Packit Service fa4841
					nRunLength -= 13;
Packit Service fa4841
					cRawBytes = 0;
Packit Service fa4841
				}
Packit Service fa4841
				else
Packit Service fa4841
				{
Packit Service fa4841
					controlByte = PLANAR_CONTROL_BYTE(15, cRawBytes);
Packit Service fa4841
					nRunLength -= 15;
Packit Service fa4841
					cRawBytes = 0;
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				controlByte = PLANAR_CONTROL_BYTE(nRunLength, cRawBytes);
Packit Service fa4841
				nRunLength = 0;
Packit Service fa4841
				cRawBytes = 0;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			controlByte = PLANAR_CONTROL_BYTE(0, 15);
Packit Service fa4841
			cRawBytes -= 15;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (outBufferSize < 1)
Packit Service fa4841
			return 0;
Packit Service fa4841
Packit Service fa4841
		outBufferSize--;
Packit Service fa4841
		*pOutput = controlByte;
Packit Service fa4841
		pOutput++;
Packit Service fa4841
		nBytesToWrite = (int)(controlByte >> 4);
Packit Service fa4841
Packit Service fa4841
		if (nBytesToWrite)
Packit Service fa4841
		{
Packit Service fa4841
			if (outBufferSize < nBytesToWrite)
Packit Service fa4841
				return 0;
Packit Service fa4841
Packit Service fa4841
			outBufferSize -= nBytesToWrite;
Packit Service fa4841
			CopyMemory(pOutput, pInput, nBytesToWrite);
Packit Service fa4841
			pOutput += nBytesToWrite;
Packit Service fa4841
			pInput += nBytesToWrite;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	while (nRunLength)
Packit Service fa4841
	{
Packit Service fa4841
		if (nRunLength > 47)
Packit Service fa4841
		{
Packit Service fa4841
			if (nRunLength < 50)
Packit Service fa4841
			{
Packit Service fa4841
				controlByte = PLANAR_CONTROL_BYTE(2, 13);
Packit Service fa4841
				nRunLength -= 45;
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				controlByte = PLANAR_CONTROL_BYTE(2, 15);
Packit Service fa4841
				nRunLength -= 47;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
		else if (nRunLength > 31)
Packit Service fa4841
		{
Packit Service fa4841
			controlByte = PLANAR_CONTROL_BYTE(2, (nRunLength - 32));
Packit Service fa4841
			nRunLength = 0;
Packit Service fa4841
		}
Packit Service fa4841
		else if (nRunLength > 15)
Packit Service fa4841
		{
Packit Service fa4841
			controlByte = PLANAR_CONTROL_BYTE(1, (nRunLength - 16));
Packit Service fa4841
			nRunLength = 0;
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			controlByte = PLANAR_CONTROL_BYTE(nRunLength, 0);
Packit Service fa4841
			nRunLength = 0;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (outBufferSize < 1)
Packit Service fa4841
			return 0;
Packit Service fa4841
Packit Service fa4841
		--outBufferSize;
Packit Service fa4841
		*pOutput = controlByte;
Packit Service fa4841
		pOutput++;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return (pOutput - pOutBuffer);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static INLINE UINT32 freerdp_bitmap_planar_encode_rle_bytes(const BYTE* pInBuffer,
Packit Service b1ea74
                                                            UINT32 inBufferSize, BYTE* pOutBuffer,
Packit Service b1ea74
                                                            UINT32 outBufferSize)
Packit Service fa4841
{
Packit Service fa4841
	BYTE symbol;
Packit Service fa4841
	const BYTE* pInput;
Packit Service fa4841
	BYTE* pOutput;
Packit Service fa4841
	const BYTE* pBytes;
Packit Service fa4841
	UINT32 cRawBytes;
Packit Service fa4841
	UINT32 nRunLength;
Packit Service fa4841
	UINT32 bSymbolMatch;
Packit Service fa4841
	UINT32 nBytesWritten;
Packit Service fa4841
	UINT32 nTotalBytesWritten;
Packit Service fa4841
	symbol = 0;
Packit Service fa4841
	cRawBytes = 0;
Packit Service fa4841
	nRunLength = 0;
Packit Service fa4841
	pInput = pInBuffer;
Packit Service fa4841
	pOutput = pOutBuffer;
Packit Service fa4841
	nTotalBytesWritten = 0;
Packit Service fa4841
Packit Service fa4841
	if (!outBufferSize)
Packit Service fa4841
		return 0;
Packit Service fa4841
Packit Service fa4841
	do
Packit Service fa4841
	{
Packit Service fa4841
		if (!inBufferSize)
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		bSymbolMatch = (symbol == *pInput) ? TRUE : FALSE;
Packit Service fa4841
		symbol = *pInput;
Packit Service fa4841
		pInput++;
Packit Service fa4841
		inBufferSize--;
Packit Service fa4841
Packit Service fa4841
		if (nRunLength && !bSymbolMatch)
Packit Service fa4841
		{
Packit Service fa4841
			if (nRunLength < 3)
Packit Service fa4841
			{
Packit Service fa4841
				cRawBytes += nRunLength;
Packit Service fa4841
				nRunLength = 0;
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				pBytes = pInput - (cRawBytes + nRunLength + 1);
Packit Service b1ea74
				nBytesWritten = freerdp_bitmap_planar_write_rle_bytes(pBytes, cRawBytes, nRunLength,
Packit Service b1ea74
				                                                      pOutput, outBufferSize);
Packit Service fa4841
				nRunLength = 0;
Packit Service fa4841
Packit Service fa4841
				if (!nBytesWritten || (nBytesWritten > outBufferSize))
Packit Service fa4841
					return nRunLength;
Packit Service fa4841
Packit Service fa4841
				nTotalBytesWritten += nBytesWritten;
Packit Service fa4841
				outBufferSize -= nBytesWritten;
Packit Service fa4841
				pOutput += nBytesWritten;
Packit Service fa4841
				cRawBytes = 0;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		nRunLength += bSymbolMatch;
Packit Service fa4841
		cRawBytes += (!bSymbolMatch) ? TRUE : FALSE;
Packit Service b1ea74
	} while (outBufferSize);
Packit Service fa4841
Packit Service fa4841
	if (cRawBytes || nRunLength)
Packit Service fa4841
	{
Packit Service fa4841
		pBytes = pInput - (cRawBytes + nRunLength);
Packit Service b1ea74
		nBytesWritten = freerdp_bitmap_planar_write_rle_bytes(pBytes, cRawBytes, nRunLength,
Packit Service b1ea74
		                                                      pOutput, outBufferSize);
Packit Service fa4841
Packit Service fa4841
		if (!nBytesWritten)
Packit Service fa4841
			return 0;
Packit Service fa4841
Packit Service fa4841
		nTotalBytesWritten += nBytesWritten;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (inBufferSize)
Packit Service fa4841
		return 0;
Packit Service fa4841
Packit Service fa4841
	return nTotalBytesWritten;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL freerdp_bitmap_planar_compress_plane_rle(const BYTE* inPlane, UINT32 width, UINT32 height,
Packit Service b1ea74
                                              BYTE* outPlane, UINT32* dstSize)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 index;
Packit Service fa4841
	const BYTE* pInput;
Packit Service fa4841
	BYTE* pOutput;
Packit Service fa4841
	UINT32 outBufferSize;
Packit Service fa4841
	UINT32 nBytesWritten;
Packit Service fa4841
	UINT32 nTotalBytesWritten;
Packit Service fa4841
Packit Service fa4841
	if (!outPlane)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	index = 0;
Packit Service fa4841
	pInput = inPlane;
Packit Service fa4841
	pOutput = outPlane;
Packit Service fa4841
	outBufferSize = *dstSize;
Packit Service fa4841
	nTotalBytesWritten = 0;
Packit Service fa4841
Packit Service fa4841
	while (outBufferSize)
Packit Service fa4841
	{
Packit Service b1ea74
		nBytesWritten =
Packit Service b1ea74
		    freerdp_bitmap_planar_encode_rle_bytes(pInput, width, pOutput, outBufferSize);
Packit Service fa4841
Packit Service fa4841
		if ((!nBytesWritten) || (nBytesWritten > outBufferSize))
Packit Service fa4841
			return FALSE;
Packit Service fa4841
Packit Service fa4841
		outBufferSize -= nBytesWritten;
Packit Service fa4841
		nTotalBytesWritten += nBytesWritten;
Packit Service fa4841
		pOutput += nBytesWritten;
Packit Service fa4841
		pInput += width;
Packit Service fa4841
		index++;
Packit Service fa4841
Packit Service fa4841
		if (index >= height)
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	*dstSize = nTotalBytesWritten;
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static INLINE BOOL freerdp_bitmap_planar_compress_planes_rle(BYTE* inPlanes[4], UINT32 width,
Packit Service b1ea74
                                                             UINT32 height, BYTE* outPlanes,
Packit Service b1ea74
                                                             UINT32* dstSizes, BOOL skipAlpha)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 outPlanesSize = width * height * 4;
Packit Service fa4841
Packit Service fa4841
	/* AlphaPlane */
Packit Service fa4841
	if (skipAlpha)
Packit Service fa4841
	{
Packit Service fa4841
		dstSizes[0] = 0;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		dstSizes[0] = outPlanesSize;
Packit Service fa4841
Packit Service b1ea74
		if (!freerdp_bitmap_planar_compress_plane_rle(inPlanes[0], width, height, outPlanes,
Packit Service b1ea74
		                                              &dstSizes[0]))
Packit Service fa4841
			return FALSE;
Packit Service fa4841
Packit Service fa4841
		outPlanes += dstSizes[0];
Packit Service fa4841
		outPlanesSize -= dstSizes[0];
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* LumaOrRedPlane */
Packit Service fa4841
	dstSizes[1] = outPlanesSize;
Packit Service fa4841
Packit Service b1ea74
	if (!freerdp_bitmap_planar_compress_plane_rle(inPlanes[1], width, height, outPlanes,
Packit Service b1ea74
	                                              &dstSizes[1]))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	outPlanes += dstSizes[1];
Packit Service fa4841
	outPlanesSize -= dstSizes[1];
Packit Service fa4841
	/* OrangeChromaOrGreenPlane */
Packit Service fa4841
	dstSizes[2] = outPlanesSize;
Packit Service fa4841
Packit Service b1ea74
	if (!freerdp_bitmap_planar_compress_plane_rle(inPlanes[2], width, height, outPlanes,
Packit Service b1ea74
	                                              &dstSizes[2]))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	outPlanes += dstSizes[2];
Packit Service fa4841
	outPlanesSize -= dstSizes[2];
Packit Service fa4841
	/* GreenChromeOrBluePlane */
Packit Service fa4841
	dstSizes[3] = outPlanesSize;
Packit Service fa4841
Packit Service b1ea74
	if (!freerdp_bitmap_planar_compress_plane_rle(inPlanes[3], width, height, outPlanes,
Packit Service b1ea74
	                                              &dstSizes[3]))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BYTE* freerdp_bitmap_planar_delta_encode_plane(const BYTE* inPlane, UINT32 width, UINT32 height,
Packit Service b1ea74
                                               BYTE* outPlane)
Packit Service fa4841
{
Packit Service fa4841
	char s2c;
Packit Service fa4841
	UINT32 y, x;
Packit Service fa4841
	BYTE* outPtr;
Packit Service b1ea74
	const BYTE *srcPtr, *prevLinePtr;
Packit Service fa4841
Packit Service fa4841
	if (!outPlane)
Packit Service fa4841
	{
Packit Service fa4841
		if (width * height == 0)
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service b1ea74
		if (!(outPlane = (BYTE*)calloc(height, width)))
Packit Service fa4841
			return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	// first line is copied as is
Packit Service fa4841
	CopyMemory(outPlane, inPlane, width);
Packit Service fa4841
	outPtr = outPlane + width;
Packit Service fa4841
	srcPtr = inPlane + width;
Packit Service fa4841
	prevLinePtr = inPlane;
Packit Service fa4841
Packit Service fa4841
	for (y = 1; y < height; y++)
Packit Service fa4841
	{
Packit Service fa4841
		for (x = 0; x < width; x++, outPtr++, srcPtr++, prevLinePtr++)
Packit Service fa4841
		{
Packit Service b1ea74
			INT32 delta = *srcPtr - *prevLinePtr;
Packit Service b1ea74
			s2c = (delta >= 0) ? (char)delta : (char)(~((BYTE)(-delta)) + 1);
Packit Service b1ea74
			s2c = (s2c >= 0) ? ((UINT32)s2c << 1) : (char)(((UINT32)(~((BYTE)s2c) + 1) << 1) - 1);
Packit Service fa4841
			*outPtr = (BYTE)s2c;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return outPlane;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static INLINE BOOL freerdp_bitmap_planar_delta_encode_planes(BYTE* inPlanes[4], UINT32 width,
Packit Service b1ea74
                                                             UINT32 height, BYTE* outPlanes[4])
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < 4; i++)
Packit Service fa4841
	{
Packit Service b1ea74
		outPlanes[i] =
Packit Service b1ea74
		    freerdp_bitmap_planar_delta_encode_plane(inPlanes[i], width, height, outPlanes[i]);
Packit Service fa4841
Packit Service fa4841
		if (!outPlanes[i])
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BYTE* freerdp_bitmap_compress_planar(BITMAP_PLANAR_CONTEXT* context, const BYTE* data,
Packit Service b1ea74
                                     UINT32 format, UINT32 width, UINT32 height, UINT32 scanline,
Packit Service fa4841
                                     BYTE* dstData, UINT32* pDstSize)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 size;
Packit Service fa4841
	BYTE* dstp;
Packit Service fa4841
	UINT32 planeSize;
Packit Service fa4841
	UINT32 dstSizes[4] = { 0 };
Packit Service fa4841
	BYTE FormatHeader = 0;
Packit Service fa4841
Packit Service fa4841
	if (!context || !context->rlePlanesBuffer)
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	if (context->AllowSkipAlpha)
Packit Service fa4841
		FormatHeader |= PLANAR_FORMAT_HEADER_NA;
Packit Service fa4841
Packit Service fa4841
	planeSize = width * height;
Packit Service fa4841
Packit Service b1ea74
	if (!freerdp_split_color_planes(data, format, width, height, scanline, context->planes))
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	if (context->AllowRunLengthEncoding)
Packit Service fa4841
	{
Packit Service b1ea74
		if (!freerdp_bitmap_planar_delta_encode_planes(context->planes, width, height,
Packit Service b1ea74
		                                               context->deltaPlanes))
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service b1ea74
		if (!freerdp_bitmap_planar_compress_planes_rle(context->deltaPlanes, width, height,
Packit Service b1ea74
		                                               context->rlePlanesBuffer, dstSizes,
Packit Service b1ea74
		                                               context->AllowSkipAlpha))
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service fa4841
		{
Packit Service fa4841
			int offset = 0;
Packit Service fa4841
			FormatHeader |= PLANAR_FORMAT_HEADER_RLE;
Packit Service fa4841
			context->rlePlanes[0] = &context->rlePlanesBuffer[offset];
Packit Service fa4841
			offset += dstSizes[0];
Packit Service fa4841
			context->rlePlanes[1] = &context->rlePlanesBuffer[offset];
Packit Service fa4841
			offset += dstSizes[1];
Packit Service fa4841
			context->rlePlanes[2] = &context->rlePlanesBuffer[offset];
Packit Service fa4841
			offset += dstSizes[2];
Packit Service fa4841
			context->rlePlanes[3] = &context->rlePlanesBuffer[offset];
Packit Service b1ea74
			// WLog_DBG(TAG, "R: [%"PRIu32"/%"PRIu32"] G: [%"PRIu32"/%"PRIu32"] B:
Packit Service b1ea74
			// [%"PRIu32"/%"PRIu32"]", 		dstSizes[1], planeSize, dstSizes[2], planeSize,
Packit Service b1ea74
			// dstSizes[3],
Packit Service b1ea74
			// planeSize);
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
	{
Packit Service fa4841
		if (!context->AllowRunLengthEncoding)
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service fa4841
		if (context->rlePlanes[0] == NULL)
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service fa4841
		if (context->rlePlanes[1] == NULL)
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service fa4841
		if (context->rlePlanes[2] == NULL)
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service fa4841
		if (context->rlePlanes[3] == NULL)
Packit Service fa4841
			return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!dstData)
Packit Service fa4841
	{
Packit Service fa4841
		size = 1;
Packit Service fa4841
Packit Service fa4841
		if (!(FormatHeader & PLANAR_FORMAT_HEADER_NA))
Packit Service fa4841
		{
Packit Service fa4841
			if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
				size += dstSizes[0];
Packit Service fa4841
			else
Packit Service fa4841
				size += planeSize;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
			size += (dstSizes[1] + dstSizes[2] + dstSizes[3]);
Packit Service fa4841
		else
Packit Service fa4841
			size += (planeSize * 3);
Packit Service fa4841
Packit Service fa4841
		if (!(FormatHeader & PLANAR_FORMAT_HEADER_RLE))
Packit Service fa4841
			size++;
Packit Service fa4841
Packit Service fa4841
		dstData = malloc(size);
Packit Service fa4841
Packit Service fa4841
		if (!dstData)
Packit Service fa4841
			return NULL;
Packit Service fa4841
Packit Service fa4841
		*pDstSize = size;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	dstp = dstData;
Packit Service fa4841
	*dstp = FormatHeader; /* FormatHeader */
Packit Service fa4841
	dstp++;
Packit Service fa4841
Packit Service fa4841
	/* AlphaPlane */
Packit Service fa4841
Packit Service fa4841
	if (!(FormatHeader & PLANAR_FORMAT_HEADER_NA))
Packit Service fa4841
	{
Packit Service fa4841
		if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
		{
Packit Service fa4841
			CopyMemory(dstp, context->rlePlanes[0], dstSizes[0]); /* Alpha */
Packit Service fa4841
			dstp += dstSizes[0];
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			CopyMemory(dstp, context->planes[0], planeSize); /* Alpha */
Packit Service fa4841
			dstp += planeSize;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* LumaOrRedPlane */
Packit Service fa4841
Packit Service fa4841
	if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(dstp, context->rlePlanes[1], dstSizes[1]); /* Red */
Packit Service fa4841
		dstp += dstSizes[1];
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(dstp, context->planes[1], planeSize); /* Red */
Packit Service fa4841
		dstp += planeSize;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* OrangeChromaOrGreenPlane */
Packit Service fa4841
Packit Service fa4841
	if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(dstp, context->rlePlanes[2], dstSizes[2]); /* Green */
Packit Service fa4841
		dstp += dstSizes[2];
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(dstp, context->planes[2], planeSize); /* Green */
Packit Service fa4841
		dstp += planeSize;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* GreenChromeOrBluePlane */
Packit Service fa4841
Packit Service fa4841
	if (FormatHeader & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(dstp, context->rlePlanes[3], dstSizes[3]); /* Blue */
Packit Service fa4841
		dstp += dstSizes[3];
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		CopyMemory(dstp, context->planes[3], planeSize); /* Blue */
Packit Service fa4841
		dstp += planeSize;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* Pad1 (1 byte) */
Packit Service fa4841
Packit Service fa4841
	if (!(FormatHeader & PLANAR_FORMAT_HEADER_RLE))
Packit Service fa4841
	{
Packit Service fa4841
		*dstp = 0;
Packit Service fa4841
		dstp++;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	size = (dstp - dstData);
Packit Service fa4841
	*pDstSize = size;
Packit Service fa4841
	return dstData;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL freerdp_bitmap_planar_context_reset(BITMAP_PLANAR_CONTEXT* context, UINT32 width,
Packit Service b1ea74
                                         UINT32 height)
Packit Service fa4841
{
Packit Service fa4841
	if (!context)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	context->maxWidth = width;
Packit Service fa4841
	context->maxHeight = height;
Packit Service fa4841
	context->maxPlaneSize = context->maxWidth * context->maxHeight;
Packit Service fa4841
	context->nTempStep = context->maxWidth * 4;
Packit Service fa4841
	free(context->planesBuffer);
Packit Service fa4841
	free(context->pTempData);
Packit Service fa4841
	free(context->deltaPlanesBuffer);
Packit Service fa4841
	free(context->rlePlanesBuffer);
Packit Service fa4841
	context->planesBuffer = calloc(context->maxPlaneSize, 4);
Packit Service b1ea74
	context->pTempData = calloc(context->maxPlaneSize, 6);
Packit Service fa4841
	context->deltaPlanesBuffer = calloc(context->maxPlaneSize, 4);
Packit Service fa4841
	context->rlePlanesBuffer = calloc(context->maxPlaneSize, 4);
Packit Service fa4841
Packit Service b1ea74
	if (!context->planesBuffer || !context->pTempData || !context->deltaPlanesBuffer ||
Packit Service b1ea74
	    !context->rlePlanesBuffer)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	context->planes[0] = &context->planesBuffer[context->maxPlaneSize * 0];
Packit Service fa4841
	context->planes[1] = &context->planesBuffer[context->maxPlaneSize * 1];
Packit Service fa4841
	context->planes[2] = &context->planesBuffer[context->maxPlaneSize * 2];
Packit Service fa4841
	context->planes[3] = &context->planesBuffer[context->maxPlaneSize * 3];
Packit Service b1ea74
	context->deltaPlanes[0] = &context->deltaPlanesBuffer[context->maxPlaneSize * 0];
Packit Service b1ea74
	context->deltaPlanes[1] = &context->deltaPlanesBuffer[context->maxPlaneSize * 1];
Packit Service b1ea74
	context->deltaPlanes[2] = &context->deltaPlanesBuffer[context->maxPlaneSize * 2];
Packit Service b1ea74
	context->deltaPlanes[3] = &context->deltaPlanesBuffer[context->maxPlaneSize * 3];
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BITMAP_PLANAR_CONTEXT* freerdp_bitmap_planar_context_new(DWORD flags, UINT32 maxWidth,
Packit Service b1ea74
                                                         UINT32 maxHeight)
Packit Service fa4841
{
Packit Service fa4841
	BITMAP_PLANAR_CONTEXT* context;
Packit Service b1ea74
	context = (BITMAP_PLANAR_CONTEXT*)calloc(1, sizeof(BITMAP_PLANAR_CONTEXT));
Packit Service fa4841
Packit Service fa4841
	if (!context)
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	if (flags & PLANAR_FORMAT_HEADER_NA)
Packit Service fa4841
		context->AllowSkipAlpha = TRUE;
Packit Service fa4841
Packit Service fa4841
	if (flags & PLANAR_FORMAT_HEADER_RLE)
Packit Service fa4841
		context->AllowRunLengthEncoding = TRUE;
Packit Service fa4841
Packit Service fa4841
	if (flags & PLANAR_FORMAT_HEADER_CS)
Packit Service fa4841
		context->AllowColorSubsampling = TRUE;
Packit Service fa4841
Packit Service fa4841
	context->ColorLossLevel = flags & PLANAR_FORMAT_HEADER_CLL_MASK;
Packit Service fa4841
Packit Service fa4841
	if (context->ColorLossLevel)
Packit Service fa4841
		context->AllowDynamicColorFidelity = TRUE;
Packit Service fa4841
Packit Service fa4841
	if (!freerdp_bitmap_planar_context_reset(context, maxWidth, maxHeight))
Packit Service fa4841
	{
Packit Service fa4841
		freerdp_bitmap_planar_context_free(context);
Packit Service fa4841
		return NULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return context;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void freerdp_bitmap_planar_context_free(BITMAP_PLANAR_CONTEXT* context)
Packit Service fa4841
{
Packit Service fa4841
	if (!context)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	free(context->pTempData);
Packit Service fa4841
	free(context->planesBuffer);
Packit Service fa4841
	free(context->deltaPlanesBuffer);
Packit Service fa4841
	free(context->rlePlanesBuffer);
Packit Service fa4841
	free(context);
Packit Service fa4841
}