|
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 |
}
|