Blame libfreerdp/codec/interleaved.c

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