Coverage Report

Created: 2023-11-19 06:16

/src/FreeRDP/libfreerdp/codec/interleaved.c
Line
Count
Source (jump to first uncovered line)
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * Interleaved RLE Bitmap Codec
4
 *
5
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
6
 * Copyright 2015 Thincast Technologies GmbH
7
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
8
 * Copyright 2016 Armin Novak <armin.novak@thincast.com>
9
 * Copyright 2016 Thincast Technologies GmbH
10
 *
11
 * Licensed under the Apache License, Version 2.0 (the "License");
12
 * you may not use this file except in compliance with the License.
13
 * You may obtain a copy of the License at
14
 *
15
 *     http://www.apache.org/licenses/LICENSE-2.0
16
 *
17
 * Unless required by applicable law or agreed to in writing, software
18
 * distributed under the License is distributed on an "AS IS" BASIS,
19
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20
 * See the License for the specific language governing permissions and
21
 * limitations under the License.
22
 */
23
24
#include <winpr/assert.h>
25
#include <freerdp/config.h>
26
27
#include <freerdp/codec/interleaved.h>
28
#include <freerdp/log.h>
29
30
#define TAG FREERDP_TAG("codec")
31
32
#define UNROLL_BODY(_exp, _count)      \
33
0
  do                                 \
34
0
  {                                  \
35
0
    size_t x;                      \
36
0
    for (x = 0; x < (_count); x++) \
37
0
    {                              \
38
0
      do                         \
39
0
      {                          \
40
0
        _exp                   \
41
0
      } while (FALSE);           \
42
0
    }                              \
43
0
  } while (FALSE)
44
45
#define UNROLL_MULTIPLE(_condition, _exp, _count) \
46
0
  do                                            \
47
0
  {                                             \
48
0
    while ((_condition) >= _count)            \
49
0
    {                                         \
50
0
      UNROLL_BODY(_exp, _count);            \
51
0
      (_condition) -= _count;               \
52
0
    }                                         \
53
0
  } while (FALSE)
54
55
#define UNROLL(_condition, _exp)               \
56
0
  do                                         \
57
0
  {                                          \
58
0
    UNROLL_MULTIPLE(_condition, _exp, 16); \
59
0
    UNROLL_MULTIPLE(_condition, _exp, 4);  \
60
0
    UNROLL_MULTIPLE(_condition, _exp, 1);  \
61
0
  } while (FALSE)
62
63
/*
64
   RLE Compressed Bitmap Stream (RLE_BITMAP_STREAM)
65
   http://msdn.microsoft.com/en-us/library/cc240895%28v=prot.10%29.aspx
66
   pseudo-code
67
   http://msdn.microsoft.com/en-us/library/dd240593%28v=prot.10%29.aspx
68
*/
69
70
0
#define REGULAR_BG_RUN 0x00
71
0
#define MEGA_MEGA_BG_RUN 0xF0
72
0
#define REGULAR_FG_RUN 0x01
73
0
#define MEGA_MEGA_FG_RUN 0xF1
74
0
#define LITE_SET_FG_FG_RUN 0x0C
75
0
#define MEGA_MEGA_SET_FG_RUN 0xF6
76
0
#define LITE_DITHERED_RUN 0x0E
77
0
#define MEGA_MEGA_DITHERED_RUN 0xF8
78
0
#define REGULAR_COLOR_RUN 0x03
79
0
#define MEGA_MEGA_COLOR_RUN 0xF3
80
0
#define REGULAR_FGBG_IMAGE 0x02
81
0
#define MEGA_MEGA_FGBG_IMAGE 0xF2
82
0
#define LITE_SET_FG_FGBG_IMAGE 0x0D
83
0
#define MEGA_MEGA_SET_FGBG_IMAGE 0xF7
84
0
#define REGULAR_COLOR_IMAGE 0x04
85
0
#define MEGA_MEGA_COLOR_IMAGE 0xF4
86
0
#define SPECIAL_FGBG_1 0xF9
87
0
#define SPECIAL_FGBG_2 0xFA
88
0
#define SPECIAL_WHITE 0xFD
89
0
#define SPECIAL_BLACK 0xFE
90
91
#define BLACK_PIXEL 0x000000
92
93
typedef UINT32 PIXEL;
94
95
static const BYTE g_MaskSpecialFgBg1 = 0x03;
96
static const BYTE g_MaskSpecialFgBg2 = 0x05;
97
98
static const BYTE g_MaskRegularRunLength = 0x1F;
99
static const BYTE g_MaskLiteRunLength = 0x0F;
100
101
static const char* rle_code_str(UINT32 code)
102
0
{
103
0
  switch (code)
104
0
  {
105
0
    case REGULAR_BG_RUN:
106
0
      return "REGULAR_BG_RUN";
107
0
    case MEGA_MEGA_BG_RUN:
108
0
      return "MEGA_MEGA_BG_RUN";
109
0
    case REGULAR_FG_RUN:
110
0
      return "REGULAR_FG_RUN";
111
0
    case MEGA_MEGA_FG_RUN:
112
0
      return "MEGA_MEGA_FG_RUN";
113
0
    case LITE_SET_FG_FG_RUN:
114
0
      return "LITE_SET_FG_FG_RUN";
115
0
    case MEGA_MEGA_SET_FG_RUN:
116
0
      return "MEGA_MEGA_SET_FG_RUN";
117
0
    case LITE_DITHERED_RUN:
118
0
      return "LITE_DITHERED_RUN";
119
0
    case MEGA_MEGA_DITHERED_RUN:
120
0
      return "MEGA_MEGA_DITHERED_RUN";
121
0
    case REGULAR_COLOR_RUN:
122
0
      return "REGULAR_COLOR_RUN";
123
0
    case MEGA_MEGA_COLOR_RUN:
124
0
      return "MEGA_MEGA_COLOR_RUN";
125
0
    case REGULAR_FGBG_IMAGE:
126
0
      return "REGULAR_FGBG_IMAGE";
127
0
    case MEGA_MEGA_FGBG_IMAGE:
128
0
      return "MEGA_MEGA_FGBG_IMAGE";
129
0
    case LITE_SET_FG_FGBG_IMAGE:
130
0
      return "LITE_SET_FG_FGBG_IMAGE";
131
0
    case MEGA_MEGA_SET_FGBG_IMAGE:
132
0
      return "MEGA_MEGA_SET_FGBG_IMAGE";
133
0
    case REGULAR_COLOR_IMAGE:
134
0
      return "REGULAR_COLOR_IMAGE";
135
0
    case MEGA_MEGA_COLOR_IMAGE:
136
0
      return "MEGA_MEGA_COLOR_IMAGE";
137
0
    case SPECIAL_FGBG_1:
138
0
      return "SPECIAL_FGBG_1";
139
0
    case SPECIAL_FGBG_2:
140
0
      return "SPECIAL_FGBG_2";
141
0
    case SPECIAL_WHITE:
142
0
      return "SPECIAL_WHITE";
143
0
    case SPECIAL_BLACK:
144
0
      return "SPECIAL_BLACK";
145
0
    default:
146
0
      return "UNKNOWN";
147
0
  }
148
0
}
149
150
static const char* rle_code_str_buffer(UINT32 code, char* buffer, size_t size)
151
0
{
152
0
  const char* str = rle_code_str(code);
153
0
  _snprintf(buffer, size, "%s [0x%08" PRIx32 "]", str, code);
154
0
  return buffer;
155
0
}
156
157
#define buffer_within_range(pbSrc, size, pbEnd) \
158
0
  buffer_within_range_((pbSrc), (size), (pbEnd), __func__, __FILE__, __LINE__)
159
static INLINE BOOL buffer_within_range_(const void* pbSrc, size_t size, const void* pbEnd,
160
                                        const char* fkt, const char* file, size_t line)
161
0
{
162
0
  WINPR_UNUSED(file);
163
0
  WINPR_ASSERT(pbSrc);
164
0
  WINPR_ASSERT(pbEnd);
165
166
0
  if ((const char*)pbSrc + size > (const char*)pbEnd)
167
0
  {
168
0
    WLog_ERR(TAG, "[%s:%" PRIuz "] pbSrc=%p + %" PRIuz " > pbEnd=%p", fkt, line, pbSrc, size,
169
0
             pbEnd);
170
0
    return FALSE;
171
0
  }
172
0
  return TRUE;
173
0
}
174
175
/**
176
 * Reads the supplied order header and extracts the compression
177
 * order code ID.
178
 */
179
static INLINE UINT32 ExtractCodeId(BYTE bOrderHdr)
180
0
{
181
0
  if ((bOrderHdr & 0xC0U) != 0xC0U)
182
0
  {
183
    /* REGULAR orders
184
     * (000x xxxx, 001x xxxx, 010x xxxx, 011x xxxx, 100x xxxx)
185
     */
186
0
    return bOrderHdr >> 5;
187
0
  }
188
0
  else if ((bOrderHdr & 0xF0U) == 0xF0U)
189
0
  {
190
    /* MEGA and SPECIAL orders (0xF*) */
191
0
    return bOrderHdr;
192
0
  }
193
0
  else
194
0
  {
195
    /* LITE orders
196
     * 1100 xxxx, 1101 xxxx, 1110 xxxx)
197
     */
198
0
    return bOrderHdr >> 4;
199
0
  }
200
0
}
201
202
/**
203
 * Extract the run length of a compression order.
204
 */
205
static UINT ExtractRunLengthRegularFgBg(const BYTE* pbOrderHdr, const BYTE* pbEnd, UINT32* advance)
206
0
{
207
0
  UINT runLength;
208
209
0
  WINPR_ASSERT(pbOrderHdr);
210
0
  WINPR_ASSERT(pbEnd);
211
0
  WINPR_ASSERT(advance);
212
213
0
  runLength = (*pbOrderHdr) & g_MaskRegularRunLength;
214
0
  if (runLength == 0)
215
0
  {
216
0
    if (!buffer_within_range(pbOrderHdr, 1, pbEnd))
217
0
    {
218
0
      *advance = 0;
219
0
      return 0;
220
0
    }
221
0
    runLength = *(pbOrderHdr + 1) + 1;
222
0
    (*advance)++;
223
0
  }
224
0
  else
225
0
    runLength = runLength * 8;
226
227
0
  return runLength;
228
0
}
229
230
static UINT ExtractRunLengthLiteFgBg(const BYTE* pbOrderHdr, const BYTE* pbEnd, UINT32* advance)
231
0
{
232
0
  UINT runLength;
233
234
0
  WINPR_ASSERT(pbOrderHdr);
235
0
  WINPR_ASSERT(pbEnd);
236
0
  WINPR_ASSERT(advance);
237
238
0
  runLength = *pbOrderHdr & g_MaskLiteRunLength;
239
0
  if (runLength == 0)
240
0
  {
241
0
    if (!buffer_within_range(pbOrderHdr, 1, pbEnd))
242
0
    {
243
0
      *advance = 0;
244
0
      return 0;
245
0
    }
246
0
    runLength = *(pbOrderHdr + 1) + 1;
247
0
    (*advance)++;
248
0
  }
249
0
  else
250
0
    runLength = runLength * 8;
251
252
0
  return runLength;
253
0
}
254
255
static UINT ExtractRunLengthRegular(const BYTE* pbOrderHdr, const BYTE* pbEnd, UINT32* advance)
256
0
{
257
0
  UINT runLength;
258
259
0
  WINPR_ASSERT(pbOrderHdr);
260
0
  WINPR_ASSERT(pbEnd);
261
0
  WINPR_ASSERT(advance);
262
263
0
  runLength = *pbOrderHdr & g_MaskRegularRunLength;
264
0
  if (runLength == 0)
265
0
  {
266
0
    if (!buffer_within_range(pbOrderHdr, 1, pbEnd))
267
0
    {
268
0
      *advance = 0;
269
0
      return 0;
270
0
    }
271
0
    runLength = *(pbOrderHdr + 1) + 32;
272
0
    (*advance)++;
273
0
  }
274
275
0
  return runLength;
276
0
}
277
278
static UINT ExtractRunLengthMegaMega(const BYTE* pbOrderHdr, const BYTE* pbEnd, UINT32* advance)
279
0
{
280
0
  UINT runLength;
281
282
0
  WINPR_ASSERT(pbOrderHdr);
283
0
  WINPR_ASSERT(pbEnd);
284
0
  WINPR_ASSERT(advance);
285
286
0
  if (!buffer_within_range(pbOrderHdr, 2, pbEnd))
287
0
  {
288
0
    *advance = 0;
289
0
    return 0;
290
0
  }
291
292
0
  runLength = ((UINT16)pbOrderHdr[1]) | (((UINT16)pbOrderHdr[2]) << 8);
293
0
  (*advance) += 2;
294
295
0
  return runLength;
296
0
}
297
298
static UINT ExtractRunLengthLite(const BYTE* pbOrderHdr, const BYTE* pbEnd, UINT32* advance)
299
0
{
300
0
  UINT runLength;
301
302
0
  WINPR_ASSERT(pbOrderHdr);
303
0
  WINPR_ASSERT(pbEnd);
304
0
  WINPR_ASSERT(advance);
305
306
0
  runLength = *pbOrderHdr & g_MaskLiteRunLength;
307
0
  if (runLength == 0)
308
0
  {
309
0
    if (!buffer_within_range(pbOrderHdr, 1, pbEnd))
310
0
    {
311
0
      *advance = 0;
312
0
      return 0;
313
0
    }
314
0
    runLength = *(pbOrderHdr + 1) + 16;
315
0
    (*advance)++;
316
0
  }
317
0
  return runLength;
318
0
}
319
320
static INLINE UINT32 ExtractRunLength(UINT32 code, const BYTE* pbOrderHdr, const BYTE* pbEnd,
321
                                      UINT32* advance)
322
0
{
323
0
  UINT32 runLength = 0;
324
0
  UINT32 ladvance = 1;
325
326
0
  WINPR_ASSERT(pbOrderHdr);
327
0
  WINPR_ASSERT(pbEnd);
328
0
  WINPR_ASSERT(advance);
329
330
0
  *advance = 0;
331
0
  if (!buffer_within_range(pbOrderHdr, 0, pbEnd))
332
0
    return 0;
333
334
0
  switch (code)
335
0
  {
336
0
    case REGULAR_FGBG_IMAGE:
337
0
      runLength = ExtractRunLengthRegularFgBg(pbOrderHdr, pbEnd, &ladvance);
338
0
      break;
339
340
0
    case LITE_SET_FG_FGBG_IMAGE:
341
0
      runLength = ExtractRunLengthLiteFgBg(pbOrderHdr, pbEnd, &ladvance);
342
0
      break;
343
344
0
    case REGULAR_BG_RUN:
345
0
    case REGULAR_FG_RUN:
346
0
    case REGULAR_COLOR_RUN:
347
0
    case REGULAR_COLOR_IMAGE:
348
0
      runLength = ExtractRunLengthRegular(pbOrderHdr, pbEnd, &ladvance);
349
0
      break;
350
351
0
    case LITE_SET_FG_FG_RUN:
352
0
    case LITE_DITHERED_RUN:
353
0
      runLength = ExtractRunLengthLite(pbOrderHdr, pbEnd, &ladvance);
354
0
      break;
355
356
0
    case MEGA_MEGA_BG_RUN:
357
0
    case MEGA_MEGA_FG_RUN:
358
0
    case MEGA_MEGA_SET_FG_RUN:
359
0
    case MEGA_MEGA_DITHERED_RUN:
360
0
    case MEGA_MEGA_COLOR_RUN:
361
0
    case MEGA_MEGA_FGBG_IMAGE:
362
0
    case MEGA_MEGA_SET_FGBG_IMAGE:
363
0
    case MEGA_MEGA_COLOR_IMAGE:
364
0
      runLength = ExtractRunLengthMegaMega(pbOrderHdr, pbEnd, &ladvance);
365
0
      break;
366
367
0
    default:
368
0
      runLength = 0;
369
0
      ladvance = 0;
370
0
      break;
371
0
  }
372
373
0
  *advance = ladvance;
374
0
  return runLength;
375
0
}
376
377
#define ensure_capacity(start, end, size, base) \
378
0
  ensure_capacity_((start), (end), (size), (base), __func__, __FILE__, __LINE__)
379
static INLINE BOOL ensure_capacity_(const BYTE* start, const BYTE* end, size_t size, size_t base,
380
                                    const char* fkt, const char* file, size_t line)
381
0
{
382
0
  const size_t available = (uintptr_t)end - (uintptr_t)start;
383
0
  const BOOL rc = available >= size * base;
384
0
  const BOOL res = rc && (start <= end);
385
386
0
  if (!res)
387
0
    WLog_ERR(TAG,
388
0
             "[%s:%" PRIuz "] failed: start=%p <= end=%p, available=%" PRIuz " >= size=%" PRIuz
389
0
             " * base=%" PRIuz,
390
0
             fkt, line, start, end, available, size, base);
391
0
  return res;
392
0
}
393
394
static INLINE void write_pixel_8(BYTE* _buf, BYTE _pix)
395
0
{
396
0
  WINPR_ASSERT(_buf);
397
0
  *_buf = _pix;
398
0
}
399
400
static INLINE void write_pixel_24(BYTE* _buf, UINT32 _pix)
401
0
{
402
0
  WINPR_ASSERT(_buf);
403
0
  (_buf)[0] = (BYTE)(_pix);
404
0
  (_buf)[1] = (BYTE)((_pix) >> 8);
405
0
  (_buf)[2] = (BYTE)((_pix) >> 16);
406
0
}
407
408
static INLINE void write_pixel_16(BYTE* _buf, UINT16 _pix)
409
0
{
410
0
  WINPR_ASSERT(_buf);
411
0
  _buf[0] = _pix & 0xFF;
412
0
  _buf[1] = (_pix >> 8) & 0xFF;
413
0
}
414
415
#undef DESTWRITEPIXEL
416
#undef DESTREADPIXEL
417
#undef SRCREADPIXEL
418
#undef WRITEFGBGIMAGE
419
#undef WRITEFIRSTLINEFGBGIMAGE
420
#undef RLEDECOMPRESS
421
#undef RLEEXTRA
422
#undef WHITE_PIXEL
423
#undef PIXEL_SIZE
424
#undef PIXEL
425
#define PIXEL_SIZE 1
426
0
#define PIXEL BYTE
427
0
#define WHITE_PIXEL 0xFF
428
#define DESTWRITEPIXEL(_buf, _pix) \
429
0
  do                             \
430
0
  {                              \
431
0
    write_pixel_8(_buf, _pix); \
432
0
    _buf += 1;                 \
433
0
  } while (0)
434
0
#define DESTREADPIXEL(_pix, _buf) _pix = (_buf)[0]
435
#define SRCREADPIXEL(_pix, _buf) \
436
0
  do                           \
437
0
  {                            \
438
0
    _pix = (_buf)[0];        \
439
0
    _buf += 1;               \
440
0
  } while (0)
441
442
0
#define WRITEFGBGIMAGE WriteFgBgImage8to8
443
0
#define WRITEFIRSTLINEFGBGIMAGE WriteFirstLineFgBgImage8to8
444
#define RLEDECOMPRESS RleDecompress8to8
445
#define RLEEXTRA
446
#undef ENSURE_CAPACITY
447
0
#define ENSURE_CAPACITY(_start, _end, _size) ensure_capacity(_start, _end, _size, 1)
448
#include "include/bitmap.c"
449
450
#undef DESTWRITEPIXEL
451
#undef DESTREADPIXEL
452
#undef SRCREADPIXEL
453
#undef WRITEFGBGIMAGE
454
#undef WRITEFIRSTLINEFGBGIMAGE
455
#undef RLEDECOMPRESS
456
#undef RLEEXTRA
457
#undef WHITE_PIXEL
458
#undef PIXEL_SIZE
459
#undef PIXEL
460
#define PIXEL_SIZE 2
461
0
#define PIXEL UINT16
462
0
#define WHITE_PIXEL 0xFFFF
463
#define DESTWRITEPIXEL(_buf, _pix)  \
464
0
  do                              \
465
0
  {                               \
466
0
    write_pixel_16(_buf, _pix); \
467
0
    _buf += 2;                  \
468
0
  } while (0)
469
0
#define DESTREADPIXEL(_pix, _buf) _pix = ((UINT16*)(_buf))[0]
470
#define SRCREADPIXEL(_pix, _buf)             \
471
0
  do                                       \
472
0
  {                                        \
473
0
    _pix = (_buf)[0] | ((_buf)[1] << 8); \
474
0
    _buf += 2;                           \
475
0
  } while (0)
476
0
#define WRITEFGBGIMAGE WriteFgBgImage16to16
477
0
#define WRITEFIRSTLINEFGBGIMAGE WriteFirstLineFgBgImage16to16
478
#define RLEDECOMPRESS RleDecompress16to16
479
#define RLEEXTRA
480
#undef ENSURE_CAPACITY
481
0
#define ENSURE_CAPACITY(_start, _end, _size) ensure_capacity(_start, _end, _size, 2)
482
#include "include/bitmap.c"
483
484
#undef DESTWRITEPIXEL
485
#undef DESTREADPIXEL
486
#undef SRCREADPIXEL
487
#undef WRITEFGBGIMAGE
488
#undef WRITEFIRSTLINEFGBGIMAGE
489
#undef RLEDECOMPRESS
490
#undef RLEEXTRA
491
#undef WHITE_PIXEL
492
#undef PIXEL_SIZE
493
#undef PIXEL
494
#define PIXEL_SIZE 3
495
0
#define PIXEL UINT32
496
0
#define WHITE_PIXEL 0xffffff
497
#define DESTWRITEPIXEL(_buf, _pix)  \
498
0
  do                              \
499
0
  {                               \
500
0
    write_pixel_24(_buf, _pix); \
501
0
    _buf += 3;                  \
502
0
  } while (0)
503
0
#define DESTREADPIXEL(_pix, _buf) _pix = (_buf)[0] | ((_buf)[1] << 8) | ((_buf)[2] << 16)
504
#define SRCREADPIXEL(_pix, _buf)                                 \
505
0
  do                                                           \
506
0
  {                                                            \
507
0
    _pix = (_buf)[0] | ((_buf)[1] << 8) | ((_buf)[2] << 16); \
508
0
    _buf += 3;                                               \
509
0
  } while (0)
510
511
0
#define WRITEFGBGIMAGE WriteFgBgImage24to24
512
0
#define WRITEFIRSTLINEFGBGIMAGE WriteFirstLineFgBgImage24to24
513
#define RLEDECOMPRESS RleDecompress24to24
514
#define RLEEXTRA
515
#undef ENSURE_CAPACITY
516
0
#define ENSURE_CAPACITY(_start, _end, _size) ensure_capacity(_start, _end, _size, 3)
517
#include "include/bitmap.c"
518
519
struct S_BITMAP_INTERLEAVED_CONTEXT
520
{
521
  BOOL Compressor;
522
523
  UINT32 TempSize;
524
  BYTE* TempBuffer;
525
526
  wStream* bts;
527
};
528
529
BOOL interleaved_decompress(BITMAP_INTERLEAVED_CONTEXT* interleaved, const BYTE* pSrcData,
530
                            UINT32 SrcSize, UINT32 nSrcWidth, UINT32 nSrcHeight, UINT32 bpp,
531
                            BYTE* pDstData, UINT32 DstFormat, UINT32 nDstStep, UINT32 nXDst,
532
                            UINT32 nYDst, UINT32 nDstWidth, UINT32 nDstHeight,
533
                            const gdiPalette* palette)
534
0
{
535
0
  UINT32 scanline;
536
0
  UINT32 SrcFormat;
537
0
  UINT32 BufferSize;
538
539
0
  if (!interleaved || !pSrcData || !pDstData)
540
0
  {
541
0
    WLog_ERR(TAG, "invalid arguments: interleaved=%p, pSrcData=%p, pDstData=%p", interleaved,
542
0
             pSrcData, pDstData);
543
0
    return FALSE;
544
0
  }
545
546
0
  switch (bpp)
547
0
  {
548
0
    case 24:
549
0
      scanline = nSrcWidth * 3;
550
0
      SrcFormat = PIXEL_FORMAT_BGR24;
551
0
      break;
552
553
0
    case 16:
554
0
      scanline = nSrcWidth * 2;
555
0
      SrcFormat = PIXEL_FORMAT_RGB16;
556
0
      break;
557
558
0
    case 15:
559
0
      scanline = nSrcWidth * 2;
560
0
      SrcFormat = PIXEL_FORMAT_RGB15;
561
0
      break;
562
563
0
    case 8:
564
0
      scanline = nSrcWidth;
565
0
      SrcFormat = PIXEL_FORMAT_RGB8;
566
0
      break;
567
568
0
    default:
569
0
      WLog_ERR(TAG, "Invalid color depth %" PRIu32 "", bpp);
570
0
      return FALSE;
571
0
  }
572
573
0
  BufferSize = scanline * nSrcHeight;
574
575
0
  if (BufferSize > interleaved->TempSize)
576
0
  {
577
0
    interleaved->TempBuffer =
578
0
        winpr_aligned_recalloc(interleaved->TempBuffer, BufferSize, sizeof(BYTE), 16);
579
0
    interleaved->TempSize = BufferSize;
580
0
  }
581
582
0
  if (!interleaved->TempBuffer)
583
0
  {
584
0
    WLog_ERR(TAG, "interleaved->TempBuffer=%p", interleaved->TempBuffer);
585
0
    return FALSE;
586
0
  }
587
588
0
  switch (bpp)
589
0
  {
590
0
    case 24:
591
0
      if (!RleDecompress24to24(pSrcData, SrcSize, interleaved->TempBuffer, scanline,
592
0
                               nSrcWidth, nSrcHeight))
593
0
      {
594
0
        WLog_ERR(TAG, "RleDecompress24to24 failed");
595
0
        return FALSE;
596
0
      }
597
598
0
      break;
599
600
0
    case 16:
601
0
    case 15:
602
0
      if (!RleDecompress16to16(pSrcData, SrcSize, interleaved->TempBuffer, scanline,
603
0
                               nSrcWidth, nSrcHeight))
604
0
      {
605
0
        WLog_ERR(TAG, "RleDecompress16to16 failed");
606
0
        return FALSE;
607
0
      }
608
609
0
      break;
610
611
0
    case 8:
612
0
      if (!RleDecompress8to8(pSrcData, SrcSize, interleaved->TempBuffer, scanline, nSrcWidth,
613
0
                             nSrcHeight))
614
0
      {
615
0
        WLog_ERR(TAG, "RleDecompress8to8 failed");
616
0
        return FALSE;
617
0
      }
618
619
0
      break;
620
621
0
    default:
622
0
      WLog_ERR(TAG, "Invalid color depth %" PRIu32 "", bpp);
623
0
      return FALSE;
624
0
  }
625
626
0
  if (!freerdp_image_copy(pDstData, DstFormat, nDstStep, nXDst, nYDst, nDstWidth, nDstHeight,
627
0
                          interleaved->TempBuffer, SrcFormat, scanline, 0, 0, palette,
628
0
                          FREERDP_FLIP_VERTICAL | FREERDP_KEEP_DST_ALPHA))
629
0
  {
630
0
    WLog_ERR(TAG, "freerdp_image_copy failed");
631
0
    return FALSE;
632
0
  }
633
0
  return TRUE;
634
0
}
635
636
BOOL interleaved_compress(BITMAP_INTERLEAVED_CONTEXT* interleaved, BYTE* pDstData, UINT32* pDstSize,
637
                          UINT32 nWidth, UINT32 nHeight, const BYTE* pSrcData, UINT32 SrcFormat,
638
                          UINT32 nSrcStep, UINT32 nXSrc, UINT32 nYSrc, const gdiPalette* palette,
639
                          UINT32 bpp)
640
0
{
641
0
  BOOL status;
642
0
  wStream* s;
643
0
  UINT32 DstFormat = 0;
644
0
  const UINT32 maxSize = 64 * 64 * 4;
645
646
0
  if (!interleaved || !pDstData || !pSrcData)
647
0
    return FALSE;
648
649
0
  if ((nWidth == 0) || (nHeight == 0))
650
0
    return FALSE;
651
652
0
  if (nWidth % 4)
653
0
  {
654
0
    WLog_ERR(TAG, "interleaved_compress: width is not a multiple of 4");
655
0
    return FALSE;
656
0
  }
657
658
0
  if ((nWidth > 64) || (nHeight > 64))
659
0
  {
660
0
    WLog_ERR(TAG,
661
0
             "interleaved_compress: width (%" PRIu32 ") or height (%" PRIu32
662
0
             ") is greater than 64",
663
0
             nWidth, nHeight);
664
0
    return FALSE;
665
0
  }
666
667
0
  switch (bpp)
668
0
  {
669
0
    case 24:
670
0
      DstFormat = PIXEL_FORMAT_BGRX32;
671
0
      break;
672
673
0
    case 16:
674
0
      DstFormat = PIXEL_FORMAT_RGB16;
675
0
      break;
676
677
0
    case 15:
678
0
      DstFormat = PIXEL_FORMAT_RGB15;
679
0
      break;
680
681
0
    default:
682
0
      return FALSE;
683
0
  }
684
685
0
  if (!freerdp_image_copy(interleaved->TempBuffer, DstFormat, 0, 0, 0, nWidth, nHeight, pSrcData,
686
0
                          SrcFormat, nSrcStep, nXSrc, nYSrc, palette, FREERDP_KEEP_DST_ALPHA))
687
0
    return FALSE;
688
689
0
  s = Stream_New(pDstData, *pDstSize);
690
691
0
  if (!s)
692
0
    return FALSE;
693
694
0
  Stream_SetPosition(interleaved->bts, 0);
695
696
0
  if (freerdp_bitmap_compress(interleaved->TempBuffer, nWidth, nHeight, s, bpp, maxSize,
697
0
                              nHeight - 1, interleaved->bts, 0) < 0)
698
0
    status = FALSE;
699
0
  else
700
0
    status = TRUE;
701
702
0
  Stream_SealLength(s);
703
0
  *pDstSize = (UINT32)Stream_Length(s);
704
0
  Stream_Free(s, FALSE);
705
0
  return status;
706
0
}
707
708
BOOL bitmap_interleaved_context_reset(BITMAP_INTERLEAVED_CONTEXT* interleaved)
709
0
{
710
0
  if (!interleaved)
711
0
    return FALSE;
712
713
0
  return TRUE;
714
0
}
715
716
BITMAP_INTERLEAVED_CONTEXT* bitmap_interleaved_context_new(BOOL Compressor)
717
0
{
718
0
  BITMAP_INTERLEAVED_CONTEXT* interleaved;
719
0
  interleaved = (BITMAP_INTERLEAVED_CONTEXT*)winpr_aligned_recalloc(
720
0
      NULL, 1, sizeof(BITMAP_INTERLEAVED_CONTEXT), 32);
721
722
0
  if (interleaved)
723
0
  {
724
0
    interleaved->TempSize = 64 * 64 * 4;
725
0
    interleaved->TempBuffer = winpr_aligned_calloc(interleaved->TempSize, sizeof(BYTE), 16);
726
727
0
    if (!interleaved->TempBuffer)
728
0
      goto fail;
729
730
0
    interleaved->bts = Stream_New(NULL, interleaved->TempSize);
731
732
0
    if (!interleaved->bts)
733
0
      goto fail;
734
0
  }
735
736
0
  return interleaved;
737
738
0
fail:
739
0
  bitmap_interleaved_context_free(interleaved);
740
0
  return NULL;
741
0
}
742
743
void bitmap_interleaved_context_free(BITMAP_INTERLEAVED_CONTEXT* interleaved)
744
0
{
745
0
  if (!interleaved)
746
0
    return;
747
748
0
  winpr_aligned_free(interleaved->TempBuffer);
749
0
  Stream_Free(interleaved->bts, TRUE);
750
0
  winpr_aligned_free(interleaved);
751
0
}