FreeRDP
Loading...
Searching...
No Matches
gdi/graphics.c
1
22#include <freerdp/config.h>
23
24#include <winpr/crt.h>
25
26#include <freerdp/log.h>
27#include <freerdp/freerdp.h>
28#include <freerdp/gdi/dc.h>
29#include <freerdp/gdi/shape.h>
30#include <freerdp/gdi/region.h>
31#include <freerdp/gdi/bitmap.h>
32
33#include "clipping.h"
34#include "drawing.h"
35#include "brush.h"
36#include "graphics.h"
37
38#define TAG FREERDP_TAG("gdi")
39/* Bitmap Class */
40
41HGDI_BITMAP gdi_create_bitmap(rdpGdi* gdi, UINT32 nWidth, UINT32 nHeight, UINT32 SrcFormat,
42 BYTE* data)
43{
44 UINT32 nSrcStep = 0;
45 UINT32 nDstStep = 0;
46 BYTE* pSrcData = nullptr;
47 BYTE* pDstData = nullptr;
48 HGDI_BITMAP bitmap = nullptr;
49
50 if (!gdi)
51 return nullptr;
52
53 nDstStep = nWidth * FreeRDPGetBytesPerPixel(gdi->dstFormat);
54 pDstData = winpr_aligned_malloc(1ull * nHeight * nDstStep, 16);
55
56 if (!pDstData)
57 return nullptr;
58
59 pSrcData = data;
60 nSrcStep = nWidth * FreeRDPGetBytesPerPixel(SrcFormat);
61
62 if (!freerdp_image_copy_no_overlap(pDstData, gdi->dstFormat, nDstStep, 0, 0, nWidth, nHeight,
63 pSrcData, SrcFormat, nSrcStep, 0, 0, &gdi->palette,
64 FREERDP_FLIP_NONE))
65 {
66 winpr_aligned_free(pDstData);
67 return nullptr;
68 }
69
70 bitmap = gdi_CreateBitmap(nWidth, nHeight, gdi->dstFormat, pDstData);
71 if (!bitmap)
72 winpr_aligned_free(pDstData);
73 return bitmap;
74}
75
76static BOOL gdi_Bitmap_New(rdpContext* context, rdpBitmap* bitmap)
77{
78 gdiBitmap* gdi_bitmap = nullptr;
79 rdpGdi* gdi = context->gdi;
80 gdi_bitmap = (gdiBitmap*)bitmap;
81 gdi_bitmap->hdc = gdi_CreateCompatibleDC(gdi->hdc);
82
83 if (!gdi_bitmap->hdc)
84 return FALSE;
85
86 if (!bitmap->data)
87 gdi_bitmap->bitmap = gdi_CreateCompatibleBitmap(gdi->hdc, bitmap->width, bitmap->height);
88 else
89 {
90 UINT32 format = bitmap->format;
91 gdi_bitmap->bitmap =
92 gdi_create_bitmap(gdi, bitmap->width, bitmap->height, format, bitmap->data);
93 }
94
95 if (!gdi_bitmap->bitmap)
96 {
97 gdi_DeleteDC(gdi_bitmap->hdc);
98 gdi_bitmap->hdc = nullptr;
99 return FALSE;
100 }
101
102 gdi_bitmap->hdc->format = gdi_bitmap->bitmap->format;
103 gdi_SelectObject(gdi_bitmap->hdc, (HGDIOBJECT)gdi_bitmap->bitmap);
104 gdi_bitmap->org_bitmap = nullptr;
105 return TRUE;
106}
107
108static void gdi_Bitmap_Free(WINPR_ATTR_UNUSED rdpContext* context, rdpBitmap* bitmap)
109{
110 gdiBitmap* gdi_bitmap = (gdiBitmap*)bitmap;
111
112 if (gdi_bitmap)
113 {
114 if (gdi_bitmap->hdc)
115 gdi_SelectObject(gdi_bitmap->hdc, (HGDIOBJECT)gdi_bitmap->org_bitmap);
116
117 gdi_DeleteObject((HGDIOBJECT)gdi_bitmap->bitmap);
118 gdi_DeleteDC(gdi_bitmap->hdc);
119 winpr_aligned_free(bitmap->data);
120 }
121
122 free(bitmap);
123}
124
125static BOOL gdi_Bitmap_Paint(rdpContext* context, rdpBitmap* bitmap)
126{
127 gdiBitmap* gdi_bitmap = (gdiBitmap*)bitmap;
128 UINT32 width = bitmap->right - bitmap->left + 1;
129 UINT32 height = bitmap->bottom - bitmap->top + 1;
130 return gdi_BitBlt(context->gdi->primary->hdc, WINPR_ASSERTING_INT_CAST(int, bitmap->left),
131 WINPR_ASSERTING_INT_CAST(int, bitmap->top),
132 WINPR_ASSERTING_INT_CAST(int, width), WINPR_ASSERTING_INT_CAST(int, height),
133 gdi_bitmap->hdc, 0, 0, GDI_SRCCOPY, &context->gdi->palette);
134}
135
136WINPR_ATTR_NODISCARD
137static BOOL gdi_Bitmap_Rfx(rdpContext* context, rdpBitmap* bitmap, const BYTE* pSrcData,
138 UINT32 SrcSize)
139{
140 WINPR_ASSERT(context);
141 WINPR_ASSERT(bitmap);
142 WINPR_ASSERT(pSrcData || (SrcSize == 0));
143
144 REGION16 invalidRegion = WINPR_C_ARRAY_INIT;
145 region16_init(&invalidRegion);
146
147 const UINT32 stride = bitmap->width * FreeRDPGetBytesPerPixel(bitmap->format);
148
149 const BOOL rc =
150 rfx_process_message(context->codecs->rfx, pSrcData, SrcSize, bitmap->left, bitmap->top,
151 bitmap->data, bitmap->format, stride, bitmap->height, &invalidRegion);
152 region16_uninit(&invalidRegion);
153
154 if (!rc)
155 {
156 WLog_ERR(TAG, "rfx_process_message failed");
157 return FALSE;
158 }
159 return TRUE;
160}
161
162static BOOL gdi_Bitmap_Decompress(rdpContext* context, rdpBitmap* bitmap, const BYTE* pSrcData,
163 UINT32 DstWidth, UINT32 DstHeight, UINT32 bpp, UINT32 length,
164 BOOL compressed, UINT32 codecId)
165{
166 WINPR_ASSERT(context);
167 WINPR_ASSERT(bitmap);
168
169 UINT32 SrcSize = length;
170 rdpGdi* gdi = context->gdi;
171 WINPR_ASSERT(gdi);
172
173 bitmap->compressed = FALSE;
174 bitmap->format = gdi->dstFormat;
175
176 if ((FreeRDPGetBytesPerPixel(bitmap->format) == 0) || (DstWidth == 0) || (DstHeight == 0) ||
177 (DstWidth > UINT32_MAX / DstHeight) ||
178 ((DstWidth * DstHeight) > (UINT32_MAX / FreeRDPGetBytesPerPixel(bitmap->format))))
179 {
180 WLog_ERR(TAG, "invalid input data");
181 return FALSE;
182 }
183
184 const UINT32 stride = DstWidth * FreeRDPGetBytesPerPixel(bitmap->format);
185 bitmap->length = stride * DstHeight;
186 bitmap->data = (BYTE*)winpr_aligned_malloc(bitmap->length, 16);
187
188 if (!bitmap->data)
189 return FALSE;
190
191 if (compressed)
192 {
193 WINPR_ASSERT(context->codecs);
194 switch (codecId)
195 {
196 case RDP_CODEC_ID_REMOTEFX:
197 case RDP_CODEC_ID_IMAGE_REMOTEFX:
198 if (!gdi_Bitmap_Rfx(context, bitmap, pSrcData, SrcSize))
199 return FALSE;
200 break;
201 case RDP_CODEC_ID_NSCODEC:
202 {
203 const int status = nsc_process_message(
204 context->codecs->nsc, 32, DstWidth, DstHeight, pSrcData, SrcSize, bitmap->data,
205 bitmap->format, stride, 0, 0, DstWidth, DstHeight, FREERDP_FLIP_VERTICAL);
206
207 if (status < 1)
208 {
209 WLog_ERR(TAG, "nsc_process_message failed");
210 return FALSE;
211 }
212 }
213 break;
214 default:
215 if (bpp < 32)
216 {
217 if (!interleaved_decompress(context->codecs->interleaved, pSrcData, SrcSize,
218 DstWidth, DstHeight, bpp, bitmap->data,
219 bitmap->format, stride, 0, 0, DstWidth, DstHeight,
220 &gdi->palette))
221 {
222 WLog_ERR(TAG, "interleaved_decompress failed");
223 return FALSE;
224 }
225 }
226 else
227 {
228 const BOOL fidelity = freerdp_settings_get_bool(
229 context->settings, FreeRDP_DrawAllowDynamicColorFidelity);
230 freerdp_planar_switch_bgr(context->codecs->planar, fidelity);
231 if (!freerdp_bitmap_decompress_planar(
232 context->codecs->planar, pSrcData, SrcSize, DstWidth, DstHeight,
233 bitmap->data, bitmap->format, stride, 0, 0, DstWidth, DstHeight, TRUE))
234 {
235 WLog_ERR(TAG, "freerdp_bitmap_decompress_planar failed");
236 return FALSE;
237 }
238 }
239 break;
240 }
241 }
242 else
243 {
244 const UINT32 SrcFormat = gdi_get_pixel_format(bpp);
245 const size_t sbpp = FreeRDPGetBytesPerPixel(SrcFormat);
246 const size_t dbpp = FreeRDPGetBytesPerPixel(bitmap->format);
247
248 if ((sbpp == 0) || (dbpp == 0))
249 return FALSE;
250 else
251 {
252 const size_t dstSize = SrcSize * dbpp / sbpp;
253
254 if (dstSize < bitmap->length)
255 {
256 WLog_ERR(TAG, "dstSize %" PRIuz " < bitmap->length %" PRIu32, dstSize,
257 bitmap->length);
258 return FALSE;
259 }
260 }
261
262 if (!freerdp_image_copy_no_overlap(bitmap->data, bitmap->format, stride, 0, 0, DstWidth,
263 DstHeight, pSrcData, SrcFormat, 0, 0, 0, &gdi->palette,
264 FREERDP_FLIP_VERTICAL))
265 {
266 WLog_ERR(TAG, "freerdp_image_copy failed");
267 return FALSE;
268 }
269 }
270
271 return TRUE;
272}
273
274static BOOL gdi_Bitmap_SetSurface(rdpContext* context, rdpBitmap* bitmap, BOOL primary)
275{
276 rdpGdi* gdi = nullptr;
277
278 if (!context)
279 return FALSE;
280
281 gdi = context->gdi;
282
283 if (!gdi)
284 return FALSE;
285
286 if (primary)
287 gdi->drawing = gdi->primary;
288 else
289 gdi->drawing = (gdiBitmap*)bitmap;
290
291 return TRUE;
292}
293
294/* Glyph Class */
295static BOOL gdi_Glyph_New(rdpContext* context, rdpGlyph* glyph)
296{
297 if (!context || !glyph)
298 return FALSE;
299
300 gdiGlyph* gdi_glyph = (gdiGlyph*)glyph;
301 gdi_glyph->hdc = gdi_GetDC();
302
303 if (!gdi_glyph->hdc)
304 return FALSE;
305
306 gdi_glyph->hdc->format = PIXEL_FORMAT_MONO;
307 BYTE* data = freerdp_glyph_convert_ex(glyph->cx, glyph->cy, glyph->aj, glyph->cb);
308
309 if (!data)
310 {
311 gdi_DeleteDC(gdi_glyph->hdc);
312 return FALSE;
313 }
314
315 gdi_glyph->bitmap = gdi_CreateBitmap(glyph->cx, glyph->cy, PIXEL_FORMAT_MONO, data);
316
317 if (!gdi_glyph->bitmap)
318 {
319 gdi_DeleteDC(gdi_glyph->hdc);
320 winpr_aligned_free(data);
321 return FALSE;
322 }
323
324 gdi_SelectObject(gdi_glyph->hdc, (HGDIOBJECT)gdi_glyph->bitmap);
325 gdi_glyph->org_bitmap = nullptr;
326 return TRUE;
327}
328
329static void gdi_Glyph_Free(WINPR_ATTR_UNUSED rdpContext* context, rdpGlyph* glyph)
330{
331 gdiGlyph* gdi_glyph = nullptr;
332 gdi_glyph = (gdiGlyph*)glyph;
333
334 if (gdi_glyph)
335 {
336 gdi_SelectObject(gdi_glyph->hdc, (HGDIOBJECT)gdi_glyph->org_bitmap);
337 gdi_DeleteObject((HGDIOBJECT)gdi_glyph->bitmap);
338 gdi_DeleteDC(gdi_glyph->hdc);
339 free(glyph->aj);
340 free(glyph);
341 }
342}
343
344static BOOL gdi_Glyph_Draw(rdpContext* context, const rdpGlyph* glyph, INT32 x, INT32 y, INT32 w,
345 INT32 h, INT32 sx, INT32 sy, BOOL fOpRedundant)
346{
347 const gdiGlyph* gdi_glyph = nullptr;
348 rdpGdi* gdi = nullptr;
349 HGDI_BRUSH brush = nullptr;
350 BOOL rc = FALSE;
351
352 if (!context || !glyph)
353 return FALSE;
354
355 gdi = context->gdi;
356 gdi_glyph = (const gdiGlyph*)glyph;
357
358 if (!fOpRedundant)
359 {
360 GDI_RECT rect = WINPR_C_ARRAY_INIT;
361
362 if (x > 0)
363 rect.left = x;
364
365 if (y > 0)
366 rect.top = y;
367
368 if (x + w > 0)
369 rect.right = x + w - 1;
370
371 if (y + h > 0)
372 rect.bottom = y + h - 1;
373
374 if ((rect.left < rect.right) && (rect.top < rect.bottom))
375 {
376 brush = gdi_CreateSolidBrush(gdi->drawing->hdc->bkColor);
377
378 if (!brush)
379 return FALSE;
380
381 const BOOL res = gdi_FillRect(gdi->drawing->hdc, &rect, brush);
382 gdi_DeleteObject((HGDIOBJECT)brush);
383 if (!res)
384 return res;
385 }
386 }
387
388 brush = gdi_CreateSolidBrush(gdi->drawing->hdc->textColor);
389
390 if (!brush)
391 return FALSE;
392
393 gdi_SelectObject(gdi->drawing->hdc, (HGDIOBJECT)brush);
394 rc = gdi_BitBlt(gdi->drawing->hdc, x, y, w, h, gdi_glyph->hdc, sx, sy, GDI_GLYPH_ORDER,
395 &context->gdi->palette);
396 gdi_DeleteObject((HGDIOBJECT)brush);
397 return rc;
398}
399
400static BOOL gdi_Glyph_BeginDraw(rdpContext* context, INT32 x, INT32 y, INT32 width, INT32 height,
401 UINT32 bgcolor, UINT32 fgcolor, BOOL fOpRedundant)
402{
403 if (!context || !context->gdi)
404 return FALSE;
405
406 rdpGdi* gdi = context->gdi;
407
408 if (!gdi->drawing || !gdi->drawing->hdc)
409 return FALSE;
410
411 if (!fOpRedundant)
412 {
413 if (!gdi_decode_color(gdi, bgcolor, &bgcolor, nullptr))
414 return FALSE;
415
416 if (!gdi_decode_color(gdi, fgcolor, &fgcolor, nullptr))
417 return FALSE;
418
419 if (!gdi_SetClipRgn(gdi->drawing->hdc, x, y, width, height))
420 return FALSE;
421
422 gdi_SetTextColor(gdi->drawing->hdc, bgcolor);
423 gdi_SetBkColor(gdi->drawing->hdc, fgcolor);
424
425 {
426 GDI_RECT rect = WINPR_C_ARRAY_INIT;
427 HGDI_BRUSH brush = gdi_CreateSolidBrush(fgcolor);
428
429 if (!brush)
430 return FALSE;
431
432 if (x > 0)
433 rect.left = x;
434
435 if (y > 0)
436 rect.top = y;
437
438 rect.right = x + width - 1;
439 rect.bottom = y + height - 1;
440
441 BOOL res = TRUE;
442 if ((x + width > rect.left) && (y + height > rect.top))
443 res = gdi_FillRect(gdi->drawing->hdc, &rect, brush);
444
445 gdi_DeleteObject((HGDIOBJECT)brush);
446 if (!res)
447 return FALSE;
448 }
449
450 return gdi_SetNullClipRgn(gdi->drawing->hdc);
451 }
452
453 return TRUE;
454}
455
456static BOOL gdi_Glyph_EndDraw(rdpContext* context, WINPR_ATTR_UNUSED INT32 x,
457 WINPR_ATTR_UNUSED INT32 y, WINPR_ATTR_UNUSED INT32 width,
458 WINPR_ATTR_UNUSED INT32 height, WINPR_ATTR_UNUSED UINT32 bgcolor,
459 WINPR_ATTR_UNUSED UINT32 fgcolor)
460{
461 rdpGdi* gdi = nullptr;
462
463 if (!context || !context->gdi)
464 return FALSE;
465
466 gdi = context->gdi;
467
468 if (!gdi->drawing || !gdi->drawing->hdc)
469 return FALSE;
470
471 return gdi_SetNullClipRgn(gdi->drawing->hdc);
472}
473
474/* Graphics Module */
475BOOL gdi_register_graphics(rdpGraphics* graphics)
476{
477 rdpBitmap bitmap = WINPR_C_ARRAY_INIT;
478 rdpGlyph glyph = WINPR_C_ARRAY_INIT;
479 bitmap.size = sizeof(gdiBitmap);
480 bitmap.New = gdi_Bitmap_New;
481 bitmap.Free = gdi_Bitmap_Free;
482 bitmap.Paint = gdi_Bitmap_Paint;
483 bitmap.Decompress = gdi_Bitmap_Decompress;
484 bitmap.SetSurface = gdi_Bitmap_SetSurface;
485 graphics_register_bitmap(graphics, &bitmap);
486 glyph.size = sizeof(gdiGlyph);
487 glyph.New = gdi_Glyph_New;
488 glyph.Free = gdi_Glyph_Free;
489 glyph.Draw = gdi_Glyph_Draw;
490 glyph.BeginDraw = gdi_Glyph_BeginDraw;
491 glyph.EndDraw = gdi_Glyph_EndDraw;
492 graphics_register_glyph(graphics, &glyph);
493 return TRUE;
494}
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.