22 #include <freerdp/config.h>
28 #include <winpr/assert.h>
29 #include <winpr/cast.h>
30 #include <winpr/crt.h>
32 #include <freerdp/codec/nsc.h>
33 #include <freerdp/codec/color.h>
35 #include "nsc_types.h"
36 #include "nsc_encode.h"
48 BYTE* PlaneBuffers[5];
49 UINT32 OrgByteCount[4];
51 UINT32 LumaPlaneByteCount;
52 UINT32 OrangeChromaPlaneByteCount;
53 UINT32 GreenChromaPlaneByteCount;
54 UINT32 AlphaPlaneByteCount;
56 UINT8 ChromaSubsamplingLevel;
59 static BOOL nsc_write_message(NSC_CONTEXT* WINPR_RESTRICT context,
wStream* WINPR_RESTRICT s,
60 const NSC_MESSAGE* WINPR_RESTRICT message);
62 static BOOL nsc_context_initialize_encode(NSC_CONTEXT* WINPR_RESTRICT context)
66 UINT32 tempHeight = 0;
67 tempWidth = ROUND_UP_TO(context->width, 8);
68 tempHeight = ROUND_UP_TO(context->height, 2);
70 length = tempWidth * tempHeight + 16;
72 if (length > context->priv->PlaneBuffersLength)
74 for (
int i = 0; i < 5; i++)
76 BYTE* tmp = (BYTE*)winpr_aligned_recalloc(context->priv->PlaneBuffers[i], length,
82 context->priv->PlaneBuffers[i] = tmp;
85 context->priv->PlaneBuffersLength = length;
88 if (context->ChromaSubsamplingLevel)
90 context->OrgByteCount[0] = tempWidth * context->height;
91 context->OrgByteCount[1] = tempWidth * tempHeight / 4;
92 context->OrgByteCount[2] = tempWidth * tempHeight / 4;
93 context->OrgByteCount[3] = context->width * context->height;
97 context->OrgByteCount[0] = context->width * context->height;
98 context->OrgByteCount[1] = context->width * context->height;
99 context->OrgByteCount[2] = context->width * context->height;
100 context->OrgByteCount[3] = context->width * context->height;
106 if (length > context->priv->PlaneBuffersLength)
108 for (
int i = 0; i < 5; i++)
109 winpr_aligned_free(context->priv->PlaneBuffers[i]);
115 static BOOL nsc_encode_argb_to_aycocg(NSC_CONTEXT* WINPR_RESTRICT context,
116 const BYTE* WINPR_RESTRICT data, UINT32 scanline)
119 const BYTE* src = NULL;
121 BYTE* coplane = NULL;
122 BYTE* cgplane = NULL;
129 UINT16 tempWidth = ROUND_UP_TO(context->width, 8);
130 const UINT16 rw = (context->ChromaSubsamplingLevel ? tempWidth : context->width);
131 const BYTE ccl = WINPR_ASSERTING_INT_CAST(BYTE, context->ColorLossLevel);
133 for (; y < context->height; y++)
135 src = data + (context->height - 1 - y) * scanline;
136 yplane = context->priv->PlaneBuffers[0] + y * rw;
137 coplane = context->priv->PlaneBuffers[1] + y * rw;
138 cgplane = context->priv->PlaneBuffers[2] + y * rw;
139 aplane = context->priv->PlaneBuffers[3] + y * context->width;
142 for (; x < context->width; x++)
144 switch (context->format)
146 case PIXEL_FORMAT_BGRX32:
154 case PIXEL_FORMAT_BGRA32:
161 case PIXEL_FORMAT_RGBX32:
169 case PIXEL_FORMAT_RGBA32:
176 case PIXEL_FORMAT_BGR24:
183 case PIXEL_FORMAT_RGB24:
190 case PIXEL_FORMAT_BGR16:
191 b_val = (INT16)(((*(src + 1)) & 0xF8) | ((*(src + 1)) >> 5));
192 g_val = (INT16)((((*(src + 1)) & 0x07) << 5) | (((*src) & 0xE0) >> 3));
193 r_val = (INT16)((((*src) & 0x1F) << 3) | (((*src) >> 2) & 0x07));
198 case PIXEL_FORMAT_RGB16:
199 r_val = (INT16)(((*(src + 1)) & 0xF8) | ((*(src + 1)) >> 5));
200 g_val = (INT16)((((*(src + 1)) & 0x07) << 5) | (((*src) & 0xE0) >> 3));
201 b_val = (INT16)((((*src) & 0x1F) << 3) | (((*src) >> 2) & 0x07));
206 case PIXEL_FORMAT_A4:
210 shift = (7 - (x % 8));
211 idx = ((*src) >> shift) & 1;
212 idx |= (((*(src + 1)) >> shift) & 1) << 1;
213 idx |= (((*(src + 2)) >> shift) & 1) << 2;
214 idx |= (((*(src + 3)) >> shift) & 1) << 3;
216 r_val = (INT16)context->palette[idx];
217 g_val = (INT16)context->palette[idx + 1];
218 b_val = (INT16)context->palette[idx + 2];
227 case PIXEL_FORMAT_RGB8:
229 int idx = (*src) * 3;
230 r_val = (INT16)context->palette[idx];
231 g_val = (INT16)context->palette[idx + 1];
232 b_val = (INT16)context->palette[idx + 2];
240 r_val = g_val = b_val = a_val = 0;
244 *yplane++ = (BYTE)((r_val >> 2) + (g_val >> 1) + (b_val >> 2));
246 *coplane++ = (BYTE)((r_val - b_val) >> ccl);
247 *cgplane++ = (BYTE)((-(r_val >> 1) + g_val - (b_val >> 1)) >> ccl);
251 if (context->ChromaSubsamplingLevel && (x % 2) == 1)
253 *yplane = *(yplane - 1);
254 *coplane = *(coplane - 1);
255 *cgplane = *(cgplane - 1);
259 if (context->ChromaSubsamplingLevel && (y % 2) == 1)
261 yplane = context->priv->PlaneBuffers[0] + y * rw;
262 coplane = context->priv->PlaneBuffers[1] + y * rw;
263 cgplane = context->priv->PlaneBuffers[2] + y * rw;
264 CopyMemory(yplane, yplane - rw, rw);
265 CopyMemory(coplane, coplane - rw, rw);
266 CopyMemory(cgplane, cgplane - rw, rw);
272 static BOOL nsc_encode_subsampling(NSC_CONTEXT* WINPR_RESTRICT context)
274 UINT32 tempWidth = 0;
275 UINT32 tempHeight = 0;
280 tempWidth = ROUND_UP_TO(context->width, 8);
281 tempHeight = ROUND_UP_TO(context->height, 2);
286 if (tempWidth > context->priv->PlaneBuffersLength / tempHeight)
289 for (
size_t y = 0; y < tempHeight >> 1; y++)
291 BYTE* co_dst = context->priv->PlaneBuffers[1] + y * (tempWidth >> 1);
292 BYTE* cg_dst = context->priv->PlaneBuffers[2] + y * (tempWidth >> 1);
293 const INT8* co_src0 = (INT8*)context->priv->PlaneBuffers[1] + (y << 1) * tempWidth;
294 const INT8* co_src1 = co_src0 + tempWidth;
295 const INT8* cg_src0 = (INT8*)context->priv->PlaneBuffers[2] + (y << 1) * tempWidth;
296 const INT8* cg_src1 = cg_src0 + tempWidth;
298 for (UINT32 x = 0; x < tempWidth >> 1; x++)
300 *co_dst++ = (BYTE)(((INT16)*co_src0 + (INT16) * (co_src0 + 1) + (INT16)*co_src1 +
301 (INT16) * (co_src1 + 1)) >>
303 *cg_dst++ = (BYTE)(((INT16)*cg_src0 + (INT16) * (cg_src0 + 1) + (INT16)*cg_src1 +
304 (INT16) * (cg_src1 + 1)) >>
316 BOOL nsc_encode(NSC_CONTEXT* WINPR_RESTRICT context,
const BYTE* WINPR_RESTRICT bmpdata,
319 if (!context || !bmpdata || (rowstride == 0))
322 if (!nsc_encode_argb_to_aycocg(context, bmpdata, rowstride))
325 if (context->ChromaSubsamplingLevel)
327 if (!nsc_encode_subsampling(context))
334 static UINT32 nsc_rle_encode(
const BYTE* WINPR_RESTRICT in, BYTE* WINPR_RESTRICT out,
338 UINT32 runlength = 1;
339 UINT32 planeSize = 0;
346 while (left > 4 && planeSize < originalSize - 4)
348 if (left > 5 && *in == *(in + 1))
352 else if (runlength == 1)
357 else if (runlength < 256)
361 WINPR_ASSERT(runlength >= 2);
362 *out++ = WINPR_ASSERTING_INT_CAST(BYTE, runlength - 2);
371 *out++ = (runlength & 0x000000FF);
372 *out++ = (runlength & 0x0000FF00) >> 8;
373 *out++ = (runlength & 0x00FF0000) >> 16;
374 *out++ = (runlength & 0xFF000000) >> 24;
383 if (planeSize < originalSize - 4)
384 CopyMemory(out, in, 4);
390 static void nsc_rle_compress_data(NSC_CONTEXT* WINPR_RESTRICT context)
392 UINT32 planeSize = 0;
393 UINT32 originalSize = 0;
395 for (UINT16 i = 0; i < 4; i++)
397 originalSize = context->OrgByteCount[i];
399 if (originalSize == 0)
405 planeSize = nsc_rle_encode(context->priv->PlaneBuffers[i],
406 context->priv->PlaneBuffers[4], originalSize);
408 if (planeSize < originalSize)
409 CopyMemory(context->priv->PlaneBuffers[i], context->priv->PlaneBuffers[4],
412 planeSize = originalSize;
415 context->PlaneByteCount[i] = planeSize;
419 static UINT32 nsc_compute_byte_count(NSC_CONTEXT* WINPR_RESTRICT context,
420 UINT32* WINPR_RESTRICT ByteCount, UINT32 width, UINT32 height)
422 UINT32 tempWidth = 0;
423 UINT32 tempHeight = 0;
424 UINT32 maxPlaneSize = 0;
425 tempWidth = ROUND_UP_TO(width, 8);
426 tempHeight = ROUND_UP_TO(height, 2);
427 maxPlaneSize = tempWidth * tempHeight + 16;
429 if (context->ChromaSubsamplingLevel)
431 ByteCount[0] = tempWidth * height;
432 ByteCount[1] = tempWidth * tempHeight / 4;
433 ByteCount[2] = tempWidth * tempHeight / 4;
434 ByteCount[3] = width * height;
438 ByteCount[0] = width * height;
439 ByteCount[1] = width * height;
440 ByteCount[2] = width * height;
441 ByteCount[3] = width * height;
447 BOOL nsc_write_message(NSC_CONTEXT* WINPR_RESTRICT context,
wStream* WINPR_RESTRICT s,
448 const NSC_MESSAGE* WINPR_RESTRICT message)
450 UINT32 totalPlaneByteCount = 0;
451 totalPlaneByteCount = message->LumaPlaneByteCount + message->OrangeChromaPlaneByteCount +
452 message->GreenChromaPlaneByteCount + message->AlphaPlaneByteCount;
454 if (!Stream_EnsureRemainingCapacity(s, 20 + totalPlaneByteCount))
457 Stream_Write_UINT32(s, message->LumaPlaneByteCount);
459 s, message->OrangeChromaPlaneByteCount);
461 s, message->GreenChromaPlaneByteCount);
462 Stream_Write_UINT32(s, message->AlphaPlaneByteCount);
463 Stream_Write_UINT8(s, message->ColorLossLevel);
464 Stream_Write_UINT8(s, message->ChromaSubsamplingLevel);
465 Stream_Write_UINT16(s, 0);
467 if (message->LumaPlaneByteCount)
468 Stream_Write(s, message->PlaneBuffers[0], message->LumaPlaneByteCount);
470 if (message->OrangeChromaPlaneByteCount)
471 Stream_Write(s, message->PlaneBuffers[1],
472 message->OrangeChromaPlaneByteCount);
474 if (message->GreenChromaPlaneByteCount)
475 Stream_Write(s, message->PlaneBuffers[2],
476 message->GreenChromaPlaneByteCount);
478 if (message->AlphaPlaneByteCount)
479 Stream_Write(s, message->PlaneBuffers[3], message->AlphaPlaneByteCount);
484 BOOL nsc_compose_message(NSC_CONTEXT* WINPR_RESTRICT context,
wStream* WINPR_RESTRICT s,
485 const BYTE* WINPR_RESTRICT data, UINT32 width, UINT32 height,
489 NSC_MESSAGE message = { 0 };
491 if (!context || !s || !data)
494 context->width = WINPR_ASSERTING_INT_CAST(UINT16, width);
495 context->height = WINPR_ASSERTING_INT_CAST(UINT16, height);
497 if (!nsc_context_initialize_encode(context))
501 PROFILER_ENTER(context->priv->prof_nsc_encode)
502 rc = context->encode(context, data, scanline);
503 PROFILER_EXIT(context->priv->prof_nsc_encode)
508 PROFILER_ENTER(context->priv->prof_nsc_rle_compress_data)
509 nsc_rle_compress_data(context);
510 PROFILER_EXIT(context->priv->prof_nsc_rle_compress_data)
511 message.PlaneBuffers[0] = context->priv->PlaneBuffers[0];
512 message.PlaneBuffers[1] = context->priv->PlaneBuffers[1];
513 message.PlaneBuffers[2] = context->priv->PlaneBuffers[2];
514 message.PlaneBuffers[3] = context->priv->PlaneBuffers[3];
515 message.LumaPlaneByteCount = context->PlaneByteCount[0];
516 message.OrangeChromaPlaneByteCount = context->PlaneByteCount[1];
517 message.GreenChromaPlaneByteCount = context->PlaneByteCount[2];
518 message.AlphaPlaneByteCount = context->PlaneByteCount[3];
520 message.ColorLossLevel = WINPR_ASSERTING_INT_CAST(BYTE, context->ColorLossLevel);
521 message.ChromaSubsamplingLevel =
522 WINPR_ASSERTING_INT_CAST(BYTE, context->ChromaSubsamplingLevel);
523 return nsc_write_message(context, s, &message);
526 BOOL nsc_decompose_message(NSC_CONTEXT* WINPR_RESTRICT context,
wStream* WINPR_RESTRICT s,
527 BYTE* WINPR_RESTRICT bmpdata, UINT32 x, UINT32 y, UINT32 width,
528 UINT32 height, UINT32 rowstride, UINT32 format, UINT32 flip)
530 size_t size = Stream_GetRemainingLength(s);
532 if (size > UINT32_MAX)
535 if (!nsc_process_message(context, (UINT16)FreeRDPGetBitsPerPixel(context->format), width,
536 height, Stream_Pointer(s), (UINT32)size, bmpdata, format, rowstride, x,
537 y, width, height, flip))
539 Stream_Seek(s, size);