Blame libfreerdp/codec/interleaved.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * Interleaved RLE Bitmap Codec
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service fa4841
 * Copyright 2015 Thincast Technologies GmbH
Packit Service fa4841
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.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 fa4841
#include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <freerdp/codec/interleaved.h>
Packit Service fa4841
#include <freerdp/log.h>
Packit Service fa4841
Packit Service fa4841
#define TAG FREERDP_TAG("codec")
Packit Service fa4841
Packit Service fa4841
#define UNROLL_BODY(_exp, _count)      \
Packit Service fa4841
	do                                 \
Packit Service fa4841
	{                                  \
Packit Service fa4841
		size_t x;                      \
Packit Service fa4841
		for (x = 0; x < (_count); x++) \
Packit Service fa4841
		{                              \
Packit Service fa4841
			do                         \
Packit Service fa4841
				_exp while (FALSE);    \
Packit Service fa4841
		}                              \
Packit Service fa4841
	} while (FALSE)
Packit Service fa4841
Packit Service fa4841
#define UNROLL_MULTIPLE(_condition, _exp, _count) \
Packit Service fa4841
	do                                            \
Packit Service fa4841
	{                                             \
Packit Service fa4841
		while ((_condition) >= _count)            \
Packit Service fa4841
		{                                         \
Packit Service fa4841
			UNROLL_BODY(_exp, _count);            \
Packit Service fa4841
			(_condition) -= _count;               \
Packit Service fa4841
		}                                         \
Packit Service fa4841
	} while (FALSE)
Packit Service fa4841
Packit Service fa4841
#define UNROLL(_condition, _exp)               \
Packit Service fa4841
	do                                         \
Packit Service fa4841
	{                                          \
Packit Service fa4841
		UNROLL_MULTIPLE(_condition, _exp, 16); \
Packit Service fa4841
		UNROLL_MULTIPLE(_condition, _exp, 4);  \
Packit Service fa4841
		UNROLL_MULTIPLE(_condition, _exp, 1);  \
Packit Service fa4841
	} while (FALSE)
Packit Service fa4841
Packit Service fa4841
/*
Packit Service fa4841
   RLE Compressed Bitmap Stream (RLE_BITMAP_STREAM)
Packit Service fa4841
   http://msdn.microsoft.com/en-us/library/cc240895%28v=prot.10%29.aspx
Packit Service fa4841
   pseudo-code
Packit Service fa4841
   http://msdn.microsoft.com/en-us/library/dd240593%28v=prot.10%29.aspx
Packit Service fa4841
*/
Packit Service fa4841
Packit Service fa4841
#define REGULAR_BG_RUN 0x00
Packit Service fa4841
#define MEGA_MEGA_BG_RUN 0xF0
Packit Service fa4841
#define REGULAR_FG_RUN 0x01
Packit Service fa4841
#define MEGA_MEGA_FG_RUN 0xF1
Packit Service fa4841
#define LITE_SET_FG_FG_RUN 0x0C
Packit Service fa4841
#define MEGA_MEGA_SET_FG_RUN 0xF6
Packit Service fa4841
#define LITE_DITHERED_RUN 0x0E
Packit Service fa4841
#define MEGA_MEGA_DITHERED_RUN 0xF8
Packit Service fa4841
#define REGULAR_COLOR_RUN 0x03
Packit Service fa4841
#define MEGA_MEGA_COLOR_RUN 0xF3
Packit Service fa4841
#define REGULAR_FGBG_IMAGE 0x02
Packit Service fa4841
#define MEGA_MEGA_FGBG_IMAGE 0xF2
Packit Service fa4841
#define LITE_SET_FG_FGBG_IMAGE 0x0D
Packit Service fa4841
#define MEGA_MEGA_SET_FGBG_IMAGE 0xF7
Packit Service fa4841
#define REGULAR_COLOR_IMAGE 0x04
Packit Service fa4841
#define MEGA_MEGA_COLOR_IMAGE 0xF4
Packit Service fa4841
#define SPECIAL_FGBG_1 0xF9
Packit Service fa4841
#define SPECIAL_FGBG_2 0xFA
Packit Service fa4841
#define SPECIAL_WHITE 0xFD
Packit Service fa4841
#define SPECIAL_BLACK 0xFE
Packit Service fa4841
Packit Service fa4841
#define BLACK_PIXEL 0x000000
Packit Service fa4841
Packit Service fa4841
typedef UINT32 PIXEL;
Packit Service fa4841
Packit Service fa4841
static const BYTE g_MaskSpecialFgBg1 = 0x03;
Packit Service fa4841
static const BYTE g_MaskSpecialFgBg2 = 0x05;
Packit Service fa4841
Packit Service fa4841
static const BYTE g_MaskRegularRunLength = 0x1F;
Packit Service fa4841
static const BYTE g_MaskLiteRunLength = 0x0F;
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Reads the supplied order header and extracts the compression
Packit Service fa4841
 * order code ID.
Packit Service fa4841
 */
Packit Service fa4841
static INLINE UINT32 ExtractCodeId(BYTE bOrderHdr)
Packit Service fa4841
{
Packit Service fa4841
	if ((bOrderHdr & 0xC0U) != 0xC0U)
Packit Service fa4841
	{
Packit Service fa4841
		/* REGULAR orders
Packit Service fa4841
		 * (000x xxxx, 001x xxxx, 010x xxxx, 011x xxxx, 100x xxxx)
Packit Service fa4841
		 */
Packit Service fa4841
		return bOrderHdr >> 5;
Packit Service fa4841
	}
Packit Service fa4841
	else if ((bOrderHdr & 0xF0U) == 0xF0U)
Packit Service fa4841
	{
Packit Service fa4841
		/* MEGA and SPECIAL orders (0xF*) */
Packit Service fa4841
		return bOrderHdr;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		/* LITE orders
Packit Service fa4841
		 * 1100 xxxx, 1101 xxxx, 1110 xxxx)
Packit Service fa4841
		 */
Packit Service fa4841
		return bOrderHdr >> 4;
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Extract the run length of a compression order.
Packit Service fa4841
 */
Packit Service fa4841
static INLINE UINT32 ExtractRunLength(UINT32 code, const BYTE* pbOrderHdr, UINT32* advance)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 runLength;
Packit Service fa4841
	UINT32 ladvance;
Packit Service fa4841
	ladvance = 1;
Packit Service fa4841
	runLength = 0;
Packit Service fa4841
Packit Service fa4841
	switch (code)
Packit Service fa4841
	{
Packit Service fa4841
		case REGULAR_FGBG_IMAGE:
Packit Service fa4841
			runLength = (*pbOrderHdr) & g_MaskRegularRunLength;
Packit Service fa4841
Packit Service fa4841
			if (runLength == 0)
Packit Service fa4841
			{
Packit Service fa4841
				runLength = (*(pbOrderHdr + 1)) + 1;
Packit Service fa4841
				ladvance += 1;
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				runLength = runLength * 8;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case LITE_SET_FG_FGBG_IMAGE:
Packit Service fa4841
			runLength = (*pbOrderHdr) & g_MaskLiteRunLength;
Packit Service fa4841
Packit Service fa4841
			if (runLength == 0)
Packit Service fa4841
			{
Packit Service fa4841
				runLength = (*(pbOrderHdr + 1)) + 1;
Packit Service fa4841
				ladvance += 1;
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				runLength = runLength * 8;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case REGULAR_BG_RUN:
Packit Service fa4841
		case REGULAR_FG_RUN:
Packit Service fa4841
		case REGULAR_COLOR_RUN:
Packit Service fa4841
		case REGULAR_COLOR_IMAGE:
Packit Service fa4841
			runLength = (*pbOrderHdr) & g_MaskRegularRunLength;
Packit Service fa4841
Packit Service fa4841
			if (runLength == 0)
Packit Service fa4841
			{
Packit Service fa4841
				/* An extended (MEGA) run. */
Packit Service fa4841
				runLength = (*(pbOrderHdr + 1)) + 32;
Packit Service fa4841
				ladvance += 1;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case LITE_SET_FG_FG_RUN:
Packit Service fa4841
		case LITE_DITHERED_RUN:
Packit Service fa4841
			runLength = (*pbOrderHdr) & g_MaskLiteRunLength;
Packit Service fa4841
Packit Service fa4841
			if (runLength == 0)
Packit Service fa4841
			{
Packit Service fa4841
				/* An extended (MEGA) run. */
Packit Service fa4841
				runLength = (*(pbOrderHdr + 1)) + 16;
Packit Service fa4841
				ladvance += 1;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case MEGA_MEGA_BG_RUN:
Packit Service fa4841
		case MEGA_MEGA_FG_RUN:
Packit Service fa4841
		case MEGA_MEGA_SET_FG_RUN:
Packit Service fa4841
		case MEGA_MEGA_DITHERED_RUN:
Packit Service fa4841
		case MEGA_MEGA_COLOR_RUN:
Packit Service fa4841
		case MEGA_MEGA_FGBG_IMAGE:
Packit Service fa4841
		case MEGA_MEGA_SET_FGBG_IMAGE:
Packit Service fa4841
		case MEGA_MEGA_COLOR_IMAGE:
Packit Service fa4841
			runLength = ((UINT16)pbOrderHdr[1]) | ((UINT16)(pbOrderHdr[2] << 8));
Packit Service fa4841
			ladvance += 2;
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	*advance = ladvance;
Packit Service fa4841
	return runLength;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static INLINE BOOL ensure_capacity(const BYTE* start, const BYTE* end, size_t size, size_t base)
Packit Service fa4841
{
Packit Service fa4841
	const size_t available = (uintptr_t)end - (uintptr_t)start;
Packit Service fa4841
	const BOOL rc = available >= size * base;
Packit Service fa4841
	return rc && (start <= end);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static INLINE void write_pixel_8(BYTE* _buf, BYTE _pix)
Packit Service fa4841
{
Packit Service fa4841
	*_buf = _pix;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static INLINE void write_pixel_24(BYTE* _buf, UINT32 _pix)
Packit Service fa4841
{
Packit Service fa4841
	(_buf)[0] = (BYTE)(_pix);
Packit Service fa4841
	(_buf)[1] = (BYTE)((_pix) >> 8);
Packit Service fa4841
	(_buf)[2] = (BYTE)((_pix) >> 16);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static INLINE void write_pixel_16(BYTE* _buf, UINT16 _pix)
Packit Service fa4841
{
Packit Service fa4841
	_buf[0] = _pix & 0xFF;
Packit Service fa4841
	_buf[1] = (_pix >> 8) & 0xFF;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
#undef DESTWRITEPIXEL
Packit Service fa4841
#undef DESTREADPIXEL
Packit Service fa4841
#undef SRCREADPIXEL
Packit Service fa4841
#undef DESTNEXTPIXEL
Packit Service fa4841
#undef SRCNEXTPIXEL
Packit Service fa4841
#undef WRITEFGBGIMAGE
Packit Service fa4841
#undef WRITEFIRSTLINEFGBGIMAGE
Packit Service fa4841
#undef RLEDECOMPRESS
Packit Service fa4841
#undef RLEEXTRA
Packit Service fa4841
#undef WHITE_PIXEL
Packit Service fa4841
#define WHITE_PIXEL 0xFF
Packit Service fa4841
#define DESTWRITEPIXEL(_buf, _pix) write_pixel_8(_buf, _pix)
Packit Service fa4841
#define DESTREADPIXEL(_pix, _buf) _pix = (_buf)[0]
Packit Service fa4841
#define SRCREADPIXEL(_pix, _buf) _pix = (_buf)[0]
Packit Service fa4841
#define DESTNEXTPIXEL(_buf) _buf += 1
Packit Service fa4841
#define SRCNEXTPIXEL(_buf) _buf += 1
Packit Service fa4841
#define WRITEFGBGIMAGE WriteFgBgImage8to8
Packit Service fa4841
#define WRITEFIRSTLINEFGBGIMAGE WriteFirstLineFgBgImage8to8
Packit Service fa4841
#define RLEDECOMPRESS RleDecompress8to8
Packit Service fa4841
#define RLEEXTRA
Packit Service fa4841
#undef ENSURE_CAPACITY
Packit Service fa4841
#define ENSURE_CAPACITY(_start, _end, _size) ensure_capacity(_start, _end, _size, 1)
Packit Service fa4841
#include "include/bitmap.c"
Packit Service fa4841
Packit Service fa4841
#undef DESTWRITEPIXEL
Packit Service fa4841
#undef DESTREADPIXEL
Packit Service fa4841
#undef SRCREADPIXEL
Packit Service fa4841
#undef DESTNEXTPIXEL
Packit Service fa4841
#undef SRCNEXTPIXEL
Packit Service fa4841
#undef WRITEFGBGIMAGE
Packit Service fa4841
#undef WRITEFIRSTLINEFGBGIMAGE
Packit Service fa4841
#undef RLEDECOMPRESS
Packit Service fa4841
#undef RLEEXTRA
Packit Service fa4841
#undef WHITE_PIXEL
Packit Service fa4841
#define WHITE_PIXEL 0xFFFF
Packit Service fa4841
#define DESTWRITEPIXEL(_buf, _pix) write_pixel_16(_buf, _pix)
Packit Service fa4841
#define DESTREADPIXEL(_pix, _buf) _pix = ((UINT16*)(_buf))[0]
Packit Service fa4841
#define SRCREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8)
Packit Service fa4841
#define DESTNEXTPIXEL(_buf) _buf += 2
Packit Service fa4841
#define SRCNEXTPIXEL(_buf) _buf += 2
Packit Service fa4841
#define WRITEFGBGIMAGE WriteFgBgImage16to16
Packit Service fa4841
#define WRITEFIRSTLINEFGBGIMAGE WriteFirstLineFgBgImage16to16
Packit Service fa4841
#define RLEDECOMPRESS RleDecompress16to16
Packit Service fa4841
#define RLEEXTRA
Packit Service fa4841
#undef ENSURE_CAPACITY
Packit Service fa4841
#define ENSURE_CAPACITY(_start, _end, _size) ensure_capacity(_start, _end, _size, 2)
Packit Service fa4841
#include "include/bitmap.c"
Packit Service fa4841
Packit Service fa4841
#undef DESTWRITEPIXEL
Packit Service fa4841
#undef DESTREADPIXEL
Packit Service fa4841
#undef SRCREADPIXEL
Packit Service fa4841
#undef DESTNEXTPIXEL
Packit Service fa4841
#undef SRCNEXTPIXEL
Packit Service fa4841
#undef WRITEFGBGIMAGE
Packit Service fa4841
#undef WRITEFIRSTLINEFGBGIMAGE
Packit Service fa4841
#undef RLEDECOMPRESS
Packit Service fa4841
#undef RLEEXTRA
Packit Service fa4841
#undef WHITE_PIXEL
Packit Service fa4841
#define WHITE_PIXEL 0xFFFFFF
Packit Service fa4841
#define DESTWRITEPIXEL(_buf, _pix) write_pixel_24(_buf, _pix)
Packit Service fa4841
#define DESTREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8) | ((_buf)[2] << 16)
Packit Service fa4841
#define SRCREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8) | ((_buf)[2] << 16)
Packit Service fa4841
#define DESTNEXTPIXEL(_buf) _buf += 3
Packit Service fa4841
#define SRCNEXTPIXEL(_buf) _buf += 3
Packit Service fa4841
#define WRITEFGBGIMAGE WriteFgBgImage24to24
Packit Service fa4841
#define WRITEFIRSTLINEFGBGIMAGE WriteFirstLineFgBgImage24to24
Packit Service fa4841
#define RLEDECOMPRESS RleDecompress24to24
Packit Service fa4841
#define RLEEXTRA
Packit Service fa4841
#undef ENSURE_CAPACITY
Packit Service fa4841
#define ENSURE_CAPACITY(_start, _end, _size) ensure_capacity(_start, _end, _size, 3)
Packit Service fa4841
#include "include/bitmap.c"
Packit Service fa4841
Packit Service fa4841
BOOL interleaved_decompress(BITMAP_INTERLEAVED_CONTEXT* interleaved, const BYTE* pSrcData,
Packit Service fa4841
                            UINT32 SrcSize, UINT32 nSrcWidth, UINT32 nSrcHeight, UINT32 bpp,
Packit Service fa4841
                            BYTE* pDstData, UINT32 DstFormat, UINT32 nDstStep, UINT32 nXDst,
Packit Service fa4841
                            UINT32 nYDst, UINT32 nDstWidth, UINT32 nDstHeight,
Packit Service fa4841
                            const gdiPalette* palette)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 scanline;
Packit Service fa4841
	UINT32 SrcFormat;
Packit Service fa4841
	UINT32 BufferSize;
Packit Service fa4841
Packit Service fa4841
	if (!interleaved || !pSrcData || !pDstData)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	switch (bpp)
Packit Service fa4841
	{
Packit Service fa4841
		case 24:
Packit Service fa4841
			scanline = nSrcWidth * 3;
Packit Service fa4841
			SrcFormat = PIXEL_FORMAT_BGR24;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 16:
Packit Service fa4841
			scanline = nSrcWidth * 2;
Packit Service fa4841
			SrcFormat = PIXEL_FORMAT_RGB16;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 15:
Packit Service fa4841
			scanline = nSrcWidth * 2;
Packit Service fa4841
			SrcFormat = PIXEL_FORMAT_RGB15;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 8:
Packit Service fa4841
			scanline = nSrcWidth;
Packit Service fa4841
			SrcFormat = PIXEL_FORMAT_RGB8;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		default:
Packit Service fa4841
			WLog_ERR(TAG, "Invalid color depth %" PRIu32 "", bpp);
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	BufferSize = scanline * nSrcHeight;
Packit Service fa4841
Packit Service fa4841
	if (BufferSize > interleaved->TempSize)
Packit Service fa4841
	{
Packit Service fa4841
		interleaved->TempBuffer = _aligned_realloc(interleaved->TempBuffer, BufferSize, 16);
Packit Service fa4841
		interleaved->TempSize = BufferSize;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!interleaved->TempBuffer)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	switch (bpp)
Packit Service fa4841
	{
Packit Service fa4841
		case 24:
Packit Service fa4841
			if (!RleDecompress24to24(pSrcData, SrcSize, interleaved->TempBuffer, scanline,
Packit Service fa4841
			                         nSrcWidth, nSrcHeight))
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 16:
Packit Service fa4841
		case 15:
Packit Service fa4841
			if (!RleDecompress16to16(pSrcData, SrcSize, interleaved->TempBuffer, scanline,
Packit Service fa4841
			                         nSrcWidth, nSrcHeight))
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 8:
Packit Service fa4841
			if (!RleDecompress8to8(pSrcData, SrcSize, interleaved->TempBuffer, scanline, nSrcWidth,
Packit Service fa4841
			                       nSrcHeight))
Packit Service fa4841
				return FALSE;
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		default:
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return freerdp_image_copy(pDstData, DstFormat, nDstStep, nXDst, nYDst, nDstWidth, nDstHeight,
Packit Service fa4841
	                          interleaved->TempBuffer, SrcFormat, scanline, 0, 0, palette,
Packit Service fa4841
	                          FREERDP_FLIP_VERTICAL);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL interleaved_compress(BITMAP_INTERLEAVED_CONTEXT* interleaved, BYTE* pDstData, UINT32* pDstSize,
Packit Service fa4841
                          UINT32 nWidth, UINT32 nHeight, const BYTE* pSrcData, UINT32 SrcFormat,
Packit Service fa4841
                          UINT32 nSrcStep, UINT32 nXSrc, UINT32 nYSrc, const gdiPalette* palette,
Packit Service fa4841
                          UINT32 bpp)
Packit Service fa4841
{
Packit Service fa4841
	BOOL status;
Packit Service fa4841
	wStream* s;
Packit Service fa4841
	UINT32 DstFormat = 0;
Packit Service fa4841
	const size_t maxSize = 64 * 64 * 4;
Packit Service fa4841
Packit Service fa4841
	if (!interleaved || !pDstData || !pSrcData)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if ((nWidth == 0) || (nHeight == 0))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (nWidth % 4)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "interleaved_compress: width is not a multiple of 4");
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if ((nWidth > 64) || (nHeight > 64))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG,
Packit Service fa4841
		         "interleaved_compress: width (%" PRIu32 ") or height (%" PRIu32
Packit Service fa4841
		         ") is greater than 64",
Packit Service fa4841
		         nWidth, nHeight);
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	switch (bpp)
Packit Service fa4841
	{
Packit Service fa4841
		case 24:
Packit Service fa4841
			DstFormat = PIXEL_FORMAT_BGRX32;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 16:
Packit Service fa4841
			DstFormat = PIXEL_FORMAT_RGB16;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case 15:
Packit Service fa4841
			DstFormat = PIXEL_FORMAT_RGB15;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		default:
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!freerdp_image_copy(interleaved->TempBuffer, DstFormat, 0, 0, 0, nWidth, nHeight, pSrcData,
Packit Service fa4841
	                        SrcFormat, nSrcStep, nXSrc, nYSrc, palette, FREERDP_FLIP_NONE))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	s = Stream_New(pDstData, *pDstSize);
Packit Service fa4841
Packit Service fa4841
	if (!s)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	Stream_SetPosition(interleaved->bts, 0);
Packit Service fa4841
Packit Service fa4841
	if (freerdp_bitmap_compress(interleaved->TempBuffer, nWidth, nHeight, s, bpp, maxSize,
Packit Service fa4841
	                            nHeight - 1, interleaved->bts, 0) < 0)
Packit Service fa4841
		status = FALSE;
Packit Service fa4841
	else
Packit Service fa4841
		status = TRUE;
Packit Service fa4841
Packit Service fa4841
	Stream_SealLength(s);
Packit Service fa4841
	*pDstSize = (UINT32)Stream_Length(s);
Packit Service fa4841
	Stream_Free(s, FALSE);
Packit Service fa4841
	return status;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL bitmap_interleaved_context_reset(BITMAP_INTERLEAVED_CONTEXT* interleaved)
Packit Service fa4841
{
Packit Service fa4841
	if (!interleaved)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BITMAP_INTERLEAVED_CONTEXT* bitmap_interleaved_context_new(BOOL Compressor)
Packit Service fa4841
{
Packit Service fa4841
	BITMAP_INTERLEAVED_CONTEXT* interleaved;
Packit Service fa4841
	interleaved = (BITMAP_INTERLEAVED_CONTEXT*)calloc(1, sizeof(BITMAP_INTERLEAVED_CONTEXT));
Packit Service fa4841
Packit Service fa4841
	if (interleaved)
Packit Service fa4841
	{
Packit Service fa4841
		interleaved->TempSize = 64 * 64 * 4;
Packit Service fa4841
		interleaved->TempBuffer = _aligned_malloc(interleaved->TempSize, 16);
Packit Service fa4841
Packit Service fa4841
		if (!interleaved->TempBuffer)
Packit Service fa4841
		{
Packit Service fa4841
			free(interleaved);
Packit Service fa4841
			WLog_ERR(TAG, "_aligned_malloc failed!");
Packit Service fa4841
			return NULL;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		interleaved->bts = Stream_New(NULL, interleaved->TempSize);
Packit Service fa4841
Packit Service fa4841
		if (!interleaved->bts)
Packit Service fa4841
		{
Packit Service fa4841
			_aligned_free(interleaved->TempBuffer);
Packit Service fa4841
			free(interleaved);
Packit Service fa4841
			WLog_ERR(TAG, "Stream_New failed!");
Packit Service fa4841
			return NULL;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return interleaved;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void bitmap_interleaved_context_free(BITMAP_INTERLEAVED_CONTEXT* interleaved)
Packit Service fa4841
{
Packit Service fa4841
	if (!interleaved)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	_aligned_free(interleaved->TempBuffer);
Packit Service fa4841
	Stream_Free(interleaved->bts, TRUE);
Packit Service fa4841
	free(interleaved);
Packit Service fa4841
}