FreeRDP
Loading...
Searching...
No Matches
xf_rail.c
1
20#include <freerdp/config.h>
21
22#include <X11/Xlib.h>
23#include <X11/Xatom.h>
24#include <X11/Xutil.h>
25
26#include <winpr/cast.h>
27#include <winpr/assert.h>
28#include <winpr/wlog.h>
29#include <winpr/print.h>
30
31#include <freerdp/client/rail.h>
32
33#include "xf_window.h"
34#include "xf_rail.h"
35#include "xf_utils.h"
36
37#include <freerdp/log.h>
38#define TAG CLIENT_TAG("x11")
39
40static const char* error_code2str(UINT32 code)
41{
42#define EVCASE(x) \
43 case x: \
44 return #x
45 switch (code)
46 {
47 EVCASE(RAIL_EXEC_S_OK);
48 EVCASE(RAIL_EXEC_E_HOOK_NOT_LOADED);
49 EVCASE(RAIL_EXEC_E_DECODE_FAILED);
50 EVCASE(RAIL_EXEC_E_NOT_IN_ALLOWLIST);
51 EVCASE(RAIL_EXEC_E_FILE_NOT_FOUND);
52 EVCASE(RAIL_EXEC_E_FAIL);
53 EVCASE(RAIL_EXEC_E_SESSION_LOCKED);
54 default:
55 return "RAIL_EXEC_E_UNKNOWN";
56 }
57#undef EVCASE
58}
59
60static const char* movetype2str(UINT32 code)
61{
62#define EVCASE(x) \
63 case x: \
64 return #x
65
66 switch (code)
67 {
68
69 EVCASE(RAIL_WMSZ_LEFT);
70 EVCASE(RAIL_WMSZ_RIGHT);
71 EVCASE(RAIL_WMSZ_TOP);
72 EVCASE(RAIL_WMSZ_TOPLEFT);
73 EVCASE(RAIL_WMSZ_TOPRIGHT);
74 EVCASE(RAIL_WMSZ_BOTTOM);
75 EVCASE(RAIL_WMSZ_BOTTOMLEFT);
76 EVCASE(RAIL_WMSZ_BOTTOMRIGHT);
77 EVCASE(RAIL_WMSZ_MOVE);
78 EVCASE(RAIL_WMSZ_KEYMOVE);
79 EVCASE(RAIL_WMSZ_KEYSIZE);
80 default:
81 return "RAIL_WMSZ_INVALID";
82 }
83#undef EVCASE
84}
85
86struct xf_rail_icon
87{
88 long* data;
89 int length;
90};
91typedef struct xf_rail_icon xfRailIcon;
92
93struct xf_rail_icon_cache
94{
95 xfRailIcon* entries;
96 UINT32 numCaches;
97 UINT32 numCacheEntries;
98 xfRailIcon scratch;
99};
100
101typedef struct
102{
103 xfContext* xfc;
104 const RECTANGLE_16* rect;
105} rail_paint_fn_arg_t;
106
107BOOL xf_rail_enable_remoteapp_mode(xfContext* xfc)
108{
109 WINPR_ASSERT(xfc);
110 if (!xfc->remote_app)
111 {
112 rdpGdi* gdi = xfc->common.context.gdi;
113 WINPR_ASSERT(gdi);
114
115 const BOOL old = gdi->suppressOutput;
116 gdi->suppressOutput = TRUE;
117 xfc->remote_app = TRUE;
118 xfc->drawable = xf_CreateDummyWindow(xfc);
119 xf_DestroyDesktopWindow(xfc, xfc->window);
120 xfc->window = nullptr;
121
122 gdi->suppressOutput = old;
123 }
124 return TRUE;
125}
126
127BOOL xf_rail_disable_remoteapp_mode(xfContext* xfc)
128{
129 WINPR_ASSERT(xfc);
130 if (xfc->remote_app)
131 {
132 rdpGdi* gdi = xfc->common.context.gdi;
133 WINPR_ASSERT(gdi);
134
135 const BOOL old = gdi->suppressOutput;
136 gdi->suppressOutput = TRUE;
137
138 xfc->remote_app = FALSE;
139 xf_DestroyDummyWindow(xfc, xfc->drawable);
140 xf_destroy_window(xfc);
141 xf_create_window(xfc);
142 xf_create_image(xfc);
143
144 gdi->suppressOutput = old;
145 }
146 return TRUE;
147}
148
149BOOL xf_rail_send_activate(xfContext* xfc, Window xwindow, BOOL enabled)
150{
151 RAIL_ACTIVATE_ORDER activate = WINPR_C_ARRAY_INIT;
152 xfAppWindow* appWindow = xf_AppWindowFromX11Window(xfc, xwindow);
153
154 if (!appWindow)
155 return FALSE;
156
157 if (enabled)
158 xf_SetWindowStyle(xfc, appWindow, appWindow->dwStyle, appWindow->dwExStyle);
159
160 WINPR_ASSERT(appWindow->windowId <= UINT32_MAX);
161 activate.windowId = (UINT32)appWindow->windowId;
162 xf_rail_return_window(appWindow);
163
164 activate.enabled = enabled;
165 const UINT rc = xfc->rail->ClientActivate(xfc->rail, &activate);
166 return rc == CHANNEL_RC_OK;
167}
168
169BOOL xf_rail_send_client_system_command(xfContext* xfc, UINT64 windowId, UINT16 command)
170{
171 WINPR_ASSERT(xfc);
172 WINPR_ASSERT(xfc->rail);
173 WINPR_ASSERT(xfc->rail->ClientSystemCommand);
174 if (windowId > UINT32_MAX)
175 return FALSE;
176
177 const RAIL_SYSCOMMAND_ORDER syscommand = { .windowId = (UINT32)windowId, .command = command };
178 const UINT rc = xfc->rail->ClientSystemCommand(xfc->rail, &syscommand);
179 return rc == CHANNEL_RC_OK;
180}
181
188BOOL xf_rail_adjust_position(xfContext* xfc, xfAppWindow* appWindow)
189{
190 RAIL_WINDOW_MOVE_ORDER windowMove = WINPR_C_ARRAY_INIT;
191
192 WINPR_ASSERT(xfc);
193 WINPR_ASSERT(appWindow);
194 if (!appWindow->is_mapped || appWindow->local_move.state != LMS_NOT_ACTIVE)
195 return FALSE;
196
197 /* If current window position disagrees with RDP window position, send update to RDP server */
198 if (appWindow->x != appWindow->windowOffsetX || appWindow->y != appWindow->windowOffsetY ||
199 appWindow->width != (INT64)appWindow->windowWidth ||
200 appWindow->height != (INT64)appWindow->windowHeight)
201 {
202 WINPR_ASSERT(appWindow->windowId <= UINT32_MAX);
203 windowMove.windowId = (UINT32)appWindow->windowId;
204 /*
205 * Calculate new size/position for the rail window(new values for
206 * windowOffsetX/windowOffsetY/windowWidth/windowHeight) on the server
207 */
208 const INT16 left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginLeft);
209 const INT16 right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginRight);
210 const INT16 top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginTop);
211 const INT16 bottom = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginBottom);
212 windowMove.left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x - left);
213 windowMove.top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->y - top);
214 windowMove.right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x + appWindow->width + right);
215 windowMove.bottom =
216 WINPR_ASSERTING_INT_CAST(INT16, appWindow->y + appWindow->height + bottom);
217 const UINT rc = xfc->rail->ClientWindowMove(xfc->rail, &windowMove);
218 return rc == CHANNEL_RC_OK;
219 }
220 return TRUE;
221}
222
223BOOL xf_rail_end_local_move(xfContext* xfc, xfAppWindow* appWindow)
224{
225 int x = 0;
226 int y = 0;
227 int child_x = 0;
228 int child_y = 0;
229 unsigned int mask = 0;
230 Window root_window = 0;
231 Window child_window = 0;
232
233 WINPR_ASSERT(xfc);
234 WINPR_ASSERT(appWindow);
235
236 if ((appWindow->local_move.direction == NET_WM_MOVERESIZE_MOVE_KEYBOARD) ||
237 (appWindow->local_move.direction == NET_WM_MOVERESIZE_SIZE_KEYBOARD))
238 {
239 RAIL_WINDOW_MOVE_ORDER windowMove = WINPR_C_ARRAY_INIT;
240
241 /*
242 * For keyboard moves send and explicit update to RDP server
243 */
244 WINPR_ASSERT(appWindow->windowId <= UINT32_MAX);
245 windowMove.windowId = (UINT32)appWindow->windowId;
246 /*
247 * Calculate new size/position for the rail window(new values for
248 * windowOffsetX/windowOffsetY/windowWidth/windowHeight) on the server
249 *
250 */
251 const INT16 left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginLeft);
252 const INT16 right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginRight);
253 const INT16 top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginTop);
254 const INT16 bottom = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginBottom);
255 const INT16 w = WINPR_ASSERTING_INT_CAST(INT16, appWindow->width + right);
256 const INT16 h = WINPR_ASSERTING_INT_CAST(INT16, appWindow->height + bottom);
257 windowMove.left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x - left);
258 windowMove.top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->y - top);
259 windowMove.right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x + w); /* In the update to
260 RDP the position is one past the window */
261 windowMove.bottom = WINPR_ASSERTING_INT_CAST(INT16, appWindow->y + h);
262 const UINT rc = xfc->rail->ClientWindowMove(xfc->rail, &windowMove);
263 if (rc != CHANNEL_RC_OK)
264 return FALSE;
265 }
266
267 /*
268 * Simulate button up at new position to end the local move (per RDP spec)
269 */
270 XQueryPointer(xfc->display, appWindow->handle, &root_window, &child_window, &x, &y, &child_x,
271 &child_y, &mask);
272
273 /* only send the mouse coordinates if not a keyboard move or size */
274 if ((appWindow->local_move.direction != NET_WM_MOVERESIZE_MOVE_KEYBOARD) &&
275 (appWindow->local_move.direction != NET_WM_MOVERESIZE_SIZE_KEYBOARD))
276 {
277 if (!freerdp_client_send_button_event(&xfc->common, FALSE, PTR_FLAGS_BUTTON1, x, y))
278 return FALSE;
279 }
280
281 /*
282 * Proactively update the RAIL window dimensions. There is a race condition where
283 * we can start to receive GDI orders for the new window dimensions before we
284 * receive the RAIL ORDER for the new window size. This avoids that race condition.
285 */
286 appWindow->windowOffsetX = appWindow->x;
287 appWindow->windowOffsetY = appWindow->y;
288 appWindow->windowWidth = WINPR_ASSERTING_INT_CAST(uint32_t, appWindow->width);
289 appWindow->windowHeight = WINPR_ASSERTING_INT_CAST(uint32_t, appWindow->height);
290 appWindow->local_move.state = LMS_TERMINATING;
291 return TRUE;
292}
293
294BOOL xf_rail_paint_surface(xfContext* xfc, UINT64 windowId, const RECTANGLE_16* rect)
295{
296 xfAppWindow* appWindow = xf_rail_get_window(xfc, windowId);
297
298 WINPR_ASSERT(rect);
299
300 if (!appWindow)
301 return FALSE;
302
303 const RECTANGLE_16 windowRect = {
304 .left = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->x, 0)),
305 .top = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->y, 0)),
306 .right = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->x + appWindow->width, 0)),
307 .bottom = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->y + appWindow->height, 0))
308 };
309
310 REGION16 windowInvalidRegion = WINPR_C_ARRAY_INIT;
311 region16_init(&windowInvalidRegion);
312 if (!region16_union_rect(&windowInvalidRegion, &windowInvalidRegion, &windowRect))
313 return FALSE;
314 if (!region16_intersect_rect(&windowInvalidRegion, &windowInvalidRegion, rect))
315 return FALSE;
316
317 if (!region16_is_empty(&windowInvalidRegion))
318 {
319 const RECTANGLE_16* extents = region16_extents(&windowInvalidRegion);
320
321 const RECTANGLE_16 updateRect = {
322 .left = WINPR_ASSERTING_INT_CAST(UINT16, extents->left - appWindow->x),
323 .top = WINPR_ASSERTING_INT_CAST(UINT16, extents->top - appWindow->y),
324 .right = WINPR_ASSERTING_INT_CAST(UINT16, extents->right - appWindow->x),
325 .bottom = WINPR_ASSERTING_INT_CAST(UINT16, extents->bottom - appWindow->y)
326 };
327
328 xf_UpdateWindowArea(xfc, appWindow, updateRect.left, updateRect.top,
329 updateRect.right - updateRect.left, updateRect.bottom - updateRect.top);
330 }
331 region16_uninit(&windowInvalidRegion);
332 xf_rail_return_window(appWindow);
333 return TRUE;
334}
335
336static BOOL rail_paint_fn(const void* pvkey, WINPR_ATTR_UNUSED void* value, void* pvarg)
337{
338 rail_paint_fn_arg_t* arg = pvarg;
339 WINPR_ASSERT(pvkey);
340 WINPR_ASSERT(arg);
341
342 const UINT64 key = *(const UINT64*)pvkey;
343 return xf_rail_paint_surface(arg->xfc, key, arg->rect);
344}
345
346BOOL xf_rail_paint(xfContext* xfc, const RECTANGLE_16* rect)
347{
348 rail_paint_fn_arg_t arg = { .xfc = xfc, .rect = rect };
349
350 WINPR_ASSERT(xfc);
351 WINPR_ASSERT(rect);
352
353 if (!xfc->railWindows)
354 return TRUE;
355
356 return HashTable_Foreach(xfc->railWindows, rail_paint_fn, &arg);
357}
358
359#define window_state_log_style(log, windowState) \
360 window_state_log_style_int((log), (windowState), __FILE__, __func__, __LINE__)
361static void window_state_log_style_int(wLog* log, const WINDOW_STATE_ORDER* windowState,
362 const char* file, const char* fkt, size_t line)
363{
364 const DWORD log_level = WLOG_DEBUG;
365
366 WINPR_ASSERT(log);
367 WINPR_ASSERT(windowState);
368 if (WLog_IsLevelActive(log, log_level))
369 {
370 char buffer1[128] = WINPR_C_ARRAY_INIT;
371 char buffer2[128] = WINPR_C_ARRAY_INIT;
372
373 window_styles_to_string(windowState->style, buffer1, sizeof(buffer1));
374 window_styles_ex_to_string(windowState->extendedStyle, buffer2, sizeof(buffer2));
375 WLog_PrintTextMessage(log, log_level, line, file, fkt, "windowStyle={%s, %s}", buffer1,
376 buffer2);
377 }
378}
379
380/* RemoteApp Core Protocol Extension */
381
382static BOOL xf_rail_window_common(rdpContext* context, const WINDOW_ORDER_INFO* orderInfo,
383 const WINDOW_STATE_ORDER* windowState)
384{
385 BOOL rc = FALSE;
386 xfContext* xfc = (xfContext*)context;
387
388 WINPR_ASSERT(xfc);
389 WINPR_ASSERT(orderInfo);
390 WINPR_ASSERT(windowState);
391
392 UINT32 fieldFlags = orderInfo->fieldFlags;
393 BOOL position_or_size_updated = FALSE;
394 xfAppWindow* appWindow = xf_rail_get_window(xfc, orderInfo->windowId);
395
396 if (fieldFlags & WINDOW_ORDER_STATE_NEW)
397 {
398 if (!appWindow)
399 appWindow = xf_rail_add_window(xfc, orderInfo->windowId, windowState->windowOffsetX,
400 windowState->windowOffsetY, windowState->windowWidth,
401 windowState->windowHeight, 0xFFFFFFFF);
402
403 if (!appWindow)
404 return FALSE;
405
406 appWindow->dwStyle = windowState->style;
407 appWindow->dwExStyle = windowState->extendedStyle;
408 window_state_log_style(xfc->log, windowState);
409
410 /* Ensure window always gets a window title */
411 if (fieldFlags & WINDOW_ORDER_FIELD_TITLE)
412 {
413 union
414 {
415 WCHAR* wc;
416 BYTE* b;
417 } cnv;
418 char* title = nullptr;
419
420 cnv.b = windowState->titleInfo.string;
421 if (windowState->titleInfo.length == 0)
422 {
423 if (!(title = _strdup("")))
424 {
425 WLog_ERR(TAG, "failed to duplicate empty window title string");
426 /* error handled below */
427 }
428 }
429 else if (!(title = ConvertWCharNToUtf8Alloc(
430 cnv.wc, windowState->titleInfo.length / sizeof(WCHAR), nullptr)))
431 {
432 WLog_ERR(TAG, "failed to convert window title");
433 /* error handled below */
434 }
435
436 appWindow->title = title;
437 }
438 else
439 {
440 if (!(appWindow->title = _strdup("RdpRailWindow")))
441 WLog_ERR(TAG, "failed to duplicate default window title string");
442 }
443
444 if (!appWindow->title)
445 goto fail;
446
447 xf_AppWindowInit(xfc, appWindow);
448 }
449
450 if (!appWindow)
451 return FALSE;
452
453 /* Keep track of any position/size update so that we can force a refresh of the window */
454 if ((fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET) ||
455 (fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE) ||
456 (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET) ||
457 (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE) ||
458 (fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA) ||
459 (fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET) ||
460 (fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY))
461 {
462 position_or_size_updated = TRUE;
463 }
464
465 /* Update Parameters */
466
467 if (fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET)
468 {
469 appWindow->windowOffsetX = windowState->windowOffsetX;
470 appWindow->windowOffsetY = windowState->windowOffsetY;
471 }
472
473 if (fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE)
474 {
475 appWindow->windowWidth = windowState->windowWidth;
476 appWindow->windowHeight = windowState->windowHeight;
477 }
478
479 if (fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_X)
480 {
481 appWindow->resizeMarginLeft = windowState->resizeMarginLeft;
482 appWindow->resizeMarginRight = windowState->resizeMarginRight;
483 }
484
485 if (fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_Y)
486 {
487 appWindow->resizeMarginTop = windowState->resizeMarginTop;
488 appWindow->resizeMarginBottom = windowState->resizeMarginBottom;
489 }
490
491 if (fieldFlags & WINDOW_ORDER_FIELD_OWNER)
492 {
493 appWindow->ownerWindowId = windowState->ownerWindowId;
494 }
495
496 if (fieldFlags & WINDOW_ORDER_FIELD_STYLE)
497 {
498 appWindow->dwStyle = windowState->style;
499 appWindow->dwExStyle = windowState->extendedStyle;
500 window_state_log_style(xfc->log, windowState);
501 }
502
503 if (fieldFlags & WINDOW_ORDER_FIELD_SHOW)
504 {
505 appWindow->showState = windowState->showState;
506 }
507
508 if (fieldFlags & WINDOW_ORDER_FIELD_TITLE)
509 {
510 char* title = nullptr;
511 union
512 {
513 WCHAR* wc;
514 BYTE* b;
515 } cnv;
516
517 cnv.b = windowState->titleInfo.string;
518 if (windowState->titleInfo.length == 0)
519 {
520 if (!(title = _strdup("")))
521 {
522 WLog_ERR(TAG, "failed to duplicate empty window title string");
523 goto fail;
524 }
525 }
526 else if (!(title = ConvertWCharNToUtf8Alloc(
527 cnv.wc, windowState->titleInfo.length / sizeof(WCHAR), nullptr)))
528 {
529 WLog_ERR(TAG, "failed to convert window title");
530 goto fail;
531 }
532
533 free(appWindow->title);
534 appWindow->title = title;
535 }
536
537 if (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET)
538 {
539 appWindow->clientOffsetX = windowState->clientOffsetX;
540 appWindow->clientOffsetY = windowState->clientOffsetY;
541 }
542
543 if (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE)
544 {
545 appWindow->clientAreaWidth = windowState->clientAreaWidth;
546 appWindow->clientAreaHeight = windowState->clientAreaHeight;
547 }
548
549 if (fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA)
550 {
551 appWindow->windowClientDeltaX = windowState->windowClientDeltaX;
552 appWindow->windowClientDeltaY = windowState->windowClientDeltaY;
553 }
554
555 if (fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS)
556 {
557 if (appWindow->windowRects)
558 {
559 free(appWindow->windowRects);
560 appWindow->windowRects = nullptr;
561 }
562
563 appWindow->numWindowRects = windowState->numWindowRects;
564
565 if (appWindow->numWindowRects)
566 {
567 appWindow->windowRects =
568 (RECTANGLE_16*)calloc(appWindow->numWindowRects, sizeof(RECTANGLE_16));
569
570 if (!appWindow->windowRects)
571 goto fail;
572
573 CopyMemory(appWindow->windowRects, windowState->windowRects,
574 appWindow->numWindowRects * sizeof(RECTANGLE_16));
575 }
576 }
577
578 if (fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET)
579 {
580 appWindow->visibleOffsetX = windowState->visibleOffsetX;
581 appWindow->visibleOffsetY = windowState->visibleOffsetY;
582 }
583
584 if (fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY)
585 {
586 if (appWindow->visibilityRects)
587 {
588 free(appWindow->visibilityRects);
589 appWindow->visibilityRects = nullptr;
590 }
591
592 appWindow->numVisibilityRects = windowState->numVisibilityRects;
593
594 if (appWindow->numVisibilityRects)
595 {
596 appWindow->visibilityRects =
597 (RECTANGLE_16*)calloc(appWindow->numVisibilityRects, sizeof(RECTANGLE_16));
598
599 if (!appWindow->visibilityRects)
600 goto fail;
601
602 CopyMemory(appWindow->visibilityRects, windowState->visibilityRects,
603 appWindow->numVisibilityRects * sizeof(RECTANGLE_16));
604 }
605 }
606
607 /* Update Window */
608
609 if (fieldFlags & WINDOW_ORDER_FIELD_STYLE)
610 {
611 }
612
613 if (fieldFlags & WINDOW_ORDER_FIELD_SHOW)
614 {
615 xf_ShowWindow(xfc, appWindow, WINPR_ASSERTING_INT_CAST(UINT8, appWindow->showState));
616 }
617
618 if (fieldFlags & WINDOW_ORDER_FIELD_TITLE)
619 {
620 if (appWindow->title)
621 xf_SetWindowText(xfc, appWindow, appWindow->title);
622 }
623
624 if (position_or_size_updated)
625 {
626 const INT32 visibilityRectsOffsetX =
627 (appWindow->visibleOffsetX -
628 (appWindow->clientOffsetX - appWindow->windowClientDeltaX));
629 const INT32 visibilityRectsOffsetY =
630 (appWindow->visibleOffsetY -
631 (appWindow->clientOffsetY - appWindow->windowClientDeltaY));
632
633 /*
634 * The rail server like to set the window to a small size when it is minimized even though
635 * it is hidden in some cases this can cause the window not to restore back to its original
636 * size. Therefore we don't update our local window when that rail window state is minimized
637 */
638 if (appWindow->rail_state != WINDOW_SHOW_MINIMIZED)
639 {
640 /* Redraw window area if already in the correct position */
641 if (appWindow->x == (INT64)appWindow->windowOffsetX &&
642 appWindow->y == (INT64)appWindow->windowOffsetY &&
643 appWindow->width == (INT64)appWindow->windowWidth &&
644 appWindow->height == (INT64)appWindow->windowHeight)
645 {
646 xf_UpdateWindowArea(xfc, appWindow, 0, 0,
647 WINPR_ASSERTING_INT_CAST(int, appWindow->windowWidth),
648 WINPR_ASSERTING_INT_CAST(int, appWindow->windowHeight));
649 }
650 else
651 {
652 xf_MoveWindow(xfc, appWindow, appWindow->windowOffsetX, appWindow->windowOffsetY,
653 WINPR_ASSERTING_INT_CAST(int, appWindow->windowWidth),
654 WINPR_ASSERTING_INT_CAST(int, appWindow->windowHeight));
655 }
656
657 xf_SetWindowVisibilityRects(
658 xfc, appWindow, WINPR_ASSERTING_INT_CAST(uint32_t, visibilityRectsOffsetX),
659 WINPR_ASSERTING_INT_CAST(uint32_t, visibilityRectsOffsetY),
660 appWindow->visibilityRects,
661 WINPR_ASSERTING_INT_CAST(int, appWindow->numVisibilityRects));
662 }
663
664 if (appWindow->rail_state == WINDOW_SHOW_MAXIMIZED)
665 {
666 xf_SendClientEvent(xfc, appWindow->handle, xfc->NET_WM_STATE, 4, NET_WM_STATE_ADD,
667 xfc->NET_WM_STATE_MAXIMIZED_VERT, xfc->NET_WM_STATE_MAXIMIZED_HORZ,
668 0, 0);
669 }
670 }
671
672 if (fieldFlags & (WINDOW_ORDER_STATE_NEW | WINDOW_ORDER_FIELD_STYLE))
673 xf_SetWindowStyle(xfc, appWindow, appWindow->dwStyle, appWindow->dwExStyle);
674
675 /* We should only be using the visibility rects for shaping the window */
676 /*if (fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS)
677 {
678 xf_SetWindowRects(xfc, appWindow, appWindow->windowRects, appWindow->numWindowRects);
679 }*/
680 rc = TRUE;
681fail:
682 xf_rail_return_window(appWindow);
683 return rc;
684}
685
686static BOOL xf_rail_window_delete(rdpContext* context, const WINDOW_ORDER_INFO* orderInfo)
687{
688 xfContext* xfc = (xfContext*)context;
689 WINPR_ASSERT(xfc);
690 return xf_rail_del_window(xfc, orderInfo->windowId);
691}
692
693static xfRailIconCache* RailIconCache_New(rdpSettings* settings)
694{
695 xfRailIconCache* cache = calloc(1, sizeof(xfRailIconCache));
696
697 if (!cache)
698 return nullptr;
699
700 cache->numCaches = freerdp_settings_get_uint32(settings, FreeRDP_RemoteAppNumIconCaches);
701 cache->numCacheEntries =
702 freerdp_settings_get_uint32(settings, FreeRDP_RemoteAppNumIconCacheEntries);
703 cache->entries = calloc(1ull * cache->numCaches * cache->numCacheEntries, sizeof(xfRailIcon));
704
705 if (!cache->entries)
706 {
707 WLog_ERR(TAG, "failed to allocate icon cache %" PRIu32 " x %" PRIu32 " entries",
708 cache->numCaches, cache->numCacheEntries);
709 free(cache);
710 return nullptr;
711 }
712
713 return cache;
714}
715
716static void RailIconCache_Free(xfRailIconCache* cache)
717{
718 if (!cache)
719 return;
720
721 for (UINT32 i = 0; i < cache->numCaches * cache->numCacheEntries; i++)
722 {
723 xfRailIcon* cur = &cache->entries[i];
724 free(cur->data);
725 }
726
727 free(cache->scratch.data);
728 free(cache->entries);
729 free(cache);
730}
731
732static xfRailIcon* RailIconCache_Lookup(xfRailIconCache* cache, UINT8 cacheId, UINT16 cacheEntry)
733{
734 WINPR_ASSERT(cache);
735 /*
736 * MS-RDPERP 2.2.1.2.3 Icon Info (TS_ICON_INFO)
737 *
738 * CacheId (1 byte):
739 * If the value is 0xFFFF, the icon SHOULD NOT be cached.
740 *
741 * Yes, the spec says "0xFFFF" in the 2018-03-16 revision,
742 * but the actual protocol field is 1-byte wide.
743 */
744 if (cacheId == 0xFF)
745 return &cache->scratch;
746
747 if (cacheId >= cache->numCaches)
748 return nullptr;
749
750 if (cacheEntry >= cache->numCacheEntries)
751 return nullptr;
752
753 return &cache->entries[cache->numCacheEntries * cacheId + cacheEntry];
754}
755
756/*
757 * _NET_WM_ICON format is defined as "array of CARDINAL" values which for
758 * Xlib must be represented with an array of C's "long" values. Note that
759 * "long" != "INT32" on 64-bit systems. Therefore we can't simply cast
760 * the bitmap data as (unsigned char*), we have to copy all the pixels.
761 *
762 * The first two values are width and height followed by actual color data
763 * in ARGB format (e.g., 0xFFFF0000L is opaque red), pixels are in normal,
764 * left-to-right top-down order.
765 */
766static BOOL convert_rail_icon(const ICON_INFO* iconInfo, xfRailIcon* railIcon)
767{
768 WINPR_ASSERT(iconInfo);
769 WINPR_ASSERT(railIcon);
770
771 BYTE* nextPixel = nullptr;
772 long* pixels = nullptr;
773 BYTE* argbPixels = calloc(1ull * iconInfo->width * iconInfo->height, 4);
774
775 if (!argbPixels)
776 goto error;
777
778 if (!freerdp_image_copy_from_icon_data(
779 argbPixels, PIXEL_FORMAT_ARGB32, 0, 0, 0,
780 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->width),
781 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->height), iconInfo->bitsColor,
782 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->cbBitsColor), iconInfo->bitsMask,
783 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->cbBitsMask), iconInfo->colorTable,
784 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->cbColorTable), iconInfo->bpp))
785 goto error;
786
787 {
788 const UINT32 nelements = 2 + iconInfo->width * iconInfo->height;
789 pixels = realloc(railIcon->data, nelements * sizeof(long));
790
791 if (!pixels)
792 goto error;
793
794 railIcon->data = pixels;
795
796 railIcon->length = WINPR_ASSERTING_INT_CAST(int, nelements);
797 pixels[0] = iconInfo->width;
798 pixels[1] = iconInfo->height;
799 nextPixel = argbPixels;
800
801 for (UINT32 i = 2; i < nelements; i++)
802 {
803 pixels[i] = FreeRDPReadColor(nextPixel, PIXEL_FORMAT_BGRA32);
804 nextPixel += 4;
805 }
806 }
807
808 free(argbPixels);
809 return TRUE;
810error:
811 free(argbPixels);
812 return FALSE;
813}
814
815static void xf_rail_set_window_icon(xfContext* xfc, xfAppWindow* railWindow, xfRailIcon* icon,
816 BOOL replace)
817{
818 WINPR_ASSERT(xfc);
819
820 LogDynAndXChangeProperty(xfc->log, xfc->display, railWindow->handle, xfc->NET_WM_ICON,
821 XA_CARDINAL, 32, replace ? PropModeReplace : PropModeAppend,
822 (unsigned char*)icon->data, icon->length);
823 LogDynAndXFlush(xfc->log, xfc->display);
824}
825
826static BOOL xf_rail_window_icon(rdpContext* context, const WINDOW_ORDER_INFO* orderInfo,
827 const WINDOW_ICON_ORDER* windowIcon)
828{
829 BOOL rc = FALSE;
830 xfContext* xfc = (xfContext*)context;
831 BOOL replaceIcon = 0;
832 xfAppWindow* railWindow = xf_rail_get_window(xfc, orderInfo->windowId);
833
834 if (!railWindow)
835 return TRUE;
836
837 WINPR_ASSERT(windowIcon);
838 WINPR_ASSERT(windowIcon->iconInfo);
839 xfRailIcon* icon = RailIconCache_Lookup(
840 xfc->railIconCache, WINPR_ASSERTING_INT_CAST(UINT8, windowIcon->iconInfo->cacheId),
841 WINPR_ASSERTING_INT_CAST(UINT16, windowIcon->iconInfo->cacheEntry));
842
843 if (!icon)
844 {
845 WLog_Print(xfc->log, WLOG_WARN, "failed to get icon from cache %02X:%04X",
846 windowIcon->iconInfo->cacheId, windowIcon->iconInfo->cacheEntry);
847 }
848 else if (!convert_rail_icon(windowIcon->iconInfo, icon))
849 {
850 WLog_Print(xfc->log, WLOG_WARN, "failed to convert icon for window %08X",
851 orderInfo->windowId);
852 }
853 else
854 {
855 replaceIcon = !!(orderInfo->fieldFlags & WINDOW_ORDER_STATE_NEW);
856 xf_rail_set_window_icon(xfc, railWindow, icon, replaceIcon);
857 rc = TRUE;
858 }
859 xf_rail_return_window(railWindow);
860 return rc;
861}
862
863static BOOL xf_rail_window_cached_icon(rdpContext* context, const WINDOW_ORDER_INFO* orderInfo,
864 const WINDOW_CACHED_ICON_ORDER* windowCachedIcon)
865{
866 BOOL rc = FALSE;
867 xfContext* xfc = (xfContext*)context;
868 WINPR_ASSERT(orderInfo);
869
870 BOOL replaceIcon = 0;
871 xfAppWindow* railWindow = xf_rail_get_window(xfc, orderInfo->windowId);
872
873 if (!railWindow)
874 return TRUE;
875
876 WINPR_ASSERT(windowCachedIcon);
877
878 xfRailIcon* icon = RailIconCache_Lookup(
879 xfc->railIconCache, WINPR_ASSERTING_INT_CAST(UINT8, windowCachedIcon->cachedIcon.cacheId),
880 WINPR_ASSERTING_INT_CAST(UINT16, windowCachedIcon->cachedIcon.cacheEntry));
881
882 if (!icon)
883 {
884 WLog_Print(xfc->log, WLOG_WARN, "failed to get icon from cache %02X:%04X",
885 windowCachedIcon->cachedIcon.cacheId, windowCachedIcon->cachedIcon.cacheEntry);
886 }
887 else
888 {
889 replaceIcon = !!(orderInfo->fieldFlags & WINDOW_ORDER_STATE_NEW);
890 xf_rail_set_window_icon(xfc, railWindow, icon, replaceIcon);
891 rc = TRUE;
892 }
893 xf_rail_return_window(railWindow);
894 return rc;
895}
896
897static BOOL
898xf_rail_notify_icon_common(WINPR_ATTR_UNUSED rdpContext* context,
899 const WINDOW_ORDER_INFO* orderInfo,
900 WINPR_ATTR_UNUSED const NOTIFY_ICON_STATE_ORDER* notifyIconState)
901{
902 WLog_ERR("TODO", "TODO: implement");
903 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_VERSION)
904 {
905 }
906
907 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_TIP)
908 {
909 }
910
911 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_INFO_TIP)
912 {
913 }
914
915 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_STATE)
916 {
917 }
918
919 if (orderInfo->fieldFlags & WINDOW_ORDER_ICON)
920 {
921 }
922
923 if (orderInfo->fieldFlags & WINDOW_ORDER_CACHED_ICON)
924 {
925 }
926
927 return TRUE;
928}
929
930static BOOL xf_rail_notify_icon_create(rdpContext* context, const WINDOW_ORDER_INFO* orderInfo,
931 const NOTIFY_ICON_STATE_ORDER* notifyIconState)
932{
933 return xf_rail_notify_icon_common(context, orderInfo, notifyIconState);
934}
935
936static BOOL xf_rail_notify_icon_update(rdpContext* context, const WINDOW_ORDER_INFO* orderInfo,
937 const NOTIFY_ICON_STATE_ORDER* notifyIconState)
938{
939 return xf_rail_notify_icon_common(context, orderInfo, notifyIconState);
940}
941
942static BOOL xf_rail_notify_icon_delete(WINPR_ATTR_UNUSED rdpContext* context,
943 WINPR_ATTR_UNUSED const WINDOW_ORDER_INFO* orderInfo)
944{
945 WLog_ERR("TODO", "TODO: implement");
946 return TRUE;
947}
948
949static BOOL
950xf_rail_monitored_desktop(WINPR_ATTR_UNUSED rdpContext* context,
951 WINPR_ATTR_UNUSED const WINDOW_ORDER_INFO* orderInfo,
952 WINPR_ATTR_UNUSED const MONITORED_DESKTOP_ORDER* monitoredDesktop)
953{
954 const UINT32 mask = WINDOW_ORDER_TYPE_DESKTOP | WINDOW_ORDER_FIELD_DESKTOP_HOOKED |
955 WINDOW_ORDER_FIELD_DESKTOP_ARC_BEGAN |
956 WINDOW_ORDER_FIELD_DESKTOP_ARC_COMPLETED |
957 WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND | WINDOW_ORDER_FIELD_DESKTOP_ZORDER;
958 xfContext* xfc = (xfContext*)context;
959
960 if (!context || !orderInfo || !monitoredDesktop)
961 return FALSE;
962
963 if ((orderInfo->fieldFlags & WINDOW_ORDER_TYPE_DESKTOP) == 0)
964 {
965 WLog_Print(xfc->log, WLOG_WARN, "WINDOW_ORDER_TYPE_DESKTOP flag missing!");
966 return FALSE;
967 }
968
969 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ARC_BEGAN) &&
970 (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_HOOKED))
971 {
972 // discard all windows/notify icons
973 WLog_Print(xfc->log, WLOG_WARN,
974 "TODO: implement WINDOW_ORDER_FIELD_DESKTOP_ARC_BEGAN && "
975 "WINDOW_ORDER_FIELD_DESKTOP_HOOKED");
976 }
977 else if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_HOOKED)
978 {
979 WLog_Print(xfc->log, WLOG_WARN, "TODO: implement WINDOW_ORDER_FIELD_DESKTOP_HOOKED");
980 }
981 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ARC_COMPLETED)
982 {
983 WLog_DBG(TAG, "WINDOW_ORDER_FIELD_DESKTOP_ARC_COMPLETED -> switch to RAILS mode");
984 xf_rail_enable_remoteapp_mode(xfc);
985 }
986 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND)
987 {
988 WLog_Print(xfc->log, WLOG_WARN, "TODO: implement WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND");
989 }
990 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ZORDER)
991 {
992 WLog_Print(xfc->log, WLOG_WARN, "TODO: implement WINDOW_ORDER_FIELD_DESKTOP_ZORDER");
993 }
994 if (orderInfo->fieldFlags & ~mask)
995 {
996 WLog_Print(xfc->log, WLOG_WARN, "unknown flags 0x%08" PRIx32 "!", orderInfo->fieldFlags);
997 }
998 return TRUE;
999}
1000
1001static BOOL xf_rail_non_monitored_desktop(rdpContext* context,
1002 WINPR_ATTR_UNUSED const WINDOW_ORDER_INFO* orderInfo)
1003{
1004 xfContext* xfc = (xfContext*)context;
1005 const UINT32 mask = WINDOW_ORDER_TYPE_DESKTOP | WINDOW_ORDER_FIELD_DESKTOP_NONE;
1006
1007 if (!context || !orderInfo)
1008 return FALSE;
1009
1010 if ((orderInfo->fieldFlags & WINDOW_ORDER_TYPE_DESKTOP) == 0)
1011 {
1012 WLog_Print(xfc->log, WLOG_WARN, "TODO: implement WINDOW_ORDER_TYPE_DESKTOP");
1013 return FALSE;
1014 }
1015 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_NONE)
1016 {
1017 WLog_Print(xfc->log, WLOG_WARN, "TODO: implement WINDOW_ORDER_FIELD_DESKTOP_NONE");
1018 }
1019 if (orderInfo->fieldFlags & ~mask)
1020 {
1021 WLog_Print(xfc->log, WLOG_WARN, "unknown flags 0x%08" PRIx32 "!", orderInfo->fieldFlags);
1022 }
1023
1024 xf_rail_disable_remoteapp_mode(xfc);
1025 return TRUE;
1026}
1027
1028static void xf_rail_register_update_callbacks(rdpUpdate* update)
1029{
1030 WINPR_ASSERT(update);
1031
1032 rdpWindowUpdate* window = update->window;
1033 WINPR_ASSERT(window);
1034
1035 window->WindowCreate = xf_rail_window_common;
1036 window->WindowUpdate = xf_rail_window_common;
1037 window->WindowDelete = xf_rail_window_delete;
1038 window->WindowIcon = xf_rail_window_icon;
1039 window->WindowCachedIcon = xf_rail_window_cached_icon;
1040 window->NotifyIconCreate = xf_rail_notify_icon_create;
1041 window->NotifyIconUpdate = xf_rail_notify_icon_update;
1042 window->NotifyIconDelete = xf_rail_notify_icon_delete;
1043 window->MonitoredDesktop = xf_rail_monitored_desktop;
1044 window->NonMonitoredDesktop = xf_rail_non_monitored_desktop;
1045}
1046
1047/* RemoteApp Virtual Channel Extension */
1048
1054static UINT xf_rail_server_execute_result(RailClientContext* context,
1055 const RAIL_EXEC_RESULT_ORDER* execResult)
1056{
1057 WINPR_ASSERT(context);
1058 WINPR_ASSERT(execResult);
1059
1060 xfContext* xfc = (xfContext*)context->custom;
1061 WINPR_ASSERT(xfc);
1062
1063 if (execResult->execResult != RAIL_EXEC_S_OK)
1064 {
1065 WLog_ERR(TAG, "RAIL exec error: execResult=%s [0x%08" PRIx32 "] NtError=0x%X\n",
1066 error_code2str(execResult->execResult), execResult->execResult,
1067 execResult->rawResult);
1068 freerdp_abort_connect_context(&xfc->common.context);
1069 }
1070
1071 return CHANNEL_RC_OK;
1072}
1073
1079static UINT xf_rail_server_system_param(WINPR_ATTR_UNUSED RailClientContext* context,
1080 WINPR_ATTR_UNUSED const RAIL_SYSPARAM_ORDER* sysparam)
1081{
1082 // TODO: Actually apply param
1083 WLog_ERR("TODO", "TODO: implement");
1084 return CHANNEL_RC_OK;
1085}
1086
1092static UINT xf_rail_server_handshake(RailClientContext* context,
1093 WINPR_ATTR_UNUSED const RAIL_HANDSHAKE_ORDER* handshake)
1094{
1095 return client_rail_server_start_cmd(context);
1096}
1097
1103static UINT
1104xf_rail_server_handshake_ex(RailClientContext* context,
1105 WINPR_ATTR_UNUSED const RAIL_HANDSHAKE_EX_ORDER* handshakeEx)
1106{
1107 return client_rail_server_start_cmd(context);
1108}
1109
1115static UINT xf_rail_server_local_move_size(RailClientContext* context,
1116 const RAIL_LOCALMOVESIZE_ORDER* localMoveSize)
1117{
1118 int x = 0;
1119 int y = 0;
1120 int direction = 0;
1121 Window child_window = 0;
1122 WINPR_ASSERT(context);
1123 WINPR_ASSERT(localMoveSize);
1124
1125 xfContext* xfc = (xfContext*)context->custom;
1126 xfAppWindow* appWindow = xf_rail_get_window(xfc, localMoveSize->windowId);
1127
1128 if (!appWindow)
1129 return ERROR_INTERNAL_ERROR;
1130
1131 WLog_Print(xfc->log, WLOG_TRACE, "%s [0x%08" PRIx32 "]",
1132 movetype2str(localMoveSize->moveSizeType), localMoveSize->moveSizeType);
1133 switch (localMoveSize->moveSizeType)
1134 {
1135 case RAIL_WMSZ_LEFT:
1136 direction = NET_WM_MOVERESIZE_SIZE_LEFT;
1137 x = localMoveSize->posX;
1138 y = localMoveSize->posY;
1139 break;
1140
1141 case RAIL_WMSZ_RIGHT:
1142 direction = NET_WM_MOVERESIZE_SIZE_RIGHT;
1143 x = localMoveSize->posX;
1144 y = localMoveSize->posY;
1145 break;
1146
1147 case RAIL_WMSZ_TOP:
1148 direction = NET_WM_MOVERESIZE_SIZE_TOP;
1149 x = localMoveSize->posX;
1150 y = localMoveSize->posY;
1151 break;
1152
1153 case RAIL_WMSZ_TOPLEFT:
1154 direction = NET_WM_MOVERESIZE_SIZE_TOPLEFT;
1155 x = localMoveSize->posX;
1156 y = localMoveSize->posY;
1157 break;
1158
1159 case RAIL_WMSZ_TOPRIGHT:
1160 direction = NET_WM_MOVERESIZE_SIZE_TOPRIGHT;
1161 x = localMoveSize->posX;
1162 y = localMoveSize->posY;
1163 break;
1164
1165 case RAIL_WMSZ_BOTTOM:
1166 direction = NET_WM_MOVERESIZE_SIZE_BOTTOM;
1167 x = localMoveSize->posX;
1168 y = localMoveSize->posY;
1169 break;
1170
1171 case RAIL_WMSZ_BOTTOMLEFT:
1172 direction = NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT;
1173 x = localMoveSize->posX;
1174 y = localMoveSize->posY;
1175 break;
1176
1177 case RAIL_WMSZ_BOTTOMRIGHT:
1178 direction = NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT;
1179 x = localMoveSize->posX;
1180 y = localMoveSize->posY;
1181 break;
1182
1183 case RAIL_WMSZ_MOVE:
1184 direction = NET_WM_MOVERESIZE_MOVE;
1185 XTranslateCoordinates(xfc->display, appWindow->handle, RootWindowOfScreen(xfc->screen),
1186 localMoveSize->posX, localMoveSize->posY, &x, &y, &child_window);
1187 break;
1188
1189 case RAIL_WMSZ_KEYMOVE:
1190 direction = NET_WM_MOVERESIZE_MOVE_KEYBOARD;
1191 x = localMoveSize->posX;
1192 y = localMoveSize->posY;
1193 /* FIXME: local keyboard moves not working */
1194 break;
1195
1196 case RAIL_WMSZ_KEYSIZE:
1197 direction = NET_WM_MOVERESIZE_SIZE_KEYBOARD;
1198 x = localMoveSize->posX;
1199 y = localMoveSize->posY;
1200 /* FIXME: local keyboard moves not working */
1201 break;
1202 default:
1203 break;
1204 }
1205
1206 if (localMoveSize->isMoveSizeStart)
1207 xf_StartLocalMoveSize(xfc, appWindow, direction, x, y);
1208 else
1209 xf_EndLocalMoveSize(xfc, appWindow);
1210
1211 xf_rail_return_window(appWindow);
1212 return CHANNEL_RC_OK;
1213}
1214
1220static UINT xf_rail_server_min_max_info(RailClientContext* context,
1221 const RAIL_MINMAXINFO_ORDER* minMaxInfo)
1222{
1223 WINPR_ASSERT(context);
1224 WINPR_ASSERT(minMaxInfo);
1225
1226 xfContext* xfc = (xfContext*)context->custom;
1227 xfAppWindow* appWindow = xf_rail_get_window(xfc, minMaxInfo->windowId);
1228
1229 if (appWindow)
1230 {
1231 xf_SetWindowMinMaxInfo(xfc, appWindow, minMaxInfo->maxWidth, minMaxInfo->maxHeight,
1232 minMaxInfo->maxPosX, minMaxInfo->maxPosY, minMaxInfo->minTrackWidth,
1233 minMaxInfo->minTrackHeight, minMaxInfo->maxTrackWidth,
1234 minMaxInfo->maxTrackHeight);
1235 }
1236 xf_rail_return_window(appWindow);
1237
1238 return CHANNEL_RC_OK;
1239}
1240
1246static UINT
1247xf_rail_server_language_bar_info(WINPR_ATTR_UNUSED RailClientContext* context,
1248 WINPR_ATTR_UNUSED const RAIL_LANGBAR_INFO_ORDER* langBarInfo)
1249{
1250 WLog_ERR("TODO", "TODO: implement");
1251 return CHANNEL_RC_OK;
1252}
1253
1259static UINT
1260xf_rail_server_get_appid_response(WINPR_ATTR_UNUSED RailClientContext* context,
1261 WINPR_ATTR_UNUSED const RAIL_GET_APPID_RESP_ORDER* getAppIdResp)
1262{
1263 WLog_ERR("TODO", "TODO: implement");
1264 return CHANNEL_RC_OK;
1265}
1266
1267static BOOL rail_window_key_equals(const void* key1, const void* key2)
1268{
1269 const UINT64* k1 = (const UINT64*)key1;
1270 const UINT64* k2 = (const UINT64*)key2;
1271
1272 if (!k1 || !k2)
1273 return FALSE;
1274
1275 return *k1 == *k2;
1276}
1277
1278static UINT32 rail_window_key_hash(const void* key)
1279{
1280 const UINT64* k1 = (const UINT64*)key;
1281 return (UINT32)*k1;
1282}
1283
1284static void rail_window_free(void* value)
1285{
1286 xfAppWindow* appWindow = (xfAppWindow*)value;
1287
1288 if (!appWindow)
1289 return;
1290
1291 xf_DestroyWindow(appWindow->xfc, appWindow);
1292}
1293
1294int xf_rail_init(xfContext* xfc, RailClientContext* rail)
1295{
1296 rdpContext* context = (rdpContext*)xfc;
1297
1298 if (!xfc || !rail)
1299 return 0;
1300
1301 xfc->rail = rail;
1302 xf_rail_register_update_callbacks(context->update);
1303 rail->custom = (void*)xfc;
1304 rail->ServerExecuteResult = xf_rail_server_execute_result;
1305 rail->ServerSystemParam = xf_rail_server_system_param;
1306 rail->ServerHandshake = xf_rail_server_handshake;
1307 rail->ServerHandshakeEx = xf_rail_server_handshake_ex;
1308 rail->ServerLocalMoveSize = xf_rail_server_local_move_size;
1309 rail->ServerMinMaxInfo = xf_rail_server_min_max_info;
1310 rail->ServerLanguageBarInfo = xf_rail_server_language_bar_info;
1311 rail->ServerGetAppIdResponse = xf_rail_server_get_appid_response;
1312 xfc->railWindows = HashTable_New(TRUE);
1313
1314 if (!xfc->railWindows)
1315 return 0;
1316
1317 if (!HashTable_SetHashFunction(xfc->railWindows, rail_window_key_hash))
1318 goto fail;
1319 {
1320 wObject* obj = HashTable_KeyObject(xfc->railWindows);
1321 obj->fnObjectEquals = rail_window_key_equals;
1322 }
1323 {
1324 wObject* obj = HashTable_ValueObject(xfc->railWindows);
1325 obj->fnObjectFree = rail_window_free;
1326 }
1327 xfc->railIconCache = RailIconCache_New(xfc->common.context.settings);
1328
1329 if (!xfc->railIconCache)
1330 {
1331 }
1332
1333 return 1;
1334fail:
1335 HashTable_Free(xfc->railWindows);
1336 return 0;
1337}
1338
1339int xf_rail_uninit(xfContext* xfc, RailClientContext* rail)
1340{
1341 WINPR_UNUSED(rail);
1342
1343 if (xfc->rail)
1344 {
1345 xfc->rail->custom = nullptr;
1346 xfc->rail = nullptr;
1347 }
1348
1349 if (xfc->railWindows)
1350 {
1351 HashTable_Free(xfc->railWindows);
1352 xfc->railWindows = nullptr;
1353 }
1354
1355 if (xfc->railIconCache)
1356 {
1357 RailIconCache_Free(xfc->railIconCache);
1358 xfc->railIconCache = nullptr;
1359 }
1360
1361 return 1;
1362}
1363
1364xfAppWindow* xf_rail_add_window(xfContext* xfc, UINT64 id, INT32 x, INT32 y, UINT32 width,
1365 UINT32 height, UINT32 surfaceId)
1366{
1367 if (!xfc)
1368 return nullptr;
1369
1370 xfAppWindow* appWindow = (xfAppWindow*)calloc(1, sizeof(xfAppWindow));
1371
1372 if (!appWindow)
1373 return nullptr;
1374
1375 appWindow->xfc = xfc;
1376 appWindow->windowId = id;
1377 appWindow->surfaceId = surfaceId;
1378 appWindow->x = x;
1379 appWindow->y = y;
1380 appWindow->width = WINPR_ASSERTING_INT_CAST(int, width);
1381 appWindow->height = WINPR_ASSERTING_INT_CAST(int, height);
1382
1383 xf_AppWindowsLock(xfc);
1384 if (!xf_AppWindowCreate(xfc, appWindow))
1385 goto fail;
1386
1387 if (!HashTable_Insert(xfc->railWindows, &appWindow->windowId, (void*)appWindow))
1388 goto fail;
1389 return appWindow;
1390fail:
1391 rail_window_free(appWindow);
1392 xf_AppWindowsUnlock(xfc);
1393 return nullptr;
1394}
1395
1396BOOL xf_rail_del_window(xfContext* xfc, UINT64 id)
1397{
1398 if (!xfc)
1399 return FALSE;
1400
1401 if (!xfc->railWindows)
1402 return FALSE;
1403
1404 return HashTable_Remove(xfc->railWindows, &id);
1405}
1406
1407void xf_rail_return_windowFrom(xfAppWindow* window, const char* file, const char* fkt, size_t line)
1408{
1409 if (!window)
1410 return;
1411
1412 xfAppWindowsUnlockFrom(window->xfc, file, fkt, line);
1413}
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
This struct contains function pointer to initialize/free objects.
Definition collections.h:52
OBJECT_FREE_FN fnObjectFree
Definition collections.h:58
OBJECT_EQUALS_FN fnObjectEquals
Definition collections.h:59