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 1fb8d4
#define UNROLL_BODY(_exp, _count) do \
Packit 1fb8d4
	{ \
Packit 1fb8d4
		size_t x; \
Packit 1fb8d4
		for (x=0; x<(_count); x++) \
Packit 1fb8d4
		{ \
Packit 1fb8d4
			do _exp while(FALSE); \
Packit 1fb8d4
		} \
Packit 1fb8d4
	} \
Packit 1fb8d4
	while (FALSE)
Packit 1fb8d4
Packit 1fb8d4
#define UNROLL_MULTIPLE(_condition, _exp, _count) do \
Packit 1fb8d4
	{ \
Packit 1fb8d4
		while ((_condition) >= _count) \
Packit 1fb8d4
		{ \
Packit 1fb8d4
			UNROLL_BODY(_exp, _count); \
Packit 1fb8d4
			(_condition) -= _count; \
Packit 1fb8d4
		} \
Packit 1fb8d4
	} \
Packit 1fb8d4
	while (FALSE)
Packit 1fb8d4
Packit 1fb8d4
#define UNROLL(_condition, _exp) do \
Packit 1fb8d4
	{ \
Packit 1fb8d4
		UNROLL_MULTIPLE(_condition, _exp, 16); \
Packit 1fb8d4
		UNROLL_MULTIPLE(_condition, _exp, 4); \
Packit 1fb8d4
		UNROLL_MULTIPLE(_condition, _exp, 1); \
Packit 1fb8d4
	} \
Packit 1fb8d4
	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 1fb8d4
#define REGULAR_BG_RUN              0x00
Packit 1fb8d4
#define MEGA_MEGA_BG_RUN            0xF0
Packit 1fb8d4
#define REGULAR_FG_RUN              0x01
Packit 1fb8d4
#define MEGA_MEGA_FG_RUN            0xF1
Packit 1fb8d4
#define LITE_SET_FG_FG_RUN          0x0C
Packit 1fb8d4
#define MEGA_MEGA_SET_FG_RUN        0xF6
Packit 1fb8d4
#define LITE_DITHERED_RUN           0x0E
Packit 1fb8d4
#define MEGA_MEGA_DITHERED_RUN      0xF8
Packit 1fb8d4
#define REGULAR_COLOR_RUN           0x03
Packit 1fb8d4
#define MEGA_MEGA_COLOR_RUN         0xF3
Packit 1fb8d4
#define REGULAR_FGBG_IMAGE          0x02
Packit 1fb8d4
#define MEGA_MEGA_FGBG_IMAGE        0xF2
Packit 1fb8d4
#define LITE_SET_FG_FGBG_IMAGE      0x0D
Packit 1fb8d4
#define MEGA_MEGA_SET_FGBG_IMAGE    0xF7
Packit 1fb8d4
#define REGULAR_COLOR_IMAGE         0x04
Packit 1fb8d4
#define MEGA_MEGA_COLOR_IMAGE       0xF4
Packit 1fb8d4
#define SPECIAL_FGBG_1              0xF9
Packit 1fb8d4
#define SPECIAL_FGBG_2              0xFA
Packit 1fb8d4
#define SPECIAL_WHITE               0xFD
Packit 1fb8d4
#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 1fb8d4
static INLINE UINT32 ExtractRunLength(UINT32 code, const BYTE* pbOrderHdr,
Packit 1fb8d4
                                      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 1fb8d4
			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 1c2678
	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
#ifdef HAVE_ALIGNED_REQUIRED
Packit 1fb8d4
#define SRCREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8)
Packit 1fb8d4
#else
Packit 1fb8d4
#define SRCREADPIXEL(_pix, _buf) _pix = ((UINT16*)(_buf))[0]
Packit 1fb8d4
#endif
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 1fb8d4
#define DESTWRITEPIXEL(_buf, _pix)  write_pixel_24(_buf, _pix)
Packit 1fb8d4
#define DESTREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8) | \
Packit 1fb8d4
        ((_buf)[2] << 16)
Packit 1fb8d4
#define SRCREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8) | \
Packit 1fb8d4
                                        ((_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 1fb8d4
BOOL interleaved_decompress(BITMAP_INTERLEAVED_CONTEXT* interleaved,
Packit 1fb8d4
                            const BYTE* pSrcData, UINT32 SrcSize,
Packit 1fb8d4
                            UINT32 nSrcWidth, UINT32 nSrcHeight,
Packit 1fb8d4
                            UINT32 bpp,
Packit 1fb8d4
                            BYTE* pDstData, UINT32 DstFormat,
Packit 1fb8d4
                            UINT32 nDstStep, UINT32 nXDst, UINT32 nYDst,
Packit 1fb8d4
                            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 1fb8d4
			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 1fb8d4
		interleaved->TempBuffer = _aligned_realloc(
Packit 1fb8d4
		                              interleaved->TempBuffer,
Packit 1fb8d4
		                              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 1fb8d4
			if (!RleDecompress24to24(pSrcData, SrcSize, interleaved->TempBuffer,
Packit 1fb8d4
			                         scanline, nSrcWidth, nSrcHeight))
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case 16:
Packit 1fb8d4
		case 15:
Packit 1fb8d4
			if (!RleDecompress16to16(pSrcData, SrcSize, interleaved->TempBuffer,
Packit 1fb8d4
			                         scanline, nSrcWidth, nSrcHeight))
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case 8:
Packit 1fb8d4
			if (!RleDecompress8to8(pSrcData, SrcSize, interleaved->TempBuffer,
Packit 1fb8d4
			                       scanline, nSrcWidth, nSrcHeight))
Packit 1fb8d4
				return FALSE;
Packit 1fb8d4
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return freerdp_image_copy(pDstData, DstFormat, nDstStep, nXDst, nYDst,
Packit 1fb8d4
	                          nDstWidth, nDstHeight, interleaved->TempBuffer,
Packit 1fb8d4
	                          SrcFormat, scanline, 0, 0, palette, FREERDP_FLIP_VERTICAL);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL interleaved_compress(BITMAP_INTERLEAVED_CONTEXT* interleaved,
Packit 1fb8d4
                          BYTE* pDstData, UINT32* pDstSize,
Packit 1fb8d4
                          UINT32 nWidth, UINT32 nHeight,
Packit 1fb8d4
                          const BYTE* pSrcData, UINT32 SrcFormat,
Packit 1fb8d4
                          UINT32 nSrcStep, UINT32 nXSrc, UINT32 nYSrc,
Packit 1fb8d4
                          const gdiPalette* palette, 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 1fb8d4
		         "interleaved_compress: width (%"PRIu32") or height (%"PRIu32") is greater than 64", nWidth,
Packit 1fb8d4
		         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 1fb8d4
	if (!freerdp_image_copy(interleaved->TempBuffer, DstFormat, 0, 0, 0, nWidth,
Packit 1fb8d4
	                        nHeight, pSrcData, SrcFormat, nSrcStep, nXSrc, nYSrc,
Packit 1fb8d4
	                        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 1fb8d4
	if (freerdp_bitmap_compress(interleaved->TempBuffer, nWidth, nHeight,
Packit 1fb8d4
	                            s, bpp, maxSize, nHeight - 1,
Packit 1fb8d4
	                            interleaved->bts, 0) < 0)
Packit 1fb8d4
		status = FALSE;
Packit 1fb8d4
	else
Packit 1fb8d4
		status = TRUE;
Packit 1fb8d4
Packit 1fb8d4
	Stream_SealLength(s);
Packit 1fb8d4
	*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 1fb8d4
	interleaved = (BITMAP_INTERLEAVED_CONTEXT*) calloc(1,
Packit 1fb8d4
	              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
}