Wise&mystical  1.0
Project about Europe
Loading...
Searching...
No Matches
wl_init.c
Go to the documentation of this file.
1//========================================================================
2// GLFW 3.4 Wayland - www.glfw.org
3//------------------------------------------------------------------------
4// Copyright (c) 2014 Jonas Ådahl <jadahl@gmail.com>
5//
6// This software is provided 'as-is', without any express or implied
7// warranty. In no event will the authors be held liable for any damages
8// arising from the use of this software.
9//
10// Permission is granted to anyone to use this software for any purpose,
11// including commercial applications, and to alter it and redistribute it
12// freely, subject to the following restrictions:
13//
14// 1. The origin of this software must not be misrepresented; you must not
15// claim that you wrote the original software. If you use this software
16// in a product, an acknowledgment in the product documentation would
17// be appreciated but is not required.
18//
19// 2. Altered source versions must be plainly marked as such, and must not
20// be misrepresented as being the original software.
21//
22// 3. This notice may not be removed or altered from any source
23// distribution.
24//
25//========================================================================
26// It is fine to use C99 in this file because it will not be built with VS
27//========================================================================
28
29#define _POSIX_C_SOURCE 199309L
30
31#include "internal.h"
32
33#include <assert.h>
34#include <errno.h>
35#include <limits.h>
36#include <linux/input.h>
37#include <stdio.h>
38#include <stdlib.h>
39#include <string.h>
40#include <sys/mman.h>
41#include <sys/timerfd.h>
42#include <unistd.h>
43#include <time.h>
44#include <wayland-client.h>
45
46
47static inline int min(int n1, int n2)
48{
49 return n1 < n2 ? n1 : n2;
50}
51
52static _GLFWwindow* findWindowFromDecorationSurface(struct wl_surface* surface,
53 int* which)
54{
55 int focus;
57 if (!which)
58 which = &focus;
59 while (window)
60 {
61 if (surface == window->wl.decorations.top.surface)
62 {
63 *which = topDecoration;
64 break;
65 }
66 if (surface == window->wl.decorations.left.surface)
67 {
68 *which = leftDecoration;
69 break;
70 }
71 if (surface == window->wl.decorations.right.surface)
72 {
73 *which = rightDecoration;
74 break;
75 }
76 if (surface == window->wl.decorations.bottom.surface)
77 {
78 *which = bottomDecoration;
79 break;
80 }
81 window = window->next;
82 }
83 return window;
84}
85
86static void pointerHandleEnter(void* data,
87 struct wl_pointer* pointer,
88 uint32_t serial,
89 struct wl_surface* surface,
90 wl_fixed_t sx,
91 wl_fixed_t sy)
92{
93 // Happens in the case we just destroyed the surface.
94 if (!surface)
95 return;
96
97 int focus = 0;
98 _GLFWwindow* window = wl_surface_get_user_data(surface);
99 if (!window)
100 {
101 window = findWindowFromDecorationSurface(surface, &focus);
102 if (!window)
103 return;
104 }
105
106 window->wl.decorations.focus = focus;
107 _glfw.wl.serial = serial;
108 _glfw.wl.pointerFocus = window;
109
110 window->wl.hovered = GLFW_TRUE;
111
112 _glfwPlatformSetCursor(window, window->wl.currentCursor);
114}
115
116static void pointerHandleLeave(void* data,
117 struct wl_pointer* pointer,
118 uint32_t serial,
119 struct wl_surface* surface)
120{
121 _GLFWwindow* window = _glfw.wl.pointerFocus;
122
123 if (!window)
124 return;
125
126 window->wl.hovered = GLFW_FALSE;
127
128 _glfw.wl.serial = serial;
129 _glfw.wl.pointerFocus = NULL;
131 _glfw.wl.cursorPreviousName = NULL;
132}
133
134static void setCursor(_GLFWwindow* window, const char* name)
135{
136 struct wl_buffer* buffer;
137 struct wl_cursor* cursor;
138 struct wl_cursor_image* image;
139 struct wl_surface* surface = _glfw.wl.cursorSurface;
140 struct wl_cursor_theme* theme = _glfw.wl.cursorTheme;
141 int scale = 1;
142
143 if (window->wl.scale > 1 && _glfw.wl.cursorThemeHiDPI)
144 {
145 // We only support up to scale=2 for now, since libwayland-cursor
146 // requires us to load a different theme for each size.
147 scale = 2;
148 theme = _glfw.wl.cursorThemeHiDPI;
149 }
150
151 cursor = wl_cursor_theme_get_cursor(theme, name);
152 if (!cursor)
153 {
155 "Wayland: Standard cursor not found");
156 return;
157 }
158 // TODO: handle animated cursors too.
159 image = cursor->images[0];
160
161 if (!image)
162 return;
163
164 buffer = wl_cursor_image_get_buffer(image);
165 if (!buffer)
166 return;
167 wl_pointer_set_cursor(_glfw.wl.pointer, _glfw.wl.serial,
168 surface,
169 image->hotspot_x / scale,
170 image->hotspot_y / scale);
171 wl_surface_set_buffer_scale(surface, scale);
172 wl_surface_attach(surface, buffer, 0, 0);
173 wl_surface_damage(surface, 0, 0,
174 image->width, image->height);
175 wl_surface_commit(surface);
176 _glfw.wl.cursorPreviousName = name;
177}
178
179static void pointerHandleMotion(void* data,
180 struct wl_pointer* pointer,
181 uint32_t time,
182 wl_fixed_t sx,
183 wl_fixed_t sy)
184{
185 _GLFWwindow* window = _glfw.wl.pointerFocus;
186 const char* cursorName = NULL;
187 double x, y;
188
189 if (!window)
190 return;
191
192 if (window->cursorMode == GLFW_CURSOR_DISABLED)
193 return;
194 x = wl_fixed_to_double(sx);
195 y = wl_fixed_to_double(sy);
196
197 switch (window->wl.decorations.focus)
198 {
199 case mainWindow:
200 window->wl.cursorPosX = x;
201 window->wl.cursorPosY = y;
202 _glfwInputCursorPos(window, x, y);
203 _glfw.wl.cursorPreviousName = NULL;
204 return;
205 case topDecoration:
207 cursorName = "n-resize";
208 else
209 cursorName = "left_ptr";
210 break;
211 case leftDecoration:
213 cursorName = "nw-resize";
214 else
215 cursorName = "w-resize";
216 break;
217 case rightDecoration:
219 cursorName = "ne-resize";
220 else
221 cursorName = "e-resize";
222 break;
223 case bottomDecoration:
225 cursorName = "sw-resize";
226 else if (x > window->wl.width + _GLFW_DECORATION_WIDTH)
227 cursorName = "se-resize";
228 else
229 cursorName = "s-resize";
230 break;
231 default:
232 assert(0);
233 }
234 if (_glfw.wl.cursorPreviousName != cursorName)
235 setCursor(window, cursorName);
236}
237
238static void pointerHandleButton(void* data,
239 struct wl_pointer* pointer,
240 uint32_t serial,
241 uint32_t time,
242 uint32_t button,
243 uint32_t state)
244{
245 _GLFWwindow* window = _glfw.wl.pointerFocus;
246 int glfwButton;
247 uint32_t edges = XDG_TOPLEVEL_RESIZE_EDGE_NONE;
248
249 if (!window)
250 return;
251 if (button == BTN_LEFT)
252 {
253 switch (window->wl.decorations.focus)
254 {
255 case mainWindow:
256 break;
257 case topDecoration:
258 if (window->wl.cursorPosY < _GLFW_DECORATION_WIDTH)
259 edges = XDG_TOPLEVEL_RESIZE_EDGE_TOP;
260 else
261 {
262 xdg_toplevel_move(window->wl.xdg.toplevel, _glfw.wl.seat, serial);
263 }
264 break;
265 case leftDecoration:
266 if (window->wl.cursorPosY < _GLFW_DECORATION_WIDTH)
267 edges = XDG_TOPLEVEL_RESIZE_EDGE_TOP_LEFT;
268 else
269 edges = XDG_TOPLEVEL_RESIZE_EDGE_LEFT;
270 break;
271 case rightDecoration:
272 if (window->wl.cursorPosY < _GLFW_DECORATION_WIDTH)
273 edges = XDG_TOPLEVEL_RESIZE_EDGE_TOP_RIGHT;
274 else
275 edges = XDG_TOPLEVEL_RESIZE_EDGE_RIGHT;
276 break;
277 case bottomDecoration:
278 if (window->wl.cursorPosX < _GLFW_DECORATION_WIDTH)
279 edges = XDG_TOPLEVEL_RESIZE_EDGE_BOTTOM_LEFT;
280 else if (window->wl.cursorPosX > window->wl.width + _GLFW_DECORATION_WIDTH)
281 edges = XDG_TOPLEVEL_RESIZE_EDGE_BOTTOM_RIGHT;
282 else
283 edges = XDG_TOPLEVEL_RESIZE_EDGE_BOTTOM;
284 break;
285 default:
286 assert(0);
287 }
288 if (edges != XDG_TOPLEVEL_RESIZE_EDGE_NONE)
289 {
290 xdg_toplevel_resize(window->wl.xdg.toplevel, _glfw.wl.seat,
291 serial, edges);
292 }
293 }
294 else if (button == BTN_RIGHT)
295 {
296 if (window->wl.decorations.focus != mainWindow && window->wl.xdg.toplevel)
297 {
298 xdg_toplevel_show_window_menu(window->wl.xdg.toplevel,
299 _glfw.wl.seat, serial,
300 window->wl.cursorPosX,
301 window->wl.cursorPosY);
302 return;
303 }
304 }
305
306 // Don’t pass the button to the user if it was related to a decoration.
307 if (window->wl.decorations.focus != mainWindow)
308 return;
309
310 _glfw.wl.serial = serial;
311
312 /* Makes left, right and middle 0, 1 and 2. Overall order follows evdev
313 * codes. */
314 glfwButton = button - BTN_LEFT;
315
317 glfwButton,
318 state == WL_POINTER_BUTTON_STATE_PRESSED
319 ? GLFW_PRESS
320 : GLFW_RELEASE,
321 _glfw.wl.xkb.modifiers);
322}
323
324static void pointerHandleAxis(void* data,
325 struct wl_pointer* pointer,
326 uint32_t time,
327 uint32_t axis,
328 wl_fixed_t value)
329{
330 _GLFWwindow* window = _glfw.wl.pointerFocus;
331 double x = 0.0, y = 0.0;
332 // Wayland scroll events are in pointer motion coordinate space (think two
333 // finger scroll). The factor 10 is commonly used to convert to "scroll
334 // step means 1.0.
335 const double scrollFactor = 1.0 / 10.0;
336
337 if (!window)
338 return;
339
340 assert(axis == WL_POINTER_AXIS_HORIZONTAL_SCROLL ||
341 axis == WL_POINTER_AXIS_VERTICAL_SCROLL);
342
343 if (axis == WL_POINTER_AXIS_HORIZONTAL_SCROLL)
344 x = -wl_fixed_to_double(value) * scrollFactor;
345 else if (axis == WL_POINTER_AXIS_VERTICAL_SCROLL)
346 y = -wl_fixed_to_double(value) * scrollFactor;
347
348 _glfwInputScroll(window, x, y);
349}
350
351static const struct wl_pointer_listener pointerListener = {
352 pointerHandleEnter,
353 pointerHandleLeave,
354 pointerHandleMotion,
355 pointerHandleButton,
356 pointerHandleAxis,
357};
358
359static void keyboardHandleKeymap(void* data,
360 struct wl_keyboard* keyboard,
361 uint32_t format,
362 int fd,
363 uint32_t size)
364{
365 struct xkb_keymap* keymap;
366 struct xkb_state* state;
367
368#ifdef HAVE_XKBCOMMON_COMPOSE_H
369 struct xkb_compose_table* composeTable;
370 struct xkb_compose_state* composeState;
371#endif
372
373 char* mapStr;
374 const char* locale;
375
376 if (format != WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1)
377 {
378 close(fd);
379 return;
380 }
381
382 mapStr = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0);
383 if (mapStr == MAP_FAILED) {
384 close(fd);
385 return;
386 }
387
389 mapStr,
390 XKB_KEYMAP_FORMAT_TEXT_V1,
391 0);
392 munmap(mapStr, size);
393 close(fd);
394
395 if (!keymap)
396 {
398 "Wayland: Failed to compile keymap");
399 return;
400 }
401
402 state = xkb_state_new(keymap);
403 if (!state)
404 {
406 "Wayland: Failed to create XKB state");
407 xkb_keymap_unref(keymap);
408 return;
409 }
410
411 // Look up the preferred locale, falling back to "C" as default.
412 locale = getenv("LC_ALL");
413 if (!locale)
414 locale = getenv("LC_CTYPE");
415 if (!locale)
416 locale = getenv("LANG");
417 if (!locale)
418 locale = "C";
419
420#ifdef HAVE_XKBCOMMON_COMPOSE_H
421 composeTable =
422 xkb_compose_table_new_from_locale(_glfw.wl.xkb.context, locale,
423 XKB_COMPOSE_COMPILE_NO_FLAGS);
424 if (composeTable)
425 {
426 composeState =
427 xkb_compose_state_new(composeTable, XKB_COMPOSE_STATE_NO_FLAGS);
428 xkb_compose_table_unref(composeTable);
429 if (composeState)
430 _glfw.wl.xkb.composeState = composeState;
431 else
433 "Wayland: Failed to create XKB compose state");
434 }
435 else
436 {
438 "Wayland: Failed to create XKB compose table");
439 }
440#endif
441
442 xkb_keymap_unref(_glfw.wl.xkb.keymap);
443 xkb_state_unref(_glfw.wl.xkb.state);
444 _glfw.wl.xkb.keymap = keymap;
445 _glfw.wl.xkb.state = state;
446
447 _glfw.wl.xkb.controlMask =
448 1 << xkb_keymap_mod_get_index(_glfw.wl.xkb.keymap, "Control");
449 _glfw.wl.xkb.altMask =
450 1 << xkb_keymap_mod_get_index(_glfw.wl.xkb.keymap, "Mod1");
451 _glfw.wl.xkb.shiftMask =
452 1 << xkb_keymap_mod_get_index(_glfw.wl.xkb.keymap, "Shift");
453 _glfw.wl.xkb.superMask =
454 1 << xkb_keymap_mod_get_index(_glfw.wl.xkb.keymap, "Mod4");
455 _glfw.wl.xkb.capsLockMask =
456 1 << xkb_keymap_mod_get_index(_glfw.wl.xkb.keymap, "Lock");
457 _glfw.wl.xkb.numLockMask =
458 1 << xkb_keymap_mod_get_index(_glfw.wl.xkb.keymap, "Mod2");
459}
460
461static void keyboardHandleEnter(void* data,
462 struct wl_keyboard* keyboard,
463 uint32_t serial,
464 struct wl_surface* surface,
465 struct wl_array* keys)
466{
467 // Happens in the case we just destroyed the surface.
468 if (!surface)
469 return;
470
471 _GLFWwindow* window = wl_surface_get_user_data(surface);
472 if (!window)
473 {
474 window = findWindowFromDecorationSurface(surface, NULL);
475 if (!window)
476 return;
477 }
478
479 _glfw.wl.serial = serial;
480 _glfw.wl.keyboardFocus = window;
482}
483
484static void keyboardHandleLeave(void* data,
485 struct wl_keyboard* keyboard,
486 uint32_t serial,
487 struct wl_surface* surface)
488{
489 _GLFWwindow* window = _glfw.wl.keyboardFocus;
490
491 if (!window)
492 return;
493
494 _glfw.wl.serial = serial;
495 _glfw.wl.keyboardFocus = NULL;
497}
498
499static int toGLFWKeyCode(uint32_t key)
500{
501 if (key < sizeof(_glfw.wl.keycodes) / sizeof(_glfw.wl.keycodes[0]))
502 return _glfw.wl.keycodes[key];
503
504 return GLFW_KEY_UNKNOWN;
505}
506
507#ifdef HAVE_XKBCOMMON_COMPOSE_H
508static xkb_keysym_t composeSymbol(xkb_keysym_t sym)
509{
510 if (sym == XKB_KEY_NoSymbol || !_glfw.wl.xkb.composeState)
511 return sym;
512 if (xkb_compose_state_feed(_glfw.wl.xkb.composeState, sym)
513 != XKB_COMPOSE_FEED_ACCEPTED)
514 return sym;
515 switch (xkb_compose_state_get_status(_glfw.wl.xkb.composeState))
516 {
517 case XKB_COMPOSE_COMPOSED:
518 return xkb_compose_state_get_one_sym(_glfw.wl.xkb.composeState);
519 case XKB_COMPOSE_COMPOSING:
520 case XKB_COMPOSE_CANCELLED:
521 return XKB_KEY_NoSymbol;
522 case XKB_COMPOSE_NOTHING:
523 default:
524 return sym;
525 }
526}
527#endif
528
529static GLFWbool inputChar(_GLFWwindow* window, uint32_t key)
530{
531 uint32_t code, numSyms;
532 long cp;
533 const xkb_keysym_t *syms;
534 xkb_keysym_t sym;
535
536 code = key + 8;
537 numSyms = xkb_state_key_get_syms(_glfw.wl.xkb.state, code, &syms);
538
539 if (numSyms == 1)
540 {
541#ifdef HAVE_XKBCOMMON_COMPOSE_H
542 sym = composeSymbol(syms[0]);
543#else
544 sym = syms[0];
545#endif
546 cp = _glfwKeySym2Unicode(sym);
547 if (cp != -1)
548 {
549 const int mods = _glfw.wl.xkb.modifiers;
550 const int plain = !(mods & (GLFW_MOD_CONTROL | GLFW_MOD_ALT));
551 _glfwInputChar(window, cp, mods, plain);
552 }
553 }
554
555 return xkb_keymap_key_repeats(_glfw.wl.xkb.keymap, syms[0]);
556}
557
558static void keyboardHandleKey(void* data,
559 struct wl_keyboard* keyboard,
560 uint32_t serial,
561 uint32_t time,
562 uint32_t key,
563 uint32_t state)
564{
565 int keyCode;
566 int action;
567 _GLFWwindow* window = _glfw.wl.keyboardFocus;
568 GLFWbool shouldRepeat;
569 struct itimerspec timer = {};
570
571 if (!window)
572 return;
573
574 keyCode = toGLFWKeyCode(key);
575 action = state == WL_KEYBOARD_KEY_STATE_PRESSED
577
578 _glfw.wl.serial = serial;
579 _glfwInputKey(window, keyCode, key, action,
580 _glfw.wl.xkb.modifiers);
581
582 if (action == GLFW_PRESS)
583 {
584 shouldRepeat = inputChar(window, key);
585
586 if (shouldRepeat && _glfw.wl.keyboardRepeatRate > 0)
587 {
588 _glfw.wl.keyboardLastKey = keyCode;
589 _glfw.wl.keyboardLastScancode = key;
590 if (_glfw.wl.keyboardRepeatRate > 1)
591 timer.it_interval.tv_nsec = 1000000000 / _glfw.wl.keyboardRepeatRate;
592 else
593 timer.it_interval.tv_sec = 1;
594 timer.it_value.tv_sec = _glfw.wl.keyboardRepeatDelay / 1000;
595 timer.it_value.tv_nsec = (_glfw.wl.keyboardRepeatDelay % 1000) * 1000000;
596 }
597 }
598 timerfd_settime(_glfw.wl.timerfd, 0, &timer, NULL);
599}
600
601static void keyboardHandleModifiers(void* data,
602 struct wl_keyboard* keyboard,
603 uint32_t serial,
604 uint32_t modsDepressed,
605 uint32_t modsLatched,
606 uint32_t modsLocked,
607 uint32_t group)
608{
609 xkb_mod_mask_t mask;
610 unsigned int modifiers = 0;
611
612 _glfw.wl.serial = serial;
613
614 if (!_glfw.wl.xkb.keymap)
615 return;
616
617 xkb_state_update_mask(_glfw.wl.xkb.state,
618 modsDepressed,
619 modsLatched,
620 modsLocked,
621 0,
622 0,
623 group);
624
625 mask = xkb_state_serialize_mods(_glfw.wl.xkb.state,
626 XKB_STATE_MODS_DEPRESSED |
627 XKB_STATE_LAYOUT_DEPRESSED |
628 XKB_STATE_MODS_LATCHED |
629 XKB_STATE_LAYOUT_LATCHED);
630 if (mask & _glfw.wl.xkb.controlMask)
631 modifiers |= GLFW_MOD_CONTROL;
632 if (mask & _glfw.wl.xkb.altMask)
633 modifiers |= GLFW_MOD_ALT;
634 if (mask & _glfw.wl.xkb.shiftMask)
635 modifiers |= GLFW_MOD_SHIFT;
636 if (mask & _glfw.wl.xkb.superMask)
637 modifiers |= GLFW_MOD_SUPER;
638 if (mask & _glfw.wl.xkb.capsLockMask)
639 modifiers |= GLFW_MOD_CAPS_LOCK;
640 if (mask & _glfw.wl.xkb.numLockMask)
641 modifiers |= GLFW_MOD_NUM_LOCK;
642 _glfw.wl.xkb.modifiers = modifiers;
643}
644
645#ifdef WL_KEYBOARD_REPEAT_INFO_SINCE_VERSION
646static void keyboardHandleRepeatInfo(void* data,
647 struct wl_keyboard* keyboard,
648 int32_t rate,
649 int32_t delay)
650{
651 if (keyboard != _glfw.wl.keyboard)
652 return;
653
654 _glfw.wl.keyboardRepeatRate = rate;
655 _glfw.wl.keyboardRepeatDelay = delay;
656}
657#endif
658
659static const struct wl_keyboard_listener keyboardListener = {
660 keyboardHandleKeymap,
661 keyboardHandleEnter,
662 keyboardHandleLeave,
663 keyboardHandleKey,
664 keyboardHandleModifiers,
665#ifdef WL_KEYBOARD_REPEAT_INFO_SINCE_VERSION
666 keyboardHandleRepeatInfo,
667#endif
668};
669
670static void seatHandleCapabilities(void* data,
671 struct wl_seat* seat,
672 enum wl_seat_capability caps)
673{
674 if ((caps & WL_SEAT_CAPABILITY_POINTER) && !_glfw.wl.pointer)
675 {
676 _glfw.wl.pointer = wl_seat_get_pointer(seat);
677 wl_pointer_add_listener(_glfw.wl.pointer, &pointerListener, NULL);
678 }
679 else if (!(caps & WL_SEAT_CAPABILITY_POINTER) && _glfw.wl.pointer)
680 {
681 wl_pointer_destroy(_glfw.wl.pointer);
682 _glfw.wl.pointer = NULL;
683 }
684
685 if ((caps & WL_SEAT_CAPABILITY_KEYBOARD) && !_glfw.wl.keyboard)
686 {
687 _glfw.wl.keyboard = wl_seat_get_keyboard(seat);
688 wl_keyboard_add_listener(_glfw.wl.keyboard, &keyboardListener, NULL);
689 }
690 else if (!(caps & WL_SEAT_CAPABILITY_KEYBOARD) && _glfw.wl.keyboard)
691 {
692 wl_keyboard_destroy(_glfw.wl.keyboard);
693 _glfw.wl.keyboard = NULL;
694 }
695}
696
697static void seatHandleName(void* data,
698 struct wl_seat* seat,
699 const char* name)
700{
701}
702
703static const struct wl_seat_listener seatListener = {
704 seatHandleCapabilities,
705 seatHandleName,
706};
707
708static void dataOfferHandleOffer(void* data,
709 struct wl_data_offer* dataOffer,
710 const char* mimeType)
711{
712}
713
714static const struct wl_data_offer_listener dataOfferListener = {
715 dataOfferHandleOffer,
716};
717
718static void dataDeviceHandleDataOffer(void* data,
719 struct wl_data_device* dataDevice,
720 struct wl_data_offer* id)
721{
722 if (_glfw.wl.dataOffer)
723 wl_data_offer_destroy(_glfw.wl.dataOffer);
724
725 _glfw.wl.dataOffer = id;
726 wl_data_offer_add_listener(_glfw.wl.dataOffer, &dataOfferListener, NULL);
727}
728
729static void dataDeviceHandleEnter(void* data,
730 struct wl_data_device* dataDevice,
731 uint32_t serial,
732 struct wl_surface *surface,
733 wl_fixed_t x,
734 wl_fixed_t y,
735 struct wl_data_offer *id)
736{
737}
738
739static void dataDeviceHandleLeave(void* data,
740 struct wl_data_device* dataDevice)
741{
742}
743
744static void dataDeviceHandleMotion(void* data,
745 struct wl_data_device* dataDevice,
746 uint32_t time,
747 wl_fixed_t x,
748 wl_fixed_t y)
749{
750}
751
752static void dataDeviceHandleDrop(void* data,
753 struct wl_data_device* dataDevice)
754{
755}
756
757static void dataDeviceHandleSelection(void* data,
758 struct wl_data_device* dataDevice,
759 struct wl_data_offer* id)
760{
761}
762
763static const struct wl_data_device_listener dataDeviceListener = {
764 dataDeviceHandleDataOffer,
765 dataDeviceHandleEnter,
766 dataDeviceHandleLeave,
767 dataDeviceHandleMotion,
768 dataDeviceHandleDrop,
769 dataDeviceHandleSelection,
770};
771
772static void wmBaseHandlePing(void* data,
773 struct xdg_wm_base* wmBase,
774 uint32_t serial)
775{
776 xdg_wm_base_pong(wmBase, serial);
777}
778
779static const struct xdg_wm_base_listener wmBaseListener = {
780 wmBaseHandlePing
781};
782
783static void registryHandleGlobal(void* data,
784 struct wl_registry* registry,
785 uint32_t name,
786 const char* interface,
787 uint32_t version)
788{
789 if (strcmp(interface, "wl_compositor") == 0)
790 {
791 _glfw.wl.compositorVersion = min(3, version);
792 _glfw.wl.compositor =
793 wl_registry_bind(registry, name, &wl_compositor_interface,
794 _glfw.wl.compositorVersion);
795 }
796 else if (strcmp(interface, "wl_subcompositor") == 0)
797 {
798 _glfw.wl.subcompositor =
799 wl_registry_bind(registry, name, &wl_subcompositor_interface, 1);
800 }
801 else if (strcmp(interface, "wl_shm") == 0)
802 {
803 _glfw.wl.shm =
804 wl_registry_bind(registry, name, &wl_shm_interface, 1);
805 }
806 else if (strcmp(interface, "wl_output") == 0)
807 {
808 _glfwAddOutputWayland(name, version);
809 }
810 else if (strcmp(interface, "wl_seat") == 0)
811 {
812 if (!_glfw.wl.seat)
813 {
814 _glfw.wl.seatVersion = min(4, version);
815 _glfw.wl.seat =
816 wl_registry_bind(registry, name, &wl_seat_interface,
817 _glfw.wl.seatVersion);
818 wl_seat_add_listener(_glfw.wl.seat, &seatListener, NULL);
819 }
820 }
821 else if (strcmp(interface, "wl_data_device_manager") == 0)
822 {
823 if (!_glfw.wl.dataDeviceManager)
824 {
825 _glfw.wl.dataDeviceManager =
826 wl_registry_bind(registry, name,
827 &wl_data_device_manager_interface, 1);
828 }
829 }
830 else if (strcmp(interface, "xdg_wm_base") == 0)
831 {
832 _glfw.wl.wmBase =
833 wl_registry_bind(registry, name, &xdg_wm_base_interface, 1);
834 xdg_wm_base_add_listener(_glfw.wl.wmBase, &wmBaseListener, NULL);
835 }
836 else if (strcmp(interface, "zxdg_decoration_manager_v1") == 0)
837 {
838 _glfw.wl.decorationManager =
839 wl_registry_bind(registry, name,
840 &zxdg_decoration_manager_v1_interface,
841 1);
842 }
843 else if (strcmp(interface, "wp_viewporter") == 0)
844 {
845 _glfw.wl.viewporter =
846 wl_registry_bind(registry, name, &wp_viewporter_interface, 1);
847 }
848 else if (strcmp(interface, "zwp_relative_pointer_manager_v1") == 0)
849 {
850 _glfw.wl.relativePointerManager =
851 wl_registry_bind(registry, name,
852 &zwp_relative_pointer_manager_v1_interface,
853 1);
854 }
855 else if (strcmp(interface, "zwp_pointer_constraints_v1") == 0)
856 {
857 _glfw.wl.pointerConstraints =
858 wl_registry_bind(registry, name,
859 &zwp_pointer_constraints_v1_interface,
860 1);
861 }
862 else if (strcmp(interface, "zwp_idle_inhibit_manager_v1") == 0)
863 {
864 _glfw.wl.idleInhibitManager =
865 wl_registry_bind(registry, name,
866 &zwp_idle_inhibit_manager_v1_interface,
867 1);
868 }
869}
870
871static void registryHandleGlobalRemove(void *data,
872 struct wl_registry *registry,
873 uint32_t name)
874{
875 int i;
876 _GLFWmonitor* monitor;
877
878 for (i = 0; i < _glfw.monitorCount; ++i)
879 {
880 monitor = _glfw.monitors[i];
881 if (monitor->wl.name == name)
882 {
884 return;
885 }
886 }
887}
888
889
890static const struct wl_registry_listener registryListener = {
891 registryHandleGlobal,
892 registryHandleGlobalRemove
893};
894
895// Create key code translation tables
896//
897static void createKeyTables(void)
898{
899 int scancode;
900
901 memset(_glfw.wl.keycodes, -1, sizeof(_glfw.wl.keycodes));
902 memset(_glfw.wl.scancodes, -1, sizeof(_glfw.wl.scancodes));
903
905 _glfw.wl.keycodes[KEY_1] = GLFW_KEY_1;
906 _glfw.wl.keycodes[KEY_2] = GLFW_KEY_2;
907 _glfw.wl.keycodes[KEY_3] = GLFW_KEY_3;
908 _glfw.wl.keycodes[KEY_4] = GLFW_KEY_4;
909 _glfw.wl.keycodes[KEY_5] = GLFW_KEY_5;
910 _glfw.wl.keycodes[KEY_6] = GLFW_KEY_6;
911 _glfw.wl.keycodes[KEY_7] = GLFW_KEY_7;
912 _glfw.wl.keycodes[KEY_8] = GLFW_KEY_8;
913 _glfw.wl.keycodes[KEY_9] = GLFW_KEY_9;
914 _glfw.wl.keycodes[KEY_0] = GLFW_KEY_0;
915 _glfw.wl.keycodes[KEY_SPACE] = GLFW_KEY_SPACE;
916 _glfw.wl.keycodes[KEY_MINUS] = GLFW_KEY_MINUS;
917 _glfw.wl.keycodes[KEY_EQUAL] = GLFW_KEY_EQUAL;
918 _glfw.wl.keycodes[KEY_Q] = GLFW_KEY_Q;
919 _glfw.wl.keycodes[KEY_W] = GLFW_KEY_W;
920 _glfw.wl.keycodes[KEY_E] = GLFW_KEY_E;
921 _glfw.wl.keycodes[KEY_R] = GLFW_KEY_R;
922 _glfw.wl.keycodes[KEY_T] = GLFW_KEY_T;
923 _glfw.wl.keycodes[KEY_Y] = GLFW_KEY_Y;
924 _glfw.wl.keycodes[KEY_U] = GLFW_KEY_U;
925 _glfw.wl.keycodes[KEY_I] = GLFW_KEY_I;
926 _glfw.wl.keycodes[KEY_O] = GLFW_KEY_O;
927 _glfw.wl.keycodes[KEY_P] = GLFW_KEY_P;
928 _glfw.wl.keycodes[KEY_LEFTBRACE] = GLFW_KEY_LEFT_BRACKET;
929 _glfw.wl.keycodes[KEY_RIGHTBRACE] = GLFW_KEY_RIGHT_BRACKET;
930 _glfw.wl.keycodes[KEY_A] = GLFW_KEY_A;
931 _glfw.wl.keycodes[KEY_S] = GLFW_KEY_S;
932 _glfw.wl.keycodes[KEY_D] = GLFW_KEY_D;
933 _glfw.wl.keycodes[KEY_F] = GLFW_KEY_F;
934 _glfw.wl.keycodes[KEY_G] = GLFW_KEY_G;
935 _glfw.wl.keycodes[KEY_H] = GLFW_KEY_H;
936 _glfw.wl.keycodes[KEY_J] = GLFW_KEY_J;
937 _glfw.wl.keycodes[KEY_K] = GLFW_KEY_K;
938 _glfw.wl.keycodes[KEY_L] = GLFW_KEY_L;
941 _glfw.wl.keycodes[KEY_Z] = GLFW_KEY_Z;
942 _glfw.wl.keycodes[KEY_X] = GLFW_KEY_X;
943 _glfw.wl.keycodes[KEY_C] = GLFW_KEY_C;
944 _glfw.wl.keycodes[KEY_V] = GLFW_KEY_V;
945 _glfw.wl.keycodes[KEY_B] = GLFW_KEY_B;
946 _glfw.wl.keycodes[KEY_N] = GLFW_KEY_N;
947 _glfw.wl.keycodes[KEY_M] = GLFW_KEY_M;
948 _glfw.wl.keycodes[KEY_COMMA] = GLFW_KEY_COMMA;
949 _glfw.wl.keycodes[KEY_DOT] = GLFW_KEY_PERIOD;
950 _glfw.wl.keycodes[KEY_SLASH] = GLFW_KEY_SLASH;
952 _glfw.wl.keycodes[KEY_ESC] = GLFW_KEY_ESCAPE;
953 _glfw.wl.keycodes[KEY_TAB] = GLFW_KEY_TAB;
954 _glfw.wl.keycodes[KEY_LEFTSHIFT] = GLFW_KEY_LEFT_SHIFT;
955 _glfw.wl.keycodes[KEY_RIGHTSHIFT] = GLFW_KEY_RIGHT_SHIFT;
956 _glfw.wl.keycodes[KEY_LEFTCTRL] = GLFW_KEY_LEFT_CONTROL;
957 _glfw.wl.keycodes[KEY_RIGHTCTRL] = GLFW_KEY_RIGHT_CONTROL;
958 _glfw.wl.keycodes[KEY_LEFTALT] = GLFW_KEY_LEFT_ALT;
959 _glfw.wl.keycodes[KEY_RIGHTALT] = GLFW_KEY_RIGHT_ALT;
960 _glfw.wl.keycodes[KEY_LEFTMETA] = GLFW_KEY_LEFT_SUPER;
961 _glfw.wl.keycodes[KEY_RIGHTMETA] = GLFW_KEY_RIGHT_SUPER;
962 _glfw.wl.keycodes[KEY_MENU] = GLFW_KEY_MENU;
963 _glfw.wl.keycodes[KEY_NUMLOCK] = GLFW_KEY_NUM_LOCK;
964 _glfw.wl.keycodes[KEY_CAPSLOCK] = GLFW_KEY_CAPS_LOCK;
965 _glfw.wl.keycodes[KEY_PRINT] = GLFW_KEY_PRINT_SCREEN;
966 _glfw.wl.keycodes[KEY_SCROLLLOCK] = GLFW_KEY_SCROLL_LOCK;
967 _glfw.wl.keycodes[KEY_PAUSE] = GLFW_KEY_PAUSE;
968 _glfw.wl.keycodes[KEY_DELETE] = GLFW_KEY_DELETE;
970 _glfw.wl.keycodes[KEY_ENTER] = GLFW_KEY_ENTER;
971 _glfw.wl.keycodes[KEY_HOME] = GLFW_KEY_HOME;
972 _glfw.wl.keycodes[KEY_END] = GLFW_KEY_END;
973 _glfw.wl.keycodes[KEY_PAGEUP] = GLFW_KEY_PAGE_UP;
974 _glfw.wl.keycodes[KEY_PAGEDOWN] = GLFW_KEY_PAGE_DOWN;
975 _glfw.wl.keycodes[KEY_INSERT] = GLFW_KEY_INSERT;
976 _glfw.wl.keycodes[KEY_LEFT] = GLFW_KEY_LEFT;
977 _glfw.wl.keycodes[KEY_RIGHT] = GLFW_KEY_RIGHT;
978 _glfw.wl.keycodes[KEY_DOWN] = GLFW_KEY_DOWN;
979 _glfw.wl.keycodes[KEY_UP] = GLFW_KEY_UP;
980 _glfw.wl.keycodes[KEY_F1] = GLFW_KEY_F1;
981 _glfw.wl.keycodes[KEY_F2] = GLFW_KEY_F2;
982 _glfw.wl.keycodes[KEY_F3] = GLFW_KEY_F3;
983 _glfw.wl.keycodes[KEY_F4] = GLFW_KEY_F4;
984 _glfw.wl.keycodes[KEY_F5] = GLFW_KEY_F5;
985 _glfw.wl.keycodes[KEY_F6] = GLFW_KEY_F6;
986 _glfw.wl.keycodes[KEY_F7] = GLFW_KEY_F7;
987 _glfw.wl.keycodes[KEY_F8] = GLFW_KEY_F8;
988 _glfw.wl.keycodes[KEY_F9] = GLFW_KEY_F9;
989 _glfw.wl.keycodes[KEY_F10] = GLFW_KEY_F10;
990 _glfw.wl.keycodes[KEY_F11] = GLFW_KEY_F11;
991 _glfw.wl.keycodes[KEY_F12] = GLFW_KEY_F12;
992 _glfw.wl.keycodes[KEY_F13] = GLFW_KEY_F13;
993 _glfw.wl.keycodes[KEY_F14] = GLFW_KEY_F14;
994 _glfw.wl.keycodes[KEY_F15] = GLFW_KEY_F15;
995 _glfw.wl.keycodes[KEY_F16] = GLFW_KEY_F16;
996 _glfw.wl.keycodes[KEY_F17] = GLFW_KEY_F17;
997 _glfw.wl.keycodes[KEY_F18] = GLFW_KEY_F18;
998 _glfw.wl.keycodes[KEY_F19] = GLFW_KEY_F19;
999 _glfw.wl.keycodes[KEY_F20] = GLFW_KEY_F20;
1000 _glfw.wl.keycodes[KEY_F21] = GLFW_KEY_F21;
1001 _glfw.wl.keycodes[KEY_F22] = GLFW_KEY_F22;
1002 _glfw.wl.keycodes[KEY_F23] = GLFW_KEY_F23;
1003 _glfw.wl.keycodes[KEY_F24] = GLFW_KEY_F24;
1004 _glfw.wl.keycodes[KEY_KPSLASH] = GLFW_KEY_KP_DIVIDE;
1005 _glfw.wl.keycodes[KEY_KPDOT] = GLFW_KEY_KP_MULTIPLY;
1006 _glfw.wl.keycodes[KEY_KPMINUS] = GLFW_KEY_KP_SUBTRACT;
1007 _glfw.wl.keycodes[KEY_KPPLUS] = GLFW_KEY_KP_ADD;
1008 _glfw.wl.keycodes[KEY_KP0] = GLFW_KEY_KP_0;
1009 _glfw.wl.keycodes[KEY_KP1] = GLFW_KEY_KP_1;
1010 _glfw.wl.keycodes[KEY_KP2] = GLFW_KEY_KP_2;
1011 _glfw.wl.keycodes[KEY_KP3] = GLFW_KEY_KP_3;
1012 _glfw.wl.keycodes[KEY_KP4] = GLFW_KEY_KP_4;
1013 _glfw.wl.keycodes[KEY_KP5] = GLFW_KEY_KP_5;
1014 _glfw.wl.keycodes[KEY_KP6] = GLFW_KEY_KP_6;
1015 _glfw.wl.keycodes[KEY_KP7] = GLFW_KEY_KP_7;
1016 _glfw.wl.keycodes[KEY_KP8] = GLFW_KEY_KP_8;
1017 _glfw.wl.keycodes[KEY_KP9] = GLFW_KEY_KP_9;
1018 _glfw.wl.keycodes[KEY_KPCOMMA] = GLFW_KEY_KP_DECIMAL;
1019 _glfw.wl.keycodes[KEY_KPEQUAL] = GLFW_KEY_KP_EQUAL;
1020 _glfw.wl.keycodes[KEY_KPENTER] = GLFW_KEY_KP_ENTER;
1021
1022 for (scancode = 0; scancode < 256; scancode++)
1023 {
1024 if (_glfw.wl.keycodes[scancode] > 0)
1025 _glfw.wl.scancodes[_glfw.wl.keycodes[scancode]] = scancode;
1026 }
1027}
1028
1029
1033
1035{
1036 const char *cursorTheme;
1037 const char *cursorSizeStr;
1038 char *cursorSizeEnd;
1039 long cursorSizeLong;
1040 int cursorSize;
1041
1042 _glfw.wl.cursor.handle = _glfw_dlopen("libwayland-cursor.so.0");
1043 if (!_glfw.wl.cursor.handle)
1044 {
1046 "Wayland: Failed to open libwayland-cursor");
1047 return GLFW_FALSE;
1048 }
1049
1050 _glfw.wl.cursor.theme_load = (PFN_wl_cursor_theme_load)
1051 _glfw_dlsym(_glfw.wl.cursor.handle, "wl_cursor_theme_load");
1052 _glfw.wl.cursor.theme_destroy = (PFN_wl_cursor_theme_destroy)
1053 _glfw_dlsym(_glfw.wl.cursor.handle, "wl_cursor_theme_destroy");
1054 _glfw.wl.cursor.theme_get_cursor = (PFN_wl_cursor_theme_get_cursor)
1055 _glfw_dlsym(_glfw.wl.cursor.handle, "wl_cursor_theme_get_cursor");
1056 _glfw.wl.cursor.image_get_buffer = (PFN_wl_cursor_image_get_buffer)
1057 _glfw_dlsym(_glfw.wl.cursor.handle, "wl_cursor_image_get_buffer");
1058
1059 _glfw.wl.egl.handle = _glfw_dlopen("libwayland-egl.so.1");
1060 if (!_glfw.wl.egl.handle)
1061 {
1063 "Wayland: Failed to open libwayland-egl");
1064 return GLFW_FALSE;
1065 }
1066
1067 _glfw.wl.egl.window_create = (PFN_wl_egl_window_create)
1068 _glfw_dlsym(_glfw.wl.egl.handle, "wl_egl_window_create");
1069 _glfw.wl.egl.window_destroy = (PFN_wl_egl_window_destroy)
1070 _glfw_dlsym(_glfw.wl.egl.handle, "wl_egl_window_destroy");
1071 _glfw.wl.egl.window_resize = (PFN_wl_egl_window_resize)
1072 _glfw_dlsym(_glfw.wl.egl.handle, "wl_egl_window_resize");
1073
1074 _glfw.wl.xkb.handle = _glfw_dlopen("libxkbcommon.so.0");
1075 if (!_glfw.wl.xkb.handle)
1076 {
1078 "Wayland: Failed to open libxkbcommon");
1079 return GLFW_FALSE;
1080 }
1081
1082 _glfw.wl.xkb.context_new = (PFN_xkb_context_new)
1083 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_context_new");
1084 _glfw.wl.xkb.context_unref = (PFN_xkb_context_unref)
1085 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_context_unref");
1086 _glfw.wl.xkb.keymap_new_from_string = (PFN_xkb_keymap_new_from_string)
1087 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_keymap_new_from_string");
1088 _glfw.wl.xkb.keymap_unref = (PFN_xkb_keymap_unref)
1089 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_keymap_unref");
1090 _glfw.wl.xkb.keymap_mod_get_index = (PFN_xkb_keymap_mod_get_index)
1091 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_keymap_mod_get_index");
1092 _glfw.wl.xkb.keymap_key_repeats = (PFN_xkb_keymap_key_repeats)
1093 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_keymap_key_repeats");
1094 _glfw.wl.xkb.state_new = (PFN_xkb_state_new)
1095 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_state_new");
1096 _glfw.wl.xkb.state_unref = (PFN_xkb_state_unref)
1097 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_state_unref");
1098 _glfw.wl.xkb.state_key_get_syms = (PFN_xkb_state_key_get_syms)
1099 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_state_key_get_syms");
1100 _glfw.wl.xkb.state_update_mask = (PFN_xkb_state_update_mask)
1101 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_state_update_mask");
1102 _glfw.wl.xkb.state_serialize_mods = (PFN_xkb_state_serialize_mods)
1103 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_state_serialize_mods");
1104
1105#ifdef HAVE_XKBCOMMON_COMPOSE_H
1106 _glfw.wl.xkb.compose_table_new_from_locale = (PFN_xkb_compose_table_new_from_locale)
1107 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_table_new_from_locale");
1108 _glfw.wl.xkb.compose_table_unref = (PFN_xkb_compose_table_unref)
1109 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_table_unref");
1110 _glfw.wl.xkb.compose_state_new = (PFN_xkb_compose_state_new)
1111 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_state_new");
1112 _glfw.wl.xkb.compose_state_unref = (PFN_xkb_compose_state_unref)
1113 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_state_unref");
1114 _glfw.wl.xkb.compose_state_feed = (PFN_xkb_compose_state_feed)
1115 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_state_feed");
1116 _glfw.wl.xkb.compose_state_get_status = (PFN_xkb_compose_state_get_status)
1117 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_state_get_status");
1118 _glfw.wl.xkb.compose_state_get_one_sym = (PFN_xkb_compose_state_get_one_sym)
1119 _glfw_dlsym(_glfw.wl.xkb.handle, "xkb_compose_state_get_one_sym");
1120#endif
1121
1122 _glfw.wl.display = wl_display_connect(NULL);
1123 if (!_glfw.wl.display)
1124 {
1126 "Wayland: Failed to connect to display");
1127 return GLFW_FALSE;
1128 }
1129
1130 _glfw.wl.registry = wl_display_get_registry(_glfw.wl.display);
1131 wl_registry_add_listener(_glfw.wl.registry, &registryListener, NULL);
1132
1133 createKeyTables();
1134
1135 _glfw.wl.xkb.context = xkb_context_new(0);
1136 if (!_glfw.wl.xkb.context)
1137 {
1139 "Wayland: Failed to initialize xkb context");
1140 return GLFW_FALSE;
1141 }
1142
1143 // Sync so we got all registry objects
1144 wl_display_roundtrip(_glfw.wl.display);
1145
1146 // Sync so we got all initial output events
1147 wl_display_roundtrip(_glfw.wl.display);
1148
1150
1151 _glfw.wl.timerfd = -1;
1152 if (_glfw.wl.seatVersion >= 4)
1153 _glfw.wl.timerfd = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC);
1154
1155 if (!_glfw.wl.wmBase)
1156 {
1158 "Wayland: Failed to find xdg-shell in your compositor");
1159 return GLFW_FALSE;
1160 }
1161
1162 if (_glfw.wl.pointer && _glfw.wl.shm)
1163 {
1164 cursorTheme = getenv("XCURSOR_THEME");
1165 cursorSizeStr = getenv("XCURSOR_SIZE");
1166 cursorSize = 32;
1167 if (cursorSizeStr)
1168 {
1169 errno = 0;
1170 cursorSizeLong = strtol(cursorSizeStr, &cursorSizeEnd, 10);
1171 if (!*cursorSizeEnd && !errno && cursorSizeLong > 0 && cursorSizeLong <= INT_MAX)
1172 cursorSize = (int)cursorSizeLong;
1173 }
1174 _glfw.wl.cursorTheme =
1175 wl_cursor_theme_load(cursorTheme, cursorSize, _glfw.wl.shm);
1176 if (!_glfw.wl.cursorTheme)
1177 {
1179 "Wayland: Unable to load default cursor theme");
1180 return GLFW_FALSE;
1181 }
1182 // If this happens to be NULL, we just fallback to the scale=1 version.
1183 _glfw.wl.cursorThemeHiDPI =
1184 wl_cursor_theme_load(cursorTheme, 2 * cursorSize, _glfw.wl.shm);
1185 _glfw.wl.cursorSurface =
1186 wl_compositor_create_surface(_glfw.wl.compositor);
1187 _glfw.wl.cursorTimerfd = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC);
1188 }
1189
1190 if (_glfw.wl.seat && _glfw.wl.dataDeviceManager)
1191 {
1192 _glfw.wl.dataDevice =
1193 wl_data_device_manager_get_data_device(_glfw.wl.dataDeviceManager,
1194 _glfw.wl.seat);
1195 wl_data_device_add_listener(_glfw.wl.dataDevice, &dataDeviceListener, NULL);
1196 _glfw.wl.clipboardString = malloc(4096);
1197 if (!_glfw.wl.clipboardString)
1198 {
1200 "Wayland: Unable to allocate clipboard memory");
1201 return GLFW_FALSE;
1202 }
1203 _glfw.wl.clipboardSize = 4096;
1204 }
1205
1206 return GLFW_TRUE;
1207}
1208
1210{
1212 if (_glfw.wl.egl.handle)
1213 {
1215 _glfw.wl.egl.handle = NULL;
1216 }
1217
1218#ifdef HAVE_XKBCOMMON_COMPOSE_H
1219 if (_glfw.wl.xkb.composeState)
1220 xkb_compose_state_unref(_glfw.wl.xkb.composeState);
1221#endif
1222 if (_glfw.wl.xkb.keymap)
1223 xkb_keymap_unref(_glfw.wl.xkb.keymap);
1224 if (_glfw.wl.xkb.state)
1225 xkb_state_unref(_glfw.wl.xkb.state);
1226 if (_glfw.wl.xkb.context)
1228 if (_glfw.wl.xkb.handle)
1229 {
1230 _glfw_dlclose(_glfw.wl.xkb.handle);
1231 _glfw.wl.xkb.handle = NULL;
1232 }
1233
1234 if (_glfw.wl.cursorTheme)
1235 wl_cursor_theme_destroy(_glfw.wl.cursorTheme);
1236 if (_glfw.wl.cursorThemeHiDPI)
1237 wl_cursor_theme_destroy(_glfw.wl.cursorThemeHiDPI);
1238 if (_glfw.wl.cursor.handle)
1239 {
1240 _glfw_dlclose(_glfw.wl.cursor.handle);
1241 _glfw.wl.cursor.handle = NULL;
1242 }
1243
1244 if (_glfw.wl.cursorSurface)
1245 wl_surface_destroy(_glfw.wl.cursorSurface);
1246 if (_glfw.wl.subcompositor)
1247 wl_subcompositor_destroy(_glfw.wl.subcompositor);
1248 if (_glfw.wl.compositor)
1249 wl_compositor_destroy(_glfw.wl.compositor);
1250 if (_glfw.wl.shm)
1251 wl_shm_destroy(_glfw.wl.shm);
1252 if (_glfw.wl.viewporter)
1253 wp_viewporter_destroy(_glfw.wl.viewporter);
1254 if (_glfw.wl.decorationManager)
1255 zxdg_decoration_manager_v1_destroy(_glfw.wl.decorationManager);
1256 if (_glfw.wl.wmBase)
1257 xdg_wm_base_destroy(_glfw.wl.wmBase);
1258 if (_glfw.wl.dataSource)
1259 wl_data_source_destroy(_glfw.wl.dataSource);
1260 if (_glfw.wl.dataDevice)
1261 wl_data_device_destroy(_glfw.wl.dataDevice);
1262 if (_glfw.wl.dataOffer)
1263 wl_data_offer_destroy(_glfw.wl.dataOffer);
1264 if (_glfw.wl.dataDeviceManager)
1265 wl_data_device_manager_destroy(_glfw.wl.dataDeviceManager);
1266 if (_glfw.wl.pointer)
1267 wl_pointer_destroy(_glfw.wl.pointer);
1268 if (_glfw.wl.keyboard)
1269 wl_keyboard_destroy(_glfw.wl.keyboard);
1270 if (_glfw.wl.seat)
1271 wl_seat_destroy(_glfw.wl.seat);
1272 if (_glfw.wl.relativePointerManager)
1273 zwp_relative_pointer_manager_v1_destroy(_glfw.wl.relativePointerManager);
1274 if (_glfw.wl.pointerConstraints)
1275 zwp_pointer_constraints_v1_destroy(_glfw.wl.pointerConstraints);
1276 if (_glfw.wl.idleInhibitManager)
1277 zwp_idle_inhibit_manager_v1_destroy(_glfw.wl.idleInhibitManager);
1278 if (_glfw.wl.registry)
1279 wl_registry_destroy(_glfw.wl.registry);
1280 if (_glfw.wl.display)
1281 {
1282 wl_display_flush(_glfw.wl.display);
1283 wl_display_disconnect(_glfw.wl.display);
1284 }
1285
1286 if (_glfw.wl.timerfd >= 0)
1287 close(_glfw.wl.timerfd);
1288 if (_glfw.wl.cursorTimerfd >= 0)
1289 close(_glfw.wl.cursorTimerfd);
1290
1291 if (_glfw.wl.clipboardString)
1292 free(_glfw.wl.clipboardString);
1293 if (_glfw.wl.clipboardSendString)
1294 free(_glfw.wl.clipboardSendString);
1295}
1296
1298{
1299 return _GLFW_VERSION_NUMBER " Wayland EGL OSMesa"
1300#if defined(_POSIX_TIMERS) && defined(_POSIX_MONOTONIC_CLOCK)
1301 " clock_gettime"
1302#else
1303 " gettimeofday"
1304#endif
1305 " evdev"
1306#if defined(_GLFW_BUILD_DLL)
1307 " shared"
1308#endif
1309 ;
1310}
#define _glfw_dlopen(name)
void * id
#define _glfw_dlsym(handle, name)
#define _glfw_dlclose(handle)
void _glfwPlatformSetCursor(_GLFWwindow *window, _GLFWcursor *cursor)
void _glfwTerminateEGL(void)
Definition: egl_context.c:481
#define GLFW_CURSOR_DISABLED
Definition: glfw3.h:1106
#define GLFW_DISCONNECTED
Definition: glfw3.h:1231
#define GLFW_PLATFORM_ERROR
A platform-specific error occurred that does not match any of the more specific categories.
Definition: glfw3.h:758
#define GLFW_TRUE
One.
Definition: glfw3.h:310
#define GLFW_FALSE
Zero.
Definition: glfw3.h:319
#define GLFW_PRESS
The key or mouse button was pressed.
Definition: glfw3.h:336
#define GLFW_RELEASE
The key or mouse button was released.
Definition: glfw3.h:329
#define GLFW_KEY_C
Definition: glfw3.h:412
#define GLFW_KEY_D
Definition: glfw3.h:413
#define GLFW_KEY_A
Definition: glfw3.h:410
#define GLFW_KEY_1
Definition: glfw3.h:399
#define GLFW_KEY_RIGHT
Definition: glfw3.h:450
#define GLFW_KEY_F2
Definition: glfw3.h:464
#define GLFW_KEY_F11
Definition: glfw3.h:473
#define GLFW_KEY_KP_0
Definition: glfw3.h:488
#define GLFW_KEY_S
Definition: glfw3.h:428
#define GLFW_KEY_V
Definition: glfw3.h:431
#define GLFW_KEY_I
Definition: glfw3.h:418
#define GLFW_KEY_9
Definition: glfw3.h:407
#define GLFW_KEY_UP
Definition: glfw3.h:453
#define GLFW_KEY_F21
Definition: glfw3.h:483
#define GLFW_KEY_INSERT
Definition: glfw3.h:448
#define GLFW_KEY_PERIOD
Definition: glfw3.h:396
#define GLFW_KEY_NUM_LOCK
Definition: glfw3.h:460
#define GLFW_KEY_PAGE_UP
Definition: glfw3.h:454
#define GLFW_KEY_HOME
Definition: glfw3.h:456
#define GLFW_KEY_F23
Definition: glfw3.h:485
#define GLFW_KEY_K
Definition: glfw3.h:420
#define GLFW_KEY_R
Definition: glfw3.h:427
#define GLFW_KEY_5
Definition: glfw3.h:403
#define GLFW_KEY_M
Definition: glfw3.h:422
#define GLFW_KEY_KP_DECIMAL
Definition: glfw3.h:498
#define GLFW_KEY_KP_ENTER
Definition: glfw3.h:503
#define GLFW_KEY_0
Definition: glfw3.h:398
#define GLFW_KEY_F
Definition: glfw3.h:415
#define GLFW_KEY_APOSTROPHE
Definition: glfw3.h:393
#define GLFW_KEY_J
Definition: glfw3.h:419
#define GLFW_KEY_RIGHT_ALT
Definition: glfw3.h:511
#define GLFW_KEY_TAB
Definition: glfw3.h:446
#define GLFW_KEY_BACKSPACE
Definition: glfw3.h:447
#define GLFW_KEY_F6
Definition: glfw3.h:468
#define GLFW_KEY_F10
Definition: glfw3.h:472
#define GLFW_KEY_GRAVE_ACCENT
Definition: glfw3.h:439
#define GLFW_KEY_KP_3
Definition: glfw3.h:491
#define GLFW_KEY_LEFT_ALT
Definition: glfw3.h:507
#define GLFW_KEY_F15
Definition: glfw3.h:477
#define GLFW_KEY_PAUSE
Definition: glfw3.h:462
#define GLFW_KEY_3
Definition: glfw3.h:401
#define GLFW_KEY_F24
Definition: glfw3.h:486
#define GLFW_KEY_F20
Definition: glfw3.h:482
#define GLFW_KEY_SEMICOLON
Definition: glfw3.h:408
#define GLFW_KEY_END
Definition: glfw3.h:457
#define GLFW_KEY_RIGHT_BRACKET
Definition: glfw3.h:438
#define GLFW_KEY_KP_7
Definition: glfw3.h:495
#define GLFW_KEY_LEFT_SHIFT
Definition: glfw3.h:505
#define GLFW_KEY_B
Definition: glfw3.h:411
#define GLFW_KEY_P
Definition: glfw3.h:425
#define GLFW_KEY_F22
Definition: glfw3.h:484
#define GLFW_KEY_T
Definition: glfw3.h:429
#define GLFW_KEY_CAPS_LOCK
Definition: glfw3.h:458
#define GLFW_KEY_ENTER
Definition: glfw3.h:445
#define GLFW_KEY_F17
Definition: glfw3.h:479
#define GLFW_KEY_MENU
Definition: glfw3.h:513
#define GLFW_KEY_UNKNOWN
Definition: glfw3.h:389
#define GLFW_KEY_KP_5
Definition: glfw3.h:493
#define GLFW_KEY_KP_MULTIPLY
Definition: glfw3.h:500
#define GLFW_KEY_F4
Definition: glfw3.h:466
#define GLFW_KEY_4
Definition: glfw3.h:402
#define GLFW_KEY_6
Definition: glfw3.h:404
#define GLFW_KEY_LEFT_CONTROL
Definition: glfw3.h:506
#define GLFW_KEY_W
Definition: glfw3.h:432
#define GLFW_KEY_KP_SUBTRACT
Definition: glfw3.h:501
#define GLFW_KEY_F16
Definition: glfw3.h:478
#define GLFW_KEY_L
Definition: glfw3.h:421
#define GLFW_KEY_ESCAPE
Definition: glfw3.h:444
#define GLFW_KEY_KP_8
Definition: glfw3.h:496
#define GLFW_KEY_COMMA
Definition: glfw3.h:394
#define GLFW_KEY_7
Definition: glfw3.h:405
#define GLFW_KEY_BACKSLASH
Definition: glfw3.h:437
#define GLFW_KEY_KP_DIVIDE
Definition: glfw3.h:499
#define GLFW_KEY_E
Definition: glfw3.h:414
#define GLFW_KEY_X
Definition: glfw3.h:433
#define GLFW_KEY_Z
Definition: glfw3.h:435
#define GLFW_KEY_MINUS
Definition: glfw3.h:395
#define GLFW_KEY_F8
Definition: glfw3.h:470
#define GLFW_KEY_U
Definition: glfw3.h:430
#define GLFW_KEY_F7
Definition: glfw3.h:469
#define GLFW_KEY_KP_ADD
Definition: glfw3.h:502
#define GLFW_KEY_LEFT_BRACKET
Definition: glfw3.h:436
#define GLFW_KEY_RIGHT_CONTROL
Definition: glfw3.h:510
#define GLFW_KEY_RIGHT_SUPER
Definition: glfw3.h:512
#define GLFW_KEY_H
Definition: glfw3.h:417
#define GLFW_KEY_F13
Definition: glfw3.h:475
#define GLFW_KEY_KP_4
Definition: glfw3.h:492
#define GLFW_KEY_DELETE
Definition: glfw3.h:449
#define GLFW_KEY_2
Definition: glfw3.h:400
#define GLFW_KEY_SPACE
Definition: glfw3.h:392
#define GLFW_KEY_8
Definition: glfw3.h:406
#define GLFW_KEY_SLASH
Definition: glfw3.h:397
#define GLFW_KEY_N
Definition: glfw3.h:423
#define GLFW_KEY_LEFT
Definition: glfw3.h:451
#define GLFW_KEY_EQUAL
Definition: glfw3.h:409
#define GLFW_KEY_DOWN
Definition: glfw3.h:452
#define GLFW_KEY_F9
Definition: glfw3.h:471
#define GLFW_KEY_G
Definition: glfw3.h:416
#define GLFW_KEY_KP_EQUAL
Definition: glfw3.h:504
#define GLFW_KEY_F18
Definition: glfw3.h:480
#define GLFW_KEY_F19
Definition: glfw3.h:481
#define GLFW_KEY_O
Definition: glfw3.h:424
#define GLFW_KEY_F3
Definition: glfw3.h:465
#define GLFW_KEY_PAGE_DOWN
Definition: glfw3.h:455
#define GLFW_KEY_F14
Definition: glfw3.h:476
#define GLFW_KEY_F5
Definition: glfw3.h:467
#define GLFW_KEY_KP_1
Definition: glfw3.h:489
#define GLFW_KEY_F12
Definition: glfw3.h:474
#define GLFW_KEY_SCROLL_LOCK
Definition: glfw3.h:459
#define GLFW_KEY_KP_2
Definition: glfw3.h:490
#define GLFW_KEY_PRINT_SCREEN
Definition: glfw3.h:461
#define GLFW_KEY_LEFT_SUPER
Definition: glfw3.h:508
#define GLFW_KEY_KP_9
Definition: glfw3.h:497
#define GLFW_KEY_F1
Definition: glfw3.h:463
#define GLFW_KEY_KP_6
Definition: glfw3.h:494
#define GLFW_KEY_Y
Definition: glfw3.h:434
#define GLFW_KEY_Q
Definition: glfw3.h:426
#define GLFW_KEY_RIGHT_SHIFT
Definition: glfw3.h:509
#define GLFW_MOD_SHIFT
If this bit is set one or more Shift keys were held down.
Definition: glfw3.h:531
#define GLFW_MOD_NUM_LOCK
If this bit is set the Num Lock key is enabled.
Definition: glfw3.h:558
#define GLFW_MOD_SUPER
If this bit is set one or more Super keys were held down.
Definition: glfw3.h:546
#define GLFW_MOD_CONTROL
If this bit is set one or more Control keys were held down.
Definition: glfw3.h:536
#define GLFW_MOD_ALT
If this bit is set one or more Alt keys were held down.
Definition: glfw3.h:541
#define GLFW_MOD_CAPS_LOCK
If this bit is set the Caps Lock key is enabled.
Definition: glfw3.h:552
_GLFWlibrary _glfw
Definition: init.c:46
void _glfwInputError(int code, const char *format,...)
Definition: init.c:160
void _glfwInputCursorEnter(_GLFWwindow *window, GLFWbool entered)
Definition: input.c:370
void _glfwInputScroll(_GLFWwindow *window, double xoffset, double yoffset)
Definition: input.c:328
void _glfwInputChar(_GLFWwindow *window, unsigned int codepoint, int mods, GLFWbool plain)
Definition: input.c:308
void _glfwInputCursorPos(_GLFWwindow *window, double xpos, double ypos)
Definition: input.c:356
void _glfwInputKey(_GLFWwindow *window, int key, int scancode, int action, int mods)
Definition: input.c:277
void _glfwInputMouseClick(_GLFWwindow *window, int button, int action, int mods)
Definition: input.c:336
void _glfwInputMonitor(_GLFWmonitor *monitor, int action, int placement)
Definition: monitor.c:97
#define _GLFW_VERSION_NUMBER
Definition: internal.h:203
int GLFWbool
Definition: internal.h:61
void _glfwInputWindowFocus(_GLFWwindow *window, GLFWbool focused)
Definition: window.c:45
#define NULL
Definition: miniaudio.h:3718
void _glfwInitTimerPOSIX(void)
Definition: posix_time.c:45
@ KEY_F4
Definition: raylib.h:598
@ KEY_SPACE
Definition: raylib.h:575
@ KEY_MENU
Definition: raylib.h:636
@ KEY_UP
Definition: raylib.h:585
@ KEY_P
Definition: raylib.h:559
@ KEY_M
Definition: raylib.h:556
@ KEY_B
Definition: raylib.h:545
@ KEY_F9
Definition: raylib.h:603
@ KEY_F6
Definition: raylib.h:600
@ KEY_INSERT
Definition: raylib.h:580
@ KEY_SEMICOLON
Definition: raylib.h:542
@ KEY_ENTER
Definition: raylib.h:577
@ KEY_E
Definition: raylib.h:548
@ KEY_I
Definition: raylib.h:552
@ KEY_W
Definition: raylib.h:566
@ KEY_D
Definition: raylib.h:547
@ KEY_F7
Definition: raylib.h:601
@ KEY_GRAVE
Definition: raylib.h:573
@ KEY_F10
Definition: raylib.h:604
@ KEY_F
Definition: raylib.h:549
@ KEY_H
Definition: raylib.h:551
@ KEY_X
Definition: raylib.h:567
@ KEY_G
Definition: raylib.h:550
@ KEY_SLASH
Definition: raylib.h:531
@ KEY_Y
Definition: raylib.h:568
@ KEY_MINUS
Definition: raylib.h:529
@ KEY_TAB
Definition: raylib.h:578
@ KEY_LEFT
Definition: raylib.h:583
@ KEY_T
Definition: raylib.h:563
@ KEY_RIGHT
Definition: raylib.h:582
@ KEY_F5
Definition: raylib.h:599
@ KEY_K
Definition: raylib.h:554
@ KEY_F12
Definition: raylib.h:606
@ KEY_F2
Definition: raylib.h:596
@ KEY_F3
Definition: raylib.h:597
@ KEY_C
Definition: raylib.h:546
@ KEY_F8
Definition: raylib.h:602
@ KEY_BACKSPACE
Definition: raylib.h:579
@ KEY_R
Definition: raylib.h:561
@ KEY_N
Definition: raylib.h:557
@ KEY_A
Definition: raylib.h:544
@ KEY_END
Definition: raylib.h:589
@ KEY_DOWN
Definition: raylib.h:584
@ KEY_Z
Definition: raylib.h:569
@ KEY_S
Definition: raylib.h:562
@ KEY_J
Definition: raylib.h:553
@ KEY_Q
Definition: raylib.h:560
@ KEY_EQUAL
Definition: raylib.h:543
@ KEY_PAUSE
Definition: raylib.h:594
@ KEY_U
Definition: raylib.h:564
@ KEY_F11
Definition: raylib.h:605
@ KEY_HOME
Definition: raylib.h:588
@ KEY_DELETE
Definition: raylib.h:581
@ KEY_O
Definition: raylib.h:558
@ KEY_APOSTROPHE
Definition: raylib.h:527
@ KEY_L
Definition: raylib.h:555
@ KEY_BACKSLASH
Definition: raylib.h:571
@ KEY_COMMA
Definition: raylib.h:528
@ KEY_V
Definition: raylib.h:565
@ KEY_F1
Definition: raylib.h:595
unsigned int uint32_t
Definition: stdint.h:80
signed int int32_t
Definition: stdint.h:77
_GLFWlibraryEGL egl
Definition: internal.h:595
void * handle
Definition: internal.h:563
_GLFWctxconfig context
Definition: internal.h:535
_GLFWwindow * windowListHead
Definition: internal.h:541
_GLFWmonitor ** monitors
Definition: internal.h:543
int monitorCount
Definition: internal.h:544
char name[128]
Definition: internal.h:439
int cursorMode
Definition: internal.h:402
struct _GLFWwindow * next
Definition: internal.h:379
GLFWwindowcontentscalefun scale
Definition: internal.h:420
GLFWwindowfocusfun focus
Definition: internal.h:416
uint32_t hotspot_y
Definition: wl_platform.h:80
uint32_t hotspot_x
Definition: wl_platform.h:79
uint32_t height
Definition: wl_platform.h:78
uint32_t width
Definition: wl_platform.h:77
struct wl_cursor_image ** images
Definition: wl_platform.h:85
const char * _glfwPlatformGetVersionString(void)
Definition: wl_init.c:1297
void _glfwPlatformTerminate(void)
Definition: wl_init.c:1209
int _glfwPlatformInit(void)
Definition: wl_init.c:1034
void _glfwAddOutputWayland(uint32_t name, uint32_t version)
Definition: wl_monitor.c:121
void(* PFN_wl_cursor_theme_destroy)(struct wl_cursor_theme *)
Definition: wl_platform.h:89
void(* PFN_xkb_state_unref)(struct xkb_state *)
Definition: wl_platform.h:111
int(* PFN_xkb_keymap_key_repeats)(struct xkb_keymap *, xkb_keycode_t)
Definition: wl_platform.h:109
struct xkb_keymap *(* PFN_xkb_keymap_new_from_string)(struct xkb_context *, const char *, enum xkb_keymap_format, enum xkb_keymap_compile_flags)
Definition: wl_platform.h:106
struct xkb_state *(* PFN_xkb_state_new)(struct xkb_keymap *)
Definition: wl_platform.h:110
void(* PFN_wl_egl_window_destroy)(struct wl_egl_window *)
Definition: wl_platform.h:98
#define xkb_state_new
Definition: wl_platform.h:121
struct wl_buffer *(* PFN_wl_cursor_image_get_buffer)(struct wl_cursor_image *)
Definition: wl_platform.h:91
#define wl_cursor_image_get_buffer
Definition: wl_platform.h:95
xkb_mod_index_t(* PFN_xkb_keymap_mod_get_index)(struct xkb_keymap *, const char *)
Definition: wl_platform.h:108
#define xkb_keymap_key_repeats
Definition: wl_platform.h:120
#define xkb_keymap_new_from_string
Definition: wl_platform.h:117
#define xkb_context_unref
Definition: wl_platform.h:116
enum xkb_state_component(* PFN_xkb_state_update_mask)(struct xkb_state *, xkb_mod_mask_t, xkb_mod_mask_t, xkb_mod_mask_t, xkb_layout_index_t, xkb_layout_index_t, xkb_layout_index_t)
Definition: wl_platform.h:113
#define xkb_state_unref
Definition: wl_platform.h:122
xkb_mod_mask_t(* PFN_xkb_state_serialize_mods)(struct xkb_state *, enum xkb_state_component)
Definition: wl_platform.h:114
#define xkb_keymap_unref
Definition: wl_platform.h:118
struct wl_cursor_theme *(* PFN_wl_cursor_theme_load)(const char *, int, struct wl_shm *)
Definition: wl_platform.h:88
#define wl_cursor_theme_get_cursor
Definition: wl_platform.h:94
void(* PFN_xkb_keymap_unref)(struct xkb_keymap *)
Definition: wl_platform.h:107
struct xkb_context *(* PFN_xkb_context_new)(enum xkb_context_flags)
Definition: wl_platform.h:104
#define xkb_state_key_get_syms
Definition: wl_platform.h:123
#define xkb_state_serialize_mods
Definition: wl_platform.h:125
#define _GLFW_DECORATION_WIDTH
Definition: wl_platform.h:144
int(* PFN_xkb_state_key_get_syms)(struct xkb_state *, xkb_keycode_t, const xkb_keysym_t **)
Definition: wl_platform.h:112
struct wl_cursor *(* PFN_wl_cursor_theme_get_cursor)(struct wl_cursor_theme *, const char *)
Definition: wl_platform.h:90
#define xkb_context_new
Definition: wl_platform.h:115
@ bottomDecoration
Definition: wl_platform.h:155
@ topDecoration
Definition: wl_platform.h:152
@ leftDecoration
Definition: wl_platform.h:153
@ rightDecoration
Definition: wl_platform.h:154
@ mainWindow
Definition: wl_platform.h:151
struct wl_egl_window *(* PFN_wl_egl_window_create)(struct wl_surface *, int, int)
Definition: wl_platform.h:97
void(* PFN_wl_egl_window_resize)(struct wl_egl_window *, int, int, int, int)
Definition: wl_platform.h:99
#define xkb_keymap_mod_get_index
Definition: wl_platform.h:119
void(* PFN_xkb_context_unref)(struct xkb_context *)
Definition: wl_platform.h:105
#define wl_cursor_theme_load
Definition: wl_platform.h:92
#define wl_cursor_theme_destroy
Definition: wl_platform.h:93
#define xkb_state_update_mask
Definition: wl_platform.h:124
long _glfwKeySym2Unicode(unsigned int keysym)
Definition: xkb_unicode.c:910