GCC Code Coverage Report


Directory: ./
File: submodules/raylib/src/external/glfw/src/x11_init.c
Date: 2023-09-29 04:53:15
Exec Total Coverage
Lines: 0 801 0.0%
Branches: 0 389 0.0%

Line Branch Exec Source
1 //========================================================================
2 // GLFW 3.4 X11 - www.glfw.org
3 //------------------------------------------------------------------------
4 // Copyright (c) 2002-2006 Marcus Geelnard
5 // Copyright (c) 2006-2019 Camilla Löwy <elmindreda@glfw.org>
6 //
7 // This software is provided 'as-is', without any express or implied
8 // warranty. In no event will the authors be held liable for any damages
9 // arising from the use of this software.
10 //
11 // Permission is granted to anyone to use this software for any purpose,
12 // including commercial applications, and to alter it and redistribute it
13 // freely, subject to the following restrictions:
14 //
15 // 1. The origin of this software must not be misrepresented; you must not
16 // claim that you wrote the original software. If you use this software
17 // in a product, an acknowledgment in the product documentation would
18 // be appreciated but is not required.
19 //
20 // 2. Altered source versions must be plainly marked as such, and must not
21 // be misrepresented as being the original software.
22 //
23 // 3. This notice may not be removed or altered from any source
24 // distribution.
25 //
26 //========================================================================
27 // It is fine to use C99 in this file because it will not be built with VS
28 //========================================================================
29
30 #include "internal.h"
31
32 #include <stdlib.h>
33 #include <string.h>
34 #include <limits.h>
35 #include <stdio.h>
36 #include <locale.h>
37 #include <unistd.h>
38 #include <fcntl.h>
39 #include <errno.h>
40 #include <assert.h>
41
42
43 // Translate the X11 KeySyms for a key to a GLFW key code
44 // NOTE: This is only used as a fallback, in case the XKB method fails
45 // It is layout-dependent and will fail partially on most non-US layouts
46 //
47 static int translateKeySyms(const KeySym* keysyms, int width)
48 {
49 if (width > 1)
50 {
51 switch (keysyms[1])
52 {
53 case XK_KP_0: return GLFW_KEY_KP_0;
54 case XK_KP_1: return GLFW_KEY_KP_1;
55 case XK_KP_2: return GLFW_KEY_KP_2;
56 case XK_KP_3: return GLFW_KEY_KP_3;
57 case XK_KP_4: return GLFW_KEY_KP_4;
58 case XK_KP_5: return GLFW_KEY_KP_5;
59 case XK_KP_6: return GLFW_KEY_KP_6;
60 case XK_KP_7: return GLFW_KEY_KP_7;
61 case XK_KP_8: return GLFW_KEY_KP_8;
62 case XK_KP_9: return GLFW_KEY_KP_9;
63 case XK_KP_Separator:
64 case XK_KP_Decimal: return GLFW_KEY_KP_DECIMAL;
65 case XK_KP_Equal: return GLFW_KEY_KP_EQUAL;
66 case XK_KP_Enter: return GLFW_KEY_KP_ENTER;
67 default: break;
68 }
69 }
70
71 switch (keysyms[0])
72 {
73 case XK_Escape: return GLFW_KEY_ESCAPE;
74 case XK_Tab: return GLFW_KEY_TAB;
75 case XK_Shift_L: return GLFW_KEY_LEFT_SHIFT;
76 case XK_Shift_R: return GLFW_KEY_RIGHT_SHIFT;
77 case XK_Control_L: return GLFW_KEY_LEFT_CONTROL;
78 case XK_Control_R: return GLFW_KEY_RIGHT_CONTROL;
79 case XK_Meta_L:
80 case XK_Alt_L: return GLFW_KEY_LEFT_ALT;
81 case XK_Mode_switch: // Mapped to Alt_R on many keyboards
82 case XK_ISO_Level3_Shift: // AltGr on at least some machines
83 case XK_Meta_R:
84 case XK_Alt_R: return GLFW_KEY_RIGHT_ALT;
85 case XK_Super_L: return GLFW_KEY_LEFT_SUPER;
86 case XK_Super_R: return GLFW_KEY_RIGHT_SUPER;
87 case XK_Menu: return GLFW_KEY_MENU;
88 case XK_Num_Lock: return GLFW_KEY_NUM_LOCK;
89 case XK_Caps_Lock: return GLFW_KEY_CAPS_LOCK;
90 case XK_Print: return GLFW_KEY_PRINT_SCREEN;
91 case XK_Scroll_Lock: return GLFW_KEY_SCROLL_LOCK;
92 case XK_Pause: return GLFW_KEY_PAUSE;
93 case XK_Delete: return GLFW_KEY_DELETE;
94 case XK_BackSpace: return GLFW_KEY_BACKSPACE;
95 case XK_Return: return GLFW_KEY_ENTER;
96 case XK_Home: return GLFW_KEY_HOME;
97 case XK_End: return GLFW_KEY_END;
98 case XK_Page_Up: return GLFW_KEY_PAGE_UP;
99 case XK_Page_Down: return GLFW_KEY_PAGE_DOWN;
100 case XK_Insert: return GLFW_KEY_INSERT;
101 case XK_Left: return GLFW_KEY_LEFT;
102 case XK_Right: return GLFW_KEY_RIGHT;
103 case XK_Down: return GLFW_KEY_DOWN;
104 case XK_Up: return GLFW_KEY_UP;
105 case XK_F1: return GLFW_KEY_F1;
106 case XK_F2: return GLFW_KEY_F2;
107 case XK_F3: return GLFW_KEY_F3;
108 case XK_F4: return GLFW_KEY_F4;
109 case XK_F5: return GLFW_KEY_F5;
110 case XK_F6: return GLFW_KEY_F6;
111 case XK_F7: return GLFW_KEY_F7;
112 case XK_F8: return GLFW_KEY_F8;
113 case XK_F9: return GLFW_KEY_F9;
114 case XK_F10: return GLFW_KEY_F10;
115 case XK_F11: return GLFW_KEY_F11;
116 case XK_F12: return GLFW_KEY_F12;
117 case XK_F13: return GLFW_KEY_F13;
118 case XK_F14: return GLFW_KEY_F14;
119 case XK_F15: return GLFW_KEY_F15;
120 case XK_F16: return GLFW_KEY_F16;
121 case XK_F17: return GLFW_KEY_F17;
122 case XK_F18: return GLFW_KEY_F18;
123 case XK_F19: return GLFW_KEY_F19;
124 case XK_F20: return GLFW_KEY_F20;
125 case XK_F21: return GLFW_KEY_F21;
126 case XK_F22: return GLFW_KEY_F22;
127 case XK_F23: return GLFW_KEY_F23;
128 case XK_F24: return GLFW_KEY_F24;
129 case XK_F25: return GLFW_KEY_F25;
130
131 // Numeric keypad
132 case XK_KP_Divide: return GLFW_KEY_KP_DIVIDE;
133 case XK_KP_Multiply: return GLFW_KEY_KP_MULTIPLY;
134 case XK_KP_Subtract: return GLFW_KEY_KP_SUBTRACT;
135 case XK_KP_Add: return GLFW_KEY_KP_ADD;
136
137 // These should have been detected in secondary keysym test above!
138 case XK_KP_Insert: return GLFW_KEY_KP_0;
139 case XK_KP_End: return GLFW_KEY_KP_1;
140 case XK_KP_Down: return GLFW_KEY_KP_2;
141 case XK_KP_Page_Down: return GLFW_KEY_KP_3;
142 case XK_KP_Left: return GLFW_KEY_KP_4;
143 case XK_KP_Right: return GLFW_KEY_KP_6;
144 case XK_KP_Home: return GLFW_KEY_KP_7;
145 case XK_KP_Up: return GLFW_KEY_KP_8;
146 case XK_KP_Page_Up: return GLFW_KEY_KP_9;
147 case XK_KP_Delete: return GLFW_KEY_KP_DECIMAL;
148 case XK_KP_Equal: return GLFW_KEY_KP_EQUAL;
149 case XK_KP_Enter: return GLFW_KEY_KP_ENTER;
150
151 // Last resort: Check for printable keys (should not happen if the XKB
152 // extension is available). This will give a layout dependent mapping
153 // (which is wrong, and we may miss some keys, especially on non-US
154 // keyboards), but it's better than nothing...
155 case XK_a: return GLFW_KEY_A;
156 case XK_b: return GLFW_KEY_B;
157 case XK_c: return GLFW_KEY_C;
158 case XK_d: return GLFW_KEY_D;
159 case XK_e: return GLFW_KEY_E;
160 case XK_f: return GLFW_KEY_F;
161 case XK_g: return GLFW_KEY_G;
162 case XK_h: return GLFW_KEY_H;
163 case XK_i: return GLFW_KEY_I;
164 case XK_j: return GLFW_KEY_J;
165 case XK_k: return GLFW_KEY_K;
166 case XK_l: return GLFW_KEY_L;
167 case XK_m: return GLFW_KEY_M;
168 case XK_n: return GLFW_KEY_N;
169 case XK_o: return GLFW_KEY_O;
170 case XK_p: return GLFW_KEY_P;
171 case XK_q: return GLFW_KEY_Q;
172 case XK_r: return GLFW_KEY_R;
173 case XK_s: return GLFW_KEY_S;
174 case XK_t: return GLFW_KEY_T;
175 case XK_u: return GLFW_KEY_U;
176 case XK_v: return GLFW_KEY_V;
177 case XK_w: return GLFW_KEY_W;
178 case XK_x: return GLFW_KEY_X;
179 case XK_y: return GLFW_KEY_Y;
180 case XK_z: return GLFW_KEY_Z;
181 case XK_1: return GLFW_KEY_1;
182 case XK_2: return GLFW_KEY_2;
183 case XK_3: return GLFW_KEY_3;
184 case XK_4: return GLFW_KEY_4;
185 case XK_5: return GLFW_KEY_5;
186 case XK_6: return GLFW_KEY_6;
187 case XK_7: return GLFW_KEY_7;
188 case XK_8: return GLFW_KEY_8;
189 case XK_9: return GLFW_KEY_9;
190 case XK_0: return GLFW_KEY_0;
191 case XK_space: return GLFW_KEY_SPACE;
192 case XK_minus: return GLFW_KEY_MINUS;
193 case XK_equal: return GLFW_KEY_EQUAL;
194 case XK_bracketleft: return GLFW_KEY_LEFT_BRACKET;
195 case XK_bracketright: return GLFW_KEY_RIGHT_BRACKET;
196 case XK_backslash: return GLFW_KEY_BACKSLASH;
197 case XK_semicolon: return GLFW_KEY_SEMICOLON;
198 case XK_apostrophe: return GLFW_KEY_APOSTROPHE;
199 case XK_grave: return GLFW_KEY_GRAVE_ACCENT;
200 case XK_comma: return GLFW_KEY_COMMA;
201 case XK_period: return GLFW_KEY_PERIOD;
202 case XK_slash: return GLFW_KEY_SLASH;
203 case XK_less: return GLFW_KEY_WORLD_1; // At least in some layouts...
204 default: break;
205 }
206
207 // No matching translation was found
208 return GLFW_KEY_UNKNOWN;
209 }
210
211 // Create key code translation tables
212 //
213 static void createKeyTables(void)
214 {
215 int scancodeMin, scancodeMax;
216
217 memset(_glfw.x11.keycodes, -1, sizeof(_glfw.x11.keycodes));
218 memset(_glfw.x11.scancodes, -1, sizeof(_glfw.x11.scancodes));
219
220 if (_glfw.x11.xkb.available)
221 {
222 // Use XKB to determine physical key locations independently of the
223 // current keyboard layout
224
225 XkbDescPtr desc = XkbGetMap(_glfw.x11.display, 0, XkbUseCoreKbd);
226 XkbGetNames(_glfw.x11.display, XkbKeyNamesMask | XkbKeyAliasesMask, desc);
227
228 scancodeMin = desc->min_key_code;
229 scancodeMax = desc->max_key_code;
230
231 const struct
232 {
233 int key;
234 char* name;
235 } keymap[] =
236 {
237 { GLFW_KEY_GRAVE_ACCENT, "TLDE" },
238 { GLFW_KEY_1, "AE01" },
239 { GLFW_KEY_2, "AE02" },
240 { GLFW_KEY_3, "AE03" },
241 { GLFW_KEY_4, "AE04" },
242 { GLFW_KEY_5, "AE05" },
243 { GLFW_KEY_6, "AE06" },
244 { GLFW_KEY_7, "AE07" },
245 { GLFW_KEY_8, "AE08" },
246 { GLFW_KEY_9, "AE09" },
247 { GLFW_KEY_0, "AE10" },
248 { GLFW_KEY_MINUS, "AE11" },
249 { GLFW_KEY_EQUAL, "AE12" },
250 { GLFW_KEY_Q, "AD01" },
251 { GLFW_KEY_W, "AD02" },
252 { GLFW_KEY_E, "AD03" },
253 { GLFW_KEY_R, "AD04" },
254 { GLFW_KEY_T, "AD05" },
255 { GLFW_KEY_Y, "AD06" },
256 { GLFW_KEY_U, "AD07" },
257 { GLFW_KEY_I, "AD08" },
258 { GLFW_KEY_O, "AD09" },
259 { GLFW_KEY_P, "AD10" },
260 { GLFW_KEY_LEFT_BRACKET, "AD11" },
261 { GLFW_KEY_RIGHT_BRACKET, "AD12" },
262 { GLFW_KEY_A, "AC01" },
263 { GLFW_KEY_S, "AC02" },
264 { GLFW_KEY_D, "AC03" },
265 { GLFW_KEY_F, "AC04" },
266 { GLFW_KEY_G, "AC05" },
267 { GLFW_KEY_H, "AC06" },
268 { GLFW_KEY_J, "AC07" },
269 { GLFW_KEY_K, "AC08" },
270 { GLFW_KEY_L, "AC09" },
271 { GLFW_KEY_SEMICOLON, "AC10" },
272 { GLFW_KEY_APOSTROPHE, "AC11" },
273 { GLFW_KEY_Z, "AB01" },
274 { GLFW_KEY_X, "AB02" },
275 { GLFW_KEY_C, "AB03" },
276 { GLFW_KEY_V, "AB04" },
277 { GLFW_KEY_B, "AB05" },
278 { GLFW_KEY_N, "AB06" },
279 { GLFW_KEY_M, "AB07" },
280 { GLFW_KEY_COMMA, "AB08" },
281 { GLFW_KEY_PERIOD, "AB09" },
282 { GLFW_KEY_SLASH, "AB10" },
283 { GLFW_KEY_BACKSLASH, "BKSL" },
284 { GLFW_KEY_WORLD_1, "LSGT" },
285 { GLFW_KEY_SPACE, "SPCE" },
286 { GLFW_KEY_ESCAPE, "ESC" },
287 { GLFW_KEY_ENTER, "RTRN" },
288 { GLFW_KEY_TAB, "TAB" },
289 { GLFW_KEY_BACKSPACE, "BKSP" },
290 { GLFW_KEY_INSERT, "INS" },
291 { GLFW_KEY_DELETE, "DELE" },
292 { GLFW_KEY_RIGHT, "RGHT" },
293 { GLFW_KEY_LEFT, "LEFT" },
294 { GLFW_KEY_DOWN, "DOWN" },
295 { GLFW_KEY_UP, "UP" },
296 { GLFW_KEY_PAGE_UP, "PGUP" },
297 { GLFW_KEY_PAGE_DOWN, "PGDN" },
298 { GLFW_KEY_HOME, "HOME" },
299 { GLFW_KEY_END, "END" },
300 { GLFW_KEY_CAPS_LOCK, "CAPS" },
301 { GLFW_KEY_SCROLL_LOCK, "SCLK" },
302 { GLFW_KEY_NUM_LOCK, "NMLK" },
303 { GLFW_KEY_PRINT_SCREEN, "PRSC" },
304 { GLFW_KEY_PAUSE, "PAUS" },
305 { GLFW_KEY_F1, "FK01" },
306 { GLFW_KEY_F2, "FK02" },
307 { GLFW_KEY_F3, "FK03" },
308 { GLFW_KEY_F4, "FK04" },
309 { GLFW_KEY_F5, "FK05" },
310 { GLFW_KEY_F6, "FK06" },
311 { GLFW_KEY_F7, "FK07" },
312 { GLFW_KEY_F8, "FK08" },
313 { GLFW_KEY_F9, "FK09" },
314 { GLFW_KEY_F10, "FK10" },
315 { GLFW_KEY_F11, "FK11" },
316 { GLFW_KEY_F12, "FK12" },
317 { GLFW_KEY_F13, "FK13" },
318 { GLFW_KEY_F14, "FK14" },
319 { GLFW_KEY_F15, "FK15" },
320 { GLFW_KEY_F16, "FK16" },
321 { GLFW_KEY_F17, "FK17" },
322 { GLFW_KEY_F18, "FK18" },
323 { GLFW_KEY_F19, "FK19" },
324 { GLFW_KEY_F20, "FK20" },
325 { GLFW_KEY_F21, "FK21" },
326 { GLFW_KEY_F22, "FK22" },
327 { GLFW_KEY_F23, "FK23" },
328 { GLFW_KEY_F24, "FK24" },
329 { GLFW_KEY_F25, "FK25" },
330 { GLFW_KEY_KP_0, "KP0" },
331 { GLFW_KEY_KP_1, "KP1" },
332 { GLFW_KEY_KP_2, "KP2" },
333 { GLFW_KEY_KP_3, "KP3" },
334 { GLFW_KEY_KP_4, "KP4" },
335 { GLFW_KEY_KP_5, "KP5" },
336 { GLFW_KEY_KP_6, "KP6" },
337 { GLFW_KEY_KP_7, "KP7" },
338 { GLFW_KEY_KP_8, "KP8" },
339 { GLFW_KEY_KP_9, "KP9" },
340 { GLFW_KEY_KP_DECIMAL, "KPDL" },
341 { GLFW_KEY_KP_DIVIDE, "KPDV" },
342 { GLFW_KEY_KP_MULTIPLY, "KPMU" },
343 { GLFW_KEY_KP_SUBTRACT, "KPSU" },
344 { GLFW_KEY_KP_ADD, "KPAD" },
345 { GLFW_KEY_KP_ENTER, "KPEN" },
346 { GLFW_KEY_KP_EQUAL, "KPEQ" },
347 { GLFW_KEY_LEFT_SHIFT, "LFSH" },
348 { GLFW_KEY_LEFT_CONTROL, "LCTL" },
349 { GLFW_KEY_LEFT_ALT, "LALT" },
350 { GLFW_KEY_LEFT_SUPER, "LWIN" },
351 { GLFW_KEY_RIGHT_SHIFT, "RTSH" },
352 { GLFW_KEY_RIGHT_CONTROL, "RCTL" },
353 { GLFW_KEY_RIGHT_ALT, "RALT" },
354 { GLFW_KEY_RIGHT_ALT, "LVL3" },
355 { GLFW_KEY_RIGHT_ALT, "MDSW" },
356 { GLFW_KEY_RIGHT_SUPER, "RWIN" },
357 { GLFW_KEY_MENU, "MENU" }
358 };
359
360 // Find the X11 key code -> GLFW key code mapping
361 for (int scancode = scancodeMin; scancode <= scancodeMax; scancode++)
362 {
363 int key = GLFW_KEY_UNKNOWN;
364
365 // Map the key name to a GLFW key code. Note: We use the US
366 // keyboard layout. Because function keys aren't mapped correctly
367 // when using traditional KeySym translations, they are mapped
368 // here instead.
369 for (int i = 0; i < sizeof(keymap) / sizeof(keymap[0]); i++)
370 {
371 if (strncmp(desc->names->keys[scancode].name,
372 keymap[i].name,
373 XkbKeyNameLength) == 0)
374 {
375 key = keymap[i].key;
376 break;
377 }
378 }
379
380 // Fall back to key aliases in case the key name did not match
381 for (int i = 0; i < desc->names->num_key_aliases; i++)
382 {
383 if (key != GLFW_KEY_UNKNOWN)
384 break;
385
386 if (strncmp(desc->names->key_aliases[i].real,
387 desc->names->keys[scancode].name,
388 XkbKeyNameLength) != 0)
389 {
390 continue;
391 }
392
393 for (int j = 0; j < sizeof(keymap) / sizeof(keymap[0]); j++)
394 {
395 if (strncmp(desc->names->key_aliases[i].alias,
396 keymap[j].name,
397 XkbKeyNameLength) == 0)
398 {
399 key = keymap[j].key;
400 break;
401 }
402 }
403 }
404
405 _glfw.x11.keycodes[scancode] = key;
406 }
407
408 XkbFreeNames(desc, XkbKeyNamesMask, True);
409 XkbFreeKeyboard(desc, 0, True);
410 }
411 else
412 XDisplayKeycodes(_glfw.x11.display, &scancodeMin, &scancodeMax);
413
414 int width;
415 KeySym* keysyms = XGetKeyboardMapping(_glfw.x11.display,
416 scancodeMin,
417 scancodeMax - scancodeMin + 1,
418 &width);
419
420 for (int scancode = scancodeMin; scancode <= scancodeMax; scancode++)
421 {
422 // Translate the un-translated key codes using traditional X11 KeySym
423 // lookups
424 if (_glfw.x11.keycodes[scancode] < 0)
425 {
426 const size_t base = (scancode - scancodeMin) * width;
427 _glfw.x11.keycodes[scancode] = translateKeySyms(&keysyms[base], width);
428 }
429
430 // Store the reverse translation for faster key name lookup
431 if (_glfw.x11.keycodes[scancode] > 0)
432 _glfw.x11.scancodes[_glfw.x11.keycodes[scancode]] = scancode;
433 }
434
435 XFree(keysyms);
436 }
437
438 // Check whether the IM has a usable style
439 //
440 static GLFWbool hasUsableInputMethodStyle(void)
441 {
442 GLFWbool found = GLFW_FALSE;
443 XIMStyles* styles = NULL;
444
445 if (XGetIMValues(_glfw.x11.im, XNQueryInputStyle, &styles, NULL) != NULL)
446 return GLFW_FALSE;
447
448 for (unsigned int i = 0; i < styles->count_styles; i++)
449 {
450 if (styles->supported_styles[i] == (XIMPreeditNothing | XIMStatusNothing))
451 {
452 found = GLFW_TRUE;
453 break;
454 }
455 }
456
457 XFree(styles);
458 return found;
459 }
460
461 static void inputMethodDestroyCallback(XIM im, XPointer clientData, XPointer callData)
462 {
463 _glfw.x11.im = NULL;
464 }
465
466 static void inputMethodInstantiateCallback(Display* display,
467 XPointer clientData,
468 XPointer callData)
469 {
470 if (_glfw.x11.im)
471 return;
472
473 _glfw.x11.im = XOpenIM(_glfw.x11.display, 0, NULL, NULL);
474 if (_glfw.x11.im)
475 {
476 if (!hasUsableInputMethodStyle())
477 {
478 XCloseIM(_glfw.x11.im);
479 _glfw.x11.im = NULL;
480 }
481 }
482
483 if (_glfw.x11.im)
484 {
485 XIMCallback callback;
486 callback.callback = (XIMProc) inputMethodDestroyCallback;
487 callback.client_data = NULL;
488 XSetIMValues(_glfw.x11.im, XNDestroyCallback, &callback, NULL);
489
490 for (_GLFWwindow* window = _glfw.windowListHead; window; window = window->next)
491 _glfwCreateInputContextX11(window);
492 }
493 }
494
495 // Return the atom ID only if it is listed in the specified array
496 //
497 static Atom getAtomIfSupported(Atom* supportedAtoms,
498 unsigned long atomCount,
499 const char* atomName)
500 {
501 const Atom atom = XInternAtom(_glfw.x11.display, atomName, False);
502
503 for (unsigned long i = 0; i < atomCount; i++)
504 {
505 if (supportedAtoms[i] == atom)
506 return atom;
507 }
508
509 return None;
510 }
511
512 // Check whether the running window manager is EWMH-compliant
513 //
514 static void detectEWMH(void)
515 {
516 // First we read the _NET_SUPPORTING_WM_CHECK property on the root window
517
518 Window* windowFromRoot = NULL;
519 if (!_glfwGetWindowPropertyX11(_glfw.x11.root,
520 _glfw.x11.NET_SUPPORTING_WM_CHECK,
521 XA_WINDOW,
522 (unsigned char**) &windowFromRoot))
523 {
524 return;
525 }
526
527 _glfwGrabErrorHandlerX11();
528
529 // If it exists, it should be the XID of a top-level window
530 // Then we look for the same property on that window
531
532 Window* windowFromChild = NULL;
533 if (!_glfwGetWindowPropertyX11(*windowFromRoot,
534 _glfw.x11.NET_SUPPORTING_WM_CHECK,
535 XA_WINDOW,
536 (unsigned char**) &windowFromChild))
537 {
538 XFree(windowFromRoot);
539 return;
540 }
541
542 _glfwReleaseErrorHandlerX11();
543
544 // If the property exists, it should contain the XID of the window
545
546 if (*windowFromRoot != *windowFromChild)
547 {
548 XFree(windowFromRoot);
549 XFree(windowFromChild);
550 return;
551 }
552
553 XFree(windowFromRoot);
554 XFree(windowFromChild);
555
556 // We are now fairly sure that an EWMH-compliant WM is currently running
557 // We can now start querying the WM about what features it supports by
558 // looking in the _NET_SUPPORTED property on the root window
559 // It should contain a list of supported EWMH protocol and state atoms
560
561 Atom* supportedAtoms = NULL;
562 const unsigned long atomCount =
563 _glfwGetWindowPropertyX11(_glfw.x11.root,
564 _glfw.x11.NET_SUPPORTED,
565 XA_ATOM,
566 (unsigned char**) &supportedAtoms);
567
568 // See which of the atoms we support that are supported by the WM
569
570 _glfw.x11.NET_WM_STATE =
571 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_STATE");
572 _glfw.x11.NET_WM_STATE_ABOVE =
573 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_STATE_ABOVE");
574 _glfw.x11.NET_WM_STATE_FULLSCREEN =
575 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_STATE_FULLSCREEN");
576 _glfw.x11.NET_WM_STATE_MAXIMIZED_VERT =
577 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_STATE_MAXIMIZED_VERT");
578 _glfw.x11.NET_WM_STATE_MAXIMIZED_HORZ =
579 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_STATE_MAXIMIZED_HORZ");
580 _glfw.x11.NET_WM_STATE_DEMANDS_ATTENTION =
581 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_STATE_DEMANDS_ATTENTION");
582 _glfw.x11.NET_WM_FULLSCREEN_MONITORS =
583 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_FULLSCREEN_MONITORS");
584 _glfw.x11.NET_WM_WINDOW_TYPE =
585 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_WINDOW_TYPE");
586 _glfw.x11.NET_WM_WINDOW_TYPE_NORMAL =
587 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WM_WINDOW_TYPE_NORMAL");
588 _glfw.x11.NET_WORKAREA =
589 getAtomIfSupported(supportedAtoms, atomCount, "_NET_WORKAREA");
590 _glfw.x11.NET_CURRENT_DESKTOP =
591 getAtomIfSupported(supportedAtoms, atomCount, "_NET_CURRENT_DESKTOP");
592 _glfw.x11.NET_ACTIVE_WINDOW =
593 getAtomIfSupported(supportedAtoms, atomCount, "_NET_ACTIVE_WINDOW");
594 _glfw.x11.NET_FRAME_EXTENTS =
595 getAtomIfSupported(supportedAtoms, atomCount, "_NET_FRAME_EXTENTS");
596 _glfw.x11.NET_REQUEST_FRAME_EXTENTS =
597 getAtomIfSupported(supportedAtoms, atomCount, "_NET_REQUEST_FRAME_EXTENTS");
598
599 if (supportedAtoms)
600 XFree(supportedAtoms);
601 }
602
603 // Look for and initialize supported X11 extensions
604 //
605 static GLFWbool initExtensions(void)
606 {
607 #if defined(__OpenBSD__) || defined(__NetBSD__)
608 _glfw.x11.vidmode.handle = _glfwPlatformLoadModule("libXxf86vm.so");
609 #else
610 _glfw.x11.vidmode.handle = _glfwPlatformLoadModule("libXxf86vm.so.1");
611 #endif
612 if (_glfw.x11.vidmode.handle)
613 {
614 _glfw.x11.vidmode.QueryExtension = (PFN_XF86VidModeQueryExtension)
615 _glfwPlatformGetModuleSymbol(_glfw.x11.vidmode.handle, "XF86VidModeQueryExtension");
616 _glfw.x11.vidmode.GetGammaRamp = (PFN_XF86VidModeGetGammaRamp)
617 _glfwPlatformGetModuleSymbol(_glfw.x11.vidmode.handle, "XF86VidModeGetGammaRamp");
618 _glfw.x11.vidmode.SetGammaRamp = (PFN_XF86VidModeSetGammaRamp)
619 _glfwPlatformGetModuleSymbol(_glfw.x11.vidmode.handle, "XF86VidModeSetGammaRamp");
620 _glfw.x11.vidmode.GetGammaRampSize = (PFN_XF86VidModeGetGammaRampSize)
621 _glfwPlatformGetModuleSymbol(_glfw.x11.vidmode.handle, "XF86VidModeGetGammaRampSize");
622
623 _glfw.x11.vidmode.available =
624 XF86VidModeQueryExtension(_glfw.x11.display,
625 &_glfw.x11.vidmode.eventBase,
626 &_glfw.x11.vidmode.errorBase);
627 }
628
629 #if defined(__CYGWIN__)
630 _glfw.x11.xi.handle = _glfwPlatformLoadModule("libXi-6.so");
631 #elif defined(__OpenBSD__) || defined(__NetBSD__)
632 _glfw.x11.xi.handle = _glfwPlatformLoadModule("libXi.so");
633 #else
634 _glfw.x11.xi.handle = _glfwPlatformLoadModule("libXi.so.6");
635 #endif
636 if (_glfw.x11.xi.handle)
637 {
638 _glfw.x11.xi.QueryVersion = (PFN_XIQueryVersion)
639 _glfwPlatformGetModuleSymbol(_glfw.x11.xi.handle, "XIQueryVersion");
640 _glfw.x11.xi.SelectEvents = (PFN_XISelectEvents)
641 _glfwPlatformGetModuleSymbol(_glfw.x11.xi.handle, "XISelectEvents");
642
643 if (XQueryExtension(_glfw.x11.display,
644 "XInputExtension",
645 &_glfw.x11.xi.majorOpcode,
646 &_glfw.x11.xi.eventBase,
647 &_glfw.x11.xi.errorBase))
648 {
649 _glfw.x11.xi.major = 2;
650 _glfw.x11.xi.minor = 0;
651
652 if (XIQueryVersion(_glfw.x11.display,
653 &_glfw.x11.xi.major,
654 &_glfw.x11.xi.minor) == Success)
655 {
656 _glfw.x11.xi.available = GLFW_TRUE;
657 }
658 }
659 }
660
661 #if defined(__CYGWIN__)
662 _glfw.x11.randr.handle = _glfwPlatformLoadModule("libXrandr-2.so");
663 #elif defined(__OpenBSD__) || defined(__NetBSD__)
664 _glfw.x11.randr.handle = _glfwPlatformLoadModule("libXrandr.so");
665 #else
666 _glfw.x11.randr.handle = _glfwPlatformLoadModule("libXrandr.so.2");
667 #endif
668 if (_glfw.x11.randr.handle)
669 {
670 _glfw.x11.randr.AllocGamma = (PFN_XRRAllocGamma)
671 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRAllocGamma");
672 _glfw.x11.randr.FreeGamma = (PFN_XRRFreeGamma)
673 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRFreeGamma");
674 _glfw.x11.randr.FreeCrtcInfo = (PFN_XRRFreeCrtcInfo)
675 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRFreeCrtcInfo");
676 _glfw.x11.randr.FreeGamma = (PFN_XRRFreeGamma)
677 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRFreeGamma");
678 _glfw.x11.randr.FreeOutputInfo = (PFN_XRRFreeOutputInfo)
679 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRFreeOutputInfo");
680 _glfw.x11.randr.FreeScreenResources = (PFN_XRRFreeScreenResources)
681 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRFreeScreenResources");
682 _glfw.x11.randr.GetCrtcGamma = (PFN_XRRGetCrtcGamma)
683 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRGetCrtcGamma");
684 _glfw.x11.randr.GetCrtcGammaSize = (PFN_XRRGetCrtcGammaSize)
685 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRGetCrtcGammaSize");
686 _glfw.x11.randr.GetCrtcInfo = (PFN_XRRGetCrtcInfo)
687 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRGetCrtcInfo");
688 _glfw.x11.randr.GetOutputInfo = (PFN_XRRGetOutputInfo)
689 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRGetOutputInfo");
690 _glfw.x11.randr.GetOutputPrimary = (PFN_XRRGetOutputPrimary)
691 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRGetOutputPrimary");
692 _glfw.x11.randr.GetScreenResourcesCurrent = (PFN_XRRGetScreenResourcesCurrent)
693 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRGetScreenResourcesCurrent");
694 _glfw.x11.randr.QueryExtension = (PFN_XRRQueryExtension)
695 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRQueryExtension");
696 _glfw.x11.randr.QueryVersion = (PFN_XRRQueryVersion)
697 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRQueryVersion");
698 _glfw.x11.randr.SelectInput = (PFN_XRRSelectInput)
699 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRSelectInput");
700 _glfw.x11.randr.SetCrtcConfig = (PFN_XRRSetCrtcConfig)
701 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRSetCrtcConfig");
702 _glfw.x11.randr.SetCrtcGamma = (PFN_XRRSetCrtcGamma)
703 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRSetCrtcGamma");
704 _glfw.x11.randr.UpdateConfiguration = (PFN_XRRUpdateConfiguration)
705 _glfwPlatformGetModuleSymbol(_glfw.x11.randr.handle, "XRRUpdateConfiguration");
706
707 if (XRRQueryExtension(_glfw.x11.display,
708 &_glfw.x11.randr.eventBase,
709 &_glfw.x11.randr.errorBase))
710 {
711 if (XRRQueryVersion(_glfw.x11.display,
712 &_glfw.x11.randr.major,
713 &_glfw.x11.randr.minor))
714 {
715 // The GLFW RandR path requires at least version 1.3
716 if (_glfw.x11.randr.major > 1 || _glfw.x11.randr.minor >= 3)
717 _glfw.x11.randr.available = GLFW_TRUE;
718 }
719 else
720 {
721 _glfwInputError(GLFW_PLATFORM_ERROR,
722 "X11: Failed to query RandR version");
723 }
724 }
725 }
726
727 if (_glfw.x11.randr.available)
728 {
729 XRRScreenResources* sr = XRRGetScreenResourcesCurrent(_glfw.x11.display,
730 _glfw.x11.root);
731
732 if (!sr->ncrtc || !XRRGetCrtcGammaSize(_glfw.x11.display, sr->crtcs[0]))
733 {
734 // This is likely an older Nvidia driver with broken gamma support
735 // Flag it as useless and fall back to xf86vm gamma, if available
736 _glfw.x11.randr.gammaBroken = GLFW_TRUE;
737 }
738
739 if (!sr->ncrtc)
740 {
741 // A system without CRTCs is likely a system with broken RandR
742 // Disable the RandR monitor path and fall back to core functions
743 _glfw.x11.randr.monitorBroken = GLFW_TRUE;
744 }
745
746 XRRFreeScreenResources(sr);
747 }
748
749 if (_glfw.x11.randr.available && !_glfw.x11.randr.monitorBroken)
750 {
751 XRRSelectInput(_glfw.x11.display, _glfw.x11.root,
752 RROutputChangeNotifyMask);
753 }
754
755 #if defined(__CYGWIN__)
756 _glfw.x11.xcursor.handle = _glfwPlatformLoadModule("libXcursor-1.so");
757 #elif defined(__OpenBSD__) || defined(__NetBSD__)
758 _glfw.x11.xcursor.handle = _glfwPlatformLoadModule("libXcursor.so");
759 #else
760 _glfw.x11.xcursor.handle = _glfwPlatformLoadModule("libXcursor.so.1");
761 #endif
762 if (_glfw.x11.xcursor.handle)
763 {
764 _glfw.x11.xcursor.ImageCreate = (PFN_XcursorImageCreate)
765 _glfwPlatformGetModuleSymbol(_glfw.x11.xcursor.handle, "XcursorImageCreate");
766 _glfw.x11.xcursor.ImageDestroy = (PFN_XcursorImageDestroy)
767 _glfwPlatformGetModuleSymbol(_glfw.x11.xcursor.handle, "XcursorImageDestroy");
768 _glfw.x11.xcursor.ImageLoadCursor = (PFN_XcursorImageLoadCursor)
769 _glfwPlatformGetModuleSymbol(_glfw.x11.xcursor.handle, "XcursorImageLoadCursor");
770 _glfw.x11.xcursor.GetTheme = (PFN_XcursorGetTheme)
771 _glfwPlatformGetModuleSymbol(_glfw.x11.xcursor.handle, "XcursorGetTheme");
772 _glfw.x11.xcursor.GetDefaultSize = (PFN_XcursorGetDefaultSize)
773 _glfwPlatformGetModuleSymbol(_glfw.x11.xcursor.handle, "XcursorGetDefaultSize");
774 _glfw.x11.xcursor.LibraryLoadImage = (PFN_XcursorLibraryLoadImage)
775 _glfwPlatformGetModuleSymbol(_glfw.x11.xcursor.handle, "XcursorLibraryLoadImage");
776 }
777
778 #if defined(__CYGWIN__)
779 _glfw.x11.xinerama.handle = _glfwPlatformLoadModule("libXinerama-1.so");
780 #elif defined(__OpenBSD__) || defined(__NetBSD__)
781 _glfw.x11.xinerama.handle = _glfwPlatformLoadModule("libXinerama.so");
782 #else
783 _glfw.x11.xinerama.handle = _glfwPlatformLoadModule("libXinerama.so.1");
784 #endif
785 if (_glfw.x11.xinerama.handle)
786 {
787 _glfw.x11.xinerama.IsActive = (PFN_XineramaIsActive)
788 _glfwPlatformGetModuleSymbol(_glfw.x11.xinerama.handle, "XineramaIsActive");
789 _glfw.x11.xinerama.QueryExtension = (PFN_XineramaQueryExtension)
790 _glfwPlatformGetModuleSymbol(_glfw.x11.xinerama.handle, "XineramaQueryExtension");
791 _glfw.x11.xinerama.QueryScreens = (PFN_XineramaQueryScreens)
792 _glfwPlatformGetModuleSymbol(_glfw.x11.xinerama.handle, "XineramaQueryScreens");
793
794 if (XineramaQueryExtension(_glfw.x11.display,
795 &_glfw.x11.xinerama.major,
796 &_glfw.x11.xinerama.minor))
797 {
798 if (XineramaIsActive(_glfw.x11.display))
799 _glfw.x11.xinerama.available = GLFW_TRUE;
800 }
801 }
802
803 _glfw.x11.xkb.major = 1;
804 _glfw.x11.xkb.minor = 0;
805 _glfw.x11.xkb.available =
806 XkbQueryExtension(_glfw.x11.display,
807 &_glfw.x11.xkb.majorOpcode,
808 &_glfw.x11.xkb.eventBase,
809 &_glfw.x11.xkb.errorBase,
810 &_glfw.x11.xkb.major,
811 &_glfw.x11.xkb.minor);
812
813 if (_glfw.x11.xkb.available)
814 {
815 Bool supported;
816
817 if (XkbSetDetectableAutoRepeat(_glfw.x11.display, True, &supported))
818 {
819 if (supported)
820 _glfw.x11.xkb.detectable = GLFW_TRUE;
821 }
822
823 XkbStateRec state;
824 if (XkbGetState(_glfw.x11.display, XkbUseCoreKbd, &state) == Success)
825 _glfw.x11.xkb.group = (unsigned int)state.group;
826
827 XkbSelectEventDetails(_glfw.x11.display, XkbUseCoreKbd, XkbStateNotify,
828 XkbGroupStateMask, XkbGroupStateMask);
829 }
830
831 if (_glfw.hints.init.x11.xcbVulkanSurface)
832 {
833 #if defined(__CYGWIN__)
834 _glfw.x11.x11xcb.handle = _glfwPlatformLoadModule("libX11-xcb-1.so");
835 #elif defined(__OpenBSD__) || defined(__NetBSD__)
836 _glfw.x11.x11xcb.handle = _glfwPlatformLoadModule("libX11-xcb.so");
837 #else
838 _glfw.x11.x11xcb.handle = _glfwPlatformLoadModule("libX11-xcb.so.1");
839 #endif
840 }
841
842 if (_glfw.x11.x11xcb.handle)
843 {
844 _glfw.x11.x11xcb.GetXCBConnection = (PFN_XGetXCBConnection)
845 _glfwPlatformGetModuleSymbol(_glfw.x11.x11xcb.handle, "XGetXCBConnection");
846 }
847
848 #if defined(__CYGWIN__)
849 _glfw.x11.xrender.handle = _glfwPlatformLoadModule("libXrender-1.so");
850 #elif defined(__OpenBSD__) || defined(__NetBSD__)
851 _glfw.x11.xrender.handle = _glfwPlatformLoadModule("libXrender.so");
852 #else
853 _glfw.x11.xrender.handle = _glfwPlatformLoadModule("libXrender.so.1");
854 #endif
855 if (_glfw.x11.xrender.handle)
856 {
857 _glfw.x11.xrender.QueryExtension = (PFN_XRenderQueryExtension)
858 _glfwPlatformGetModuleSymbol(_glfw.x11.xrender.handle, "XRenderQueryExtension");
859 _glfw.x11.xrender.QueryVersion = (PFN_XRenderQueryVersion)
860 _glfwPlatformGetModuleSymbol(_glfw.x11.xrender.handle, "XRenderQueryVersion");
861 _glfw.x11.xrender.FindVisualFormat = (PFN_XRenderFindVisualFormat)
862 _glfwPlatformGetModuleSymbol(_glfw.x11.xrender.handle, "XRenderFindVisualFormat");
863
864 if (XRenderQueryExtension(_glfw.x11.display,
865 &_glfw.x11.xrender.errorBase,
866 &_glfw.x11.xrender.eventBase))
867 {
868 if (XRenderQueryVersion(_glfw.x11.display,
869 &_glfw.x11.xrender.major,
870 &_glfw.x11.xrender.minor))
871 {
872 _glfw.x11.xrender.available = GLFW_TRUE;
873 }
874 }
875 }
876
877 #if defined(__CYGWIN__)
878 _glfw.x11.xshape.handle = _glfwPlatformLoadModule("libXext-6.so");
879 #elif defined(__OpenBSD__) || defined(__NetBSD__)
880 _glfw.x11.xshape.handle = _glfwPlatformLoadModule("libXext.so");
881 #else
882 _glfw.x11.xshape.handle = _glfwPlatformLoadModule("libXext.so.6");
883 #endif
884 if (_glfw.x11.xshape.handle)
885 {
886 _glfw.x11.xshape.QueryExtension = (PFN_XShapeQueryExtension)
887 _glfwPlatformGetModuleSymbol(_glfw.x11.xshape.handle, "XShapeQueryExtension");
888 _glfw.x11.xshape.ShapeCombineRegion = (PFN_XShapeCombineRegion)
889 _glfwPlatformGetModuleSymbol(_glfw.x11.xshape.handle, "XShapeCombineRegion");
890 _glfw.x11.xshape.QueryVersion = (PFN_XShapeQueryVersion)
891 _glfwPlatformGetModuleSymbol(_glfw.x11.xshape.handle, "XShapeQueryVersion");
892 _glfw.x11.xshape.ShapeCombineMask = (PFN_XShapeCombineMask)
893 _glfwPlatformGetModuleSymbol(_glfw.x11.xshape.handle, "XShapeCombineMask");
894
895 if (XShapeQueryExtension(_glfw.x11.display,
896 &_glfw.x11.xshape.errorBase,
897 &_glfw.x11.xshape.eventBase))
898 {
899 if (XShapeQueryVersion(_glfw.x11.display,
900 &_glfw.x11.xshape.major,
901 &_glfw.x11.xshape.minor))
902 {
903 _glfw.x11.xshape.available = GLFW_TRUE;
904 }
905 }
906 }
907
908 // Update the key code LUT
909 // FIXME: We should listen to XkbMapNotify events to track changes to
910 // the keyboard mapping.
911 createKeyTables();
912
913 // String format atoms
914 _glfw.x11.NULL_ = XInternAtom(_glfw.x11.display, "NULL", False);
915 _glfw.x11.UTF8_STRING = XInternAtom(_glfw.x11.display, "UTF8_STRING", False);
916 _glfw.x11.ATOM_PAIR = XInternAtom(_glfw.x11.display, "ATOM_PAIR", False);
917
918 // Custom selection property atom
919 _glfw.x11.GLFW_SELECTION =
920 XInternAtom(_glfw.x11.display, "GLFW_SELECTION", False);
921
922 // ICCCM standard clipboard atoms
923 _glfw.x11.TARGETS = XInternAtom(_glfw.x11.display, "TARGETS", False);
924 _glfw.x11.MULTIPLE = XInternAtom(_glfw.x11.display, "MULTIPLE", False);
925 _glfw.x11.PRIMARY = XInternAtom(_glfw.x11.display, "PRIMARY", False);
926 _glfw.x11.INCR = XInternAtom(_glfw.x11.display, "INCR", False);
927 _glfw.x11.CLIPBOARD = XInternAtom(_glfw.x11.display, "CLIPBOARD", False);
928
929 // Clipboard manager atoms
930 _glfw.x11.CLIPBOARD_MANAGER =
931 XInternAtom(_glfw.x11.display, "CLIPBOARD_MANAGER", False);
932 _glfw.x11.SAVE_TARGETS =
933 XInternAtom(_glfw.x11.display, "SAVE_TARGETS", False);
934
935 // Xdnd (drag and drop) atoms
936 _glfw.x11.XdndAware = XInternAtom(_glfw.x11.display, "XdndAware", False);
937 _glfw.x11.XdndEnter = XInternAtom(_glfw.x11.display, "XdndEnter", False);
938 _glfw.x11.XdndPosition = XInternAtom(_glfw.x11.display, "XdndPosition", False);
939 _glfw.x11.XdndStatus = XInternAtom(_glfw.x11.display, "XdndStatus", False);
940 _glfw.x11.XdndActionCopy = XInternAtom(_glfw.x11.display, "XdndActionCopy", False);
941 _glfw.x11.XdndDrop = XInternAtom(_glfw.x11.display, "XdndDrop", False);
942 _glfw.x11.XdndFinished = XInternAtom(_glfw.x11.display, "XdndFinished", False);
943 _glfw.x11.XdndSelection = XInternAtom(_glfw.x11.display, "XdndSelection", False);
944 _glfw.x11.XdndTypeList = XInternAtom(_glfw.x11.display, "XdndTypeList", False);
945 _glfw.x11.text_uri_list = XInternAtom(_glfw.x11.display, "text/uri-list", False);
946
947 // ICCCM, EWMH and Motif window property atoms
948 // These can be set safely even without WM support
949 // The EWMH atoms that require WM support are handled in detectEWMH
950 _glfw.x11.WM_PROTOCOLS =
951 XInternAtom(_glfw.x11.display, "WM_PROTOCOLS", False);
952 _glfw.x11.WM_STATE =
953 XInternAtom(_glfw.x11.display, "WM_STATE", False);
954 _glfw.x11.WM_DELETE_WINDOW =
955 XInternAtom(_glfw.x11.display, "WM_DELETE_WINDOW", False);
956 _glfw.x11.NET_SUPPORTED =
957 XInternAtom(_glfw.x11.display, "_NET_SUPPORTED", False);
958 _glfw.x11.NET_SUPPORTING_WM_CHECK =
959 XInternAtom(_glfw.x11.display, "_NET_SUPPORTING_WM_CHECK", False);
960 _glfw.x11.NET_WM_ICON =
961 XInternAtom(_glfw.x11.display, "_NET_WM_ICON", False);
962 _glfw.x11.NET_WM_PING =
963 XInternAtom(_glfw.x11.display, "_NET_WM_PING", False);
964 _glfw.x11.NET_WM_PID =
965 XInternAtom(_glfw.x11.display, "_NET_WM_PID", False);
966 _glfw.x11.NET_WM_NAME =
967 XInternAtom(_glfw.x11.display, "_NET_WM_NAME", False);
968 _glfw.x11.NET_WM_ICON_NAME =
969 XInternAtom(_glfw.x11.display, "_NET_WM_ICON_NAME", False);
970 _glfw.x11.NET_WM_BYPASS_COMPOSITOR =
971 XInternAtom(_glfw.x11.display, "_NET_WM_BYPASS_COMPOSITOR", False);
972 _glfw.x11.NET_WM_WINDOW_OPACITY =
973 XInternAtom(_glfw.x11.display, "_NET_WM_WINDOW_OPACITY", False);
974 _glfw.x11.MOTIF_WM_HINTS =
975 XInternAtom(_glfw.x11.display, "_MOTIF_WM_HINTS", False);
976
977 // The compositing manager selection name contains the screen number
978 {
979 char name[32];
980 snprintf(name, sizeof(name), "_NET_WM_CM_S%u", _glfw.x11.screen);
981 _glfw.x11.NET_WM_CM_Sx = XInternAtom(_glfw.x11.display, name, False);
982 }
983
984 // Detect whether an EWMH-conformant window manager is running
985 detectEWMH();
986
987 return GLFW_TRUE;
988 }
989
990 // Retrieve system content scale via folklore heuristics
991 //
992 static void getSystemContentScale(float* xscale, float* yscale)
993 {
994 // Start by assuming the default X11 DPI
995 // NOTE: Some desktop environments (KDE) may remove the Xft.dpi field when it
996 // would be set to 96, so assume that is the case if we cannot find it
997 float xdpi = 96.f, ydpi = 96.f;
998
999 // NOTE: Basing the scale on Xft.dpi where available should provide the most
1000 // consistent user experience (matches Qt, Gtk, etc), although not
1001 // always the most accurate one
1002 char* rms = XResourceManagerString(_glfw.x11.display);
1003 if (rms)
1004 {
1005 XrmDatabase db = XrmGetStringDatabase(rms);
1006 if (db)
1007 {
1008 XrmValue value;
1009 char* type = NULL;
1010
1011 if (XrmGetResource(db, "Xft.dpi", "Xft.Dpi", &type, &value))
1012 {
1013 if (type && strcmp(type, "String") == 0)
1014 xdpi = ydpi = atof(value.addr);
1015 }
1016
1017 XrmDestroyDatabase(db);
1018 }
1019 }
1020
1021 *xscale = xdpi / 96.f;
1022 *yscale = ydpi / 96.f;
1023 }
1024
1025 // Create a blank cursor for hidden and disabled cursor modes
1026 //
1027 static Cursor createHiddenCursor(void)
1028 {
1029 unsigned char pixels[16 * 16 * 4] = { 0 };
1030 GLFWimage image = { 16, 16, pixels };
1031 return _glfwCreateNativeCursorX11(&image, 0, 0);
1032 }
1033
1034 // Create a helper window for IPC
1035 //
1036 static Window createHelperWindow(void)
1037 {
1038 XSetWindowAttributes wa;
1039 wa.event_mask = PropertyChangeMask;
1040
1041 return XCreateWindow(_glfw.x11.display, _glfw.x11.root,
1042 0, 0, 1, 1, 0, 0,
1043 InputOnly,
1044 DefaultVisual(_glfw.x11.display, _glfw.x11.screen),
1045 CWEventMask, &wa);
1046 }
1047
1048 // Create the pipe for empty events without assumuing the OS has pipe2(2)
1049 //
1050 static GLFWbool createEmptyEventPipe(void)
1051 {
1052 if (pipe(_glfw.x11.emptyEventPipe) != 0)
1053 {
1054 _glfwInputError(GLFW_PLATFORM_ERROR,
1055 "X11: Failed to create empty event pipe: %s",
1056 strerror(errno));
1057 return GLFW_FALSE;
1058 }
1059
1060 for (int i = 0; i < 2; i++)
1061 {
1062 const int sf = fcntl(_glfw.x11.emptyEventPipe[i], F_GETFL, 0);
1063 const int df = fcntl(_glfw.x11.emptyEventPipe[i], F_GETFD, 0);
1064
1065 if (sf == -1 || df == -1 ||
1066 fcntl(_glfw.x11.emptyEventPipe[i], F_SETFL, sf | O_NONBLOCK) == -1 ||
1067 fcntl(_glfw.x11.emptyEventPipe[i], F_SETFD, df | FD_CLOEXEC) == -1)
1068 {
1069 _glfwInputError(GLFW_PLATFORM_ERROR,
1070 "X11: Failed to set flags for empty event pipe: %s",
1071 strerror(errno));
1072 return GLFW_FALSE;
1073 }
1074 }
1075
1076 return GLFW_TRUE;
1077 }
1078
1079 // X error handler
1080 //
1081 static int errorHandler(Display *display, XErrorEvent* event)
1082 {
1083 if (_glfw.x11.display != display)
1084 return 0;
1085
1086 _glfw.x11.errorCode = event->error_code;
1087 return 0;
1088 }
1089
1090
1091 //////////////////////////////////////////////////////////////////////////
1092 ////// GLFW internal API //////
1093 //////////////////////////////////////////////////////////////////////////
1094
1095 // Sets the X error handler callback
1096 //
1097 void _glfwGrabErrorHandlerX11(void)
1098 {
1099 assert(_glfw.x11.errorHandler == NULL);
1100 _glfw.x11.errorCode = Success;
1101 _glfw.x11.errorHandler = XSetErrorHandler(errorHandler);
1102 }
1103
1104 // Clears the X error handler callback
1105 //
1106 void _glfwReleaseErrorHandlerX11(void)
1107 {
1108 // Synchronize to make sure all commands are processed
1109 XSync(_glfw.x11.display, False);
1110 XSetErrorHandler(_glfw.x11.errorHandler);
1111 _glfw.x11.errorHandler = NULL;
1112 }
1113
1114 // Reports the specified error, appending information about the last X error
1115 //
1116 void _glfwInputErrorX11(int error, const char* message)
1117 {
1118 char buffer[_GLFW_MESSAGE_SIZE];
1119 XGetErrorText(_glfw.x11.display, _glfw.x11.errorCode,
1120 buffer, sizeof(buffer));
1121
1122 _glfwInputError(error, "%s: %s", message, buffer);
1123 }
1124
1125 // Creates a native cursor object from the specified image and hotspot
1126 //
1127 Cursor _glfwCreateNativeCursorX11(const GLFWimage* image, int xhot, int yhot)
1128 {
1129 Cursor cursor;
1130
1131 if (!_glfw.x11.xcursor.handle)
1132 return None;
1133
1134 XcursorImage* native = XcursorImageCreate(image->width, image->height);
1135 if (native == NULL)
1136 return None;
1137
1138 native->xhot = xhot;
1139 native->yhot = yhot;
1140
1141 unsigned char* source = (unsigned char*) image->pixels;
1142 XcursorPixel* target = native->pixels;
1143
1144 for (int i = 0; i < image->width * image->height; i++, target++, source += 4)
1145 {
1146 unsigned int alpha = source[3];
1147
1148 *target = (alpha << 24) |
1149 ((unsigned char) ((source[0] * alpha) / 255) << 16) |
1150 ((unsigned char) ((source[1] * alpha) / 255) << 8) |
1151 ((unsigned char) ((source[2] * alpha) / 255) << 0);
1152 }
1153
1154 cursor = XcursorImageLoadCursor(_glfw.x11.display, native);
1155 XcursorImageDestroy(native);
1156
1157 return cursor;
1158 }
1159
1160
1161 //////////////////////////////////////////////////////////////////////////
1162 ////// GLFW platform API //////
1163 //////////////////////////////////////////////////////////////////////////
1164
1165 GLFWbool _glfwConnectX11(int platformID, _GLFWplatform* platform)
1166 {
1167 const _GLFWplatform x11 =
1168 {
1169 GLFW_PLATFORM_X11,
1170 _glfwInitX11,
1171 _glfwTerminateX11,
1172 _glfwGetCursorPosX11,
1173 _glfwSetCursorPosX11,
1174 _glfwSetCursorModeX11,
1175 _glfwSetRawMouseMotionX11,
1176 _glfwRawMouseMotionSupportedX11,
1177 _glfwCreateCursorX11,
1178 _glfwCreateStandardCursorX11,
1179 _glfwDestroyCursorX11,
1180 _glfwSetCursorX11,
1181 _glfwGetScancodeNameX11,
1182 _glfwGetKeyScancodeX11,
1183 _glfwSetClipboardStringX11,
1184 _glfwGetClipboardStringX11,
1185 #if defined(__linux__)
1186 _glfwInitJoysticksLinux,
1187 _glfwTerminateJoysticksLinux,
1188 _glfwPollJoystickLinux,
1189 _glfwGetMappingNameLinux,
1190 _glfwUpdateGamepadGUIDLinux,
1191 #else
1192 _glfwInitJoysticksNull,
1193 _glfwTerminateJoysticksNull,
1194 _glfwPollJoystickNull,
1195 _glfwGetMappingNameNull,
1196 _glfwUpdateGamepadGUIDNull,
1197 #endif
1198 _glfwFreeMonitorX11,
1199 _glfwGetMonitorPosX11,
1200 _glfwGetMonitorContentScaleX11,
1201 _glfwGetMonitorWorkareaX11,
1202 _glfwGetVideoModesX11,
1203 _glfwGetVideoModeX11,
1204 _glfwGetGammaRampX11,
1205 _glfwSetGammaRampX11,
1206 _glfwCreateWindowX11,
1207 _glfwDestroyWindowX11,
1208 _glfwSetWindowTitleX11,
1209 _glfwSetWindowIconX11,
1210 _glfwGetWindowPosX11,
1211 _glfwSetWindowPosX11,
1212 _glfwGetWindowSizeX11,
1213 _glfwSetWindowSizeX11,
1214 _glfwSetWindowSizeLimitsX11,
1215 _glfwSetWindowAspectRatioX11,
1216 _glfwGetFramebufferSizeX11,
1217 _glfwGetWindowFrameSizeX11,
1218 _glfwGetWindowContentScaleX11,
1219 _glfwIconifyWindowX11,
1220 _glfwRestoreWindowX11,
1221 _glfwMaximizeWindowX11,
1222 _glfwShowWindowX11,
1223 _glfwHideWindowX11,
1224 _glfwRequestWindowAttentionX11,
1225 _glfwFocusWindowX11,
1226 _glfwSetWindowMonitorX11,
1227 _glfwWindowFocusedX11,
1228 _glfwWindowIconifiedX11,
1229 _glfwWindowVisibleX11,
1230 _glfwWindowMaximizedX11,
1231 _glfwWindowHoveredX11,
1232 _glfwFramebufferTransparentX11,
1233 _glfwGetWindowOpacityX11,
1234 _glfwSetWindowResizableX11,
1235 _glfwSetWindowDecoratedX11,
1236 _glfwSetWindowFloatingX11,
1237 _glfwSetWindowOpacityX11,
1238 _glfwSetWindowMousePassthroughX11,
1239 _glfwPollEventsX11,
1240 _glfwWaitEventsX11,
1241 _glfwWaitEventsTimeoutX11,
1242 _glfwPostEmptyEventX11,
1243 _glfwGetEGLPlatformX11,
1244 _glfwGetEGLNativeDisplayX11,
1245 _glfwGetEGLNativeWindowX11,
1246 _glfwGetRequiredInstanceExtensionsX11,
1247 _glfwGetPhysicalDevicePresentationSupportX11,
1248 _glfwCreateWindowSurfaceX11,
1249 };
1250
1251 // HACK: If the application has left the locale as "C" then both wide
1252 // character text input and explicit UTF-8 input via XIM will break
1253 // This sets the CTYPE part of the current locale from the environment
1254 // in the hope that it is set to something more sane than "C"
1255 if (strcmp(setlocale(LC_CTYPE, NULL), "C") == 0)
1256 setlocale(LC_CTYPE, "");
1257
1258 #if defined(__CYGWIN__)
1259 void* module = _glfwPlatformLoadModule("libX11-6.so");
1260 #elif defined(__OpenBSD__) || defined(__NetBSD__)
1261 void* module = _glfwPlatformLoadModule("libX11.so");
1262 #else
1263 void* module = _glfwPlatformLoadModule("libX11.so.6");
1264 #endif
1265 if (!module)
1266 {
1267 if (platformID == GLFW_PLATFORM_X11)
1268 _glfwInputError(GLFW_PLATFORM_ERROR, "X11: Failed to load Xlib");
1269
1270 return GLFW_FALSE;
1271 }
1272
1273 PFN_XInitThreads XInitThreads = (PFN_XInitThreads)
1274 _glfwPlatformGetModuleSymbol(module, "XInitThreads");
1275 PFN_XrmInitialize XrmInitialize = (PFN_XrmInitialize)
1276 _glfwPlatformGetModuleSymbol(module, "XrmInitialize");
1277 PFN_XOpenDisplay XOpenDisplay = (PFN_XOpenDisplay)
1278 _glfwPlatformGetModuleSymbol(module, "XOpenDisplay");
1279 if (!XInitThreads || !XrmInitialize || !XOpenDisplay)
1280 {
1281 if (platformID == GLFW_PLATFORM_X11)
1282 _glfwInputError(GLFW_PLATFORM_ERROR, "X11: Failed to load Xlib entry point");
1283
1284 _glfwPlatformFreeModule(module);
1285 return GLFW_FALSE;
1286 }
1287
1288 XInitThreads();
1289 XrmInitialize();
1290
1291 Display* display = XOpenDisplay(NULL);
1292 if (!display)
1293 {
1294 if (platformID == GLFW_PLATFORM_X11)
1295 {
1296 const char* name = getenv("DISPLAY");
1297 if (name)
1298 {
1299 _glfwInputError(GLFW_PLATFORM_UNAVAILABLE,
1300 "X11: Failed to open display %s", name);
1301 }
1302 else
1303 {
1304 _glfwInputError(GLFW_PLATFORM_UNAVAILABLE,
1305 "X11: The DISPLAY environment variable is missing");
1306 }
1307 }
1308
1309 _glfwPlatformFreeModule(module);
1310 return GLFW_FALSE;
1311 }
1312
1313 _glfw.x11.display = display;
1314 _glfw.x11.xlib.handle = module;
1315
1316 *platform = x11;
1317 return GLFW_TRUE;
1318 }
1319
1320 int _glfwInitX11(void)
1321 {
1322 _glfw.x11.xlib.AllocClassHint = (PFN_XAllocClassHint)
1323 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XAllocClassHint");
1324 _glfw.x11.xlib.AllocSizeHints = (PFN_XAllocSizeHints)
1325 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XAllocSizeHints");
1326 _glfw.x11.xlib.AllocWMHints = (PFN_XAllocWMHints)
1327 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XAllocWMHints");
1328 _glfw.x11.xlib.ChangeProperty = (PFN_XChangeProperty)
1329 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XChangeProperty");
1330 _glfw.x11.xlib.ChangeWindowAttributes = (PFN_XChangeWindowAttributes)
1331 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XChangeWindowAttributes");
1332 _glfw.x11.xlib.CheckIfEvent = (PFN_XCheckIfEvent)
1333 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCheckIfEvent");
1334 _glfw.x11.xlib.CheckTypedWindowEvent = (PFN_XCheckTypedWindowEvent)
1335 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCheckTypedWindowEvent");
1336 _glfw.x11.xlib.CloseDisplay = (PFN_XCloseDisplay)
1337 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCloseDisplay");
1338 _glfw.x11.xlib.CloseIM = (PFN_XCloseIM)
1339 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCloseIM");
1340 _glfw.x11.xlib.ConvertSelection = (PFN_XConvertSelection)
1341 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XConvertSelection");
1342 _glfw.x11.xlib.CreateColormap = (PFN_XCreateColormap)
1343 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCreateColormap");
1344 _glfw.x11.xlib.CreateFontCursor = (PFN_XCreateFontCursor)
1345 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCreateFontCursor");
1346 _glfw.x11.xlib.CreateIC = (PFN_XCreateIC)
1347 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCreateIC");
1348 _glfw.x11.xlib.CreateRegion = (PFN_XCreateRegion)
1349 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCreateRegion");
1350 _glfw.x11.xlib.CreateWindow = (PFN_XCreateWindow)
1351 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XCreateWindow");
1352 _glfw.x11.xlib.DefineCursor = (PFN_XDefineCursor)
1353 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDefineCursor");
1354 _glfw.x11.xlib.DeleteContext = (PFN_XDeleteContext)
1355 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDeleteContext");
1356 _glfw.x11.xlib.DeleteProperty = (PFN_XDeleteProperty)
1357 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDeleteProperty");
1358 _glfw.x11.xlib.DestroyIC = (PFN_XDestroyIC)
1359 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDestroyIC");
1360 _glfw.x11.xlib.DestroyRegion = (PFN_XDestroyRegion)
1361 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDestroyRegion");
1362 _glfw.x11.xlib.DestroyWindow = (PFN_XDestroyWindow)
1363 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDestroyWindow");
1364 _glfw.x11.xlib.DisplayKeycodes = (PFN_XDisplayKeycodes)
1365 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XDisplayKeycodes");
1366 _glfw.x11.xlib.EventsQueued = (PFN_XEventsQueued)
1367 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XEventsQueued");
1368 _glfw.x11.xlib.FilterEvent = (PFN_XFilterEvent)
1369 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFilterEvent");
1370 _glfw.x11.xlib.FindContext = (PFN_XFindContext)
1371 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFindContext");
1372 _glfw.x11.xlib.Flush = (PFN_XFlush)
1373 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFlush");
1374 _glfw.x11.xlib.Free = (PFN_XFree)
1375 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFree");
1376 _glfw.x11.xlib.FreeColormap = (PFN_XFreeColormap)
1377 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFreeColormap");
1378 _glfw.x11.xlib.FreeCursor = (PFN_XFreeCursor)
1379 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFreeCursor");
1380 _glfw.x11.xlib.FreeEventData = (PFN_XFreeEventData)
1381 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XFreeEventData");
1382 _glfw.x11.xlib.GetErrorText = (PFN_XGetErrorText)
1383 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetErrorText");
1384 _glfw.x11.xlib.GetEventData = (PFN_XGetEventData)
1385 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetEventData");
1386 _glfw.x11.xlib.GetICValues = (PFN_XGetICValues)
1387 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetICValues");
1388 _glfw.x11.xlib.GetIMValues = (PFN_XGetIMValues)
1389 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetIMValues");
1390 _glfw.x11.xlib.GetInputFocus = (PFN_XGetInputFocus)
1391 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetInputFocus");
1392 _glfw.x11.xlib.GetKeyboardMapping = (PFN_XGetKeyboardMapping)
1393 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetKeyboardMapping");
1394 _glfw.x11.xlib.GetScreenSaver = (PFN_XGetScreenSaver)
1395 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetScreenSaver");
1396 _glfw.x11.xlib.GetSelectionOwner = (PFN_XGetSelectionOwner)
1397 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetSelectionOwner");
1398 _glfw.x11.xlib.GetVisualInfo = (PFN_XGetVisualInfo)
1399 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetVisualInfo");
1400 _glfw.x11.xlib.GetWMNormalHints = (PFN_XGetWMNormalHints)
1401 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetWMNormalHints");
1402 _glfw.x11.xlib.GetWindowAttributes = (PFN_XGetWindowAttributes)
1403 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetWindowAttributes");
1404 _glfw.x11.xlib.GetWindowProperty = (PFN_XGetWindowProperty)
1405 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGetWindowProperty");
1406 _glfw.x11.xlib.GrabPointer = (PFN_XGrabPointer)
1407 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XGrabPointer");
1408 _glfw.x11.xlib.IconifyWindow = (PFN_XIconifyWindow)
1409 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XIconifyWindow");
1410 _glfw.x11.xlib.InternAtom = (PFN_XInternAtom)
1411 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XInternAtom");
1412 _glfw.x11.xlib.LookupString = (PFN_XLookupString)
1413 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XLookupString");
1414 _glfw.x11.xlib.MapRaised = (PFN_XMapRaised)
1415 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XMapRaised");
1416 _glfw.x11.xlib.MapWindow = (PFN_XMapWindow)
1417 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XMapWindow");
1418 _glfw.x11.xlib.MoveResizeWindow = (PFN_XMoveResizeWindow)
1419 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XMoveResizeWindow");
1420 _glfw.x11.xlib.MoveWindow = (PFN_XMoveWindow)
1421 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XMoveWindow");
1422 _glfw.x11.xlib.NextEvent = (PFN_XNextEvent)
1423 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XNextEvent");
1424 _glfw.x11.xlib.OpenIM = (PFN_XOpenIM)
1425 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XOpenIM");
1426 _glfw.x11.xlib.PeekEvent = (PFN_XPeekEvent)
1427 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XPeekEvent");
1428 _glfw.x11.xlib.Pending = (PFN_XPending)
1429 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XPending");
1430 _glfw.x11.xlib.QueryExtension = (PFN_XQueryExtension)
1431 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XQueryExtension");
1432 _glfw.x11.xlib.QueryPointer = (PFN_XQueryPointer)
1433 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XQueryPointer");
1434 _glfw.x11.xlib.RaiseWindow = (PFN_XRaiseWindow)
1435 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XRaiseWindow");
1436 _glfw.x11.xlib.RegisterIMInstantiateCallback = (PFN_XRegisterIMInstantiateCallback)
1437 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XRegisterIMInstantiateCallback");
1438 _glfw.x11.xlib.ResizeWindow = (PFN_XResizeWindow)
1439 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XResizeWindow");
1440 _glfw.x11.xlib.ResourceManagerString = (PFN_XResourceManagerString)
1441 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XResourceManagerString");
1442 _glfw.x11.xlib.SaveContext = (PFN_XSaveContext)
1443 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSaveContext");
1444 _glfw.x11.xlib.SelectInput = (PFN_XSelectInput)
1445 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSelectInput");
1446 _glfw.x11.xlib.SendEvent = (PFN_XSendEvent)
1447 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSendEvent");
1448 _glfw.x11.xlib.SetClassHint = (PFN_XSetClassHint)
1449 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetClassHint");
1450 _glfw.x11.xlib.SetErrorHandler = (PFN_XSetErrorHandler)
1451 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetErrorHandler");
1452 _glfw.x11.xlib.SetICFocus = (PFN_XSetICFocus)
1453 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetICFocus");
1454 _glfw.x11.xlib.SetIMValues = (PFN_XSetIMValues)
1455 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetIMValues");
1456 _glfw.x11.xlib.SetInputFocus = (PFN_XSetInputFocus)
1457 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetInputFocus");
1458 _glfw.x11.xlib.SetLocaleModifiers = (PFN_XSetLocaleModifiers)
1459 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetLocaleModifiers");
1460 _glfw.x11.xlib.SetScreenSaver = (PFN_XSetScreenSaver)
1461 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetScreenSaver");
1462 _glfw.x11.xlib.SetSelectionOwner = (PFN_XSetSelectionOwner)
1463 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetSelectionOwner");
1464 _glfw.x11.xlib.SetWMHints = (PFN_XSetWMHints)
1465 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetWMHints");
1466 _glfw.x11.xlib.SetWMNormalHints = (PFN_XSetWMNormalHints)
1467 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetWMNormalHints");
1468 _glfw.x11.xlib.SetWMProtocols = (PFN_XSetWMProtocols)
1469 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSetWMProtocols");
1470 _glfw.x11.xlib.SupportsLocale = (PFN_XSupportsLocale)
1471 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSupportsLocale");
1472 _glfw.x11.xlib.Sync = (PFN_XSync)
1473 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XSync");
1474 _glfw.x11.xlib.TranslateCoordinates = (PFN_XTranslateCoordinates)
1475 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XTranslateCoordinates");
1476 _glfw.x11.xlib.UndefineCursor = (PFN_XUndefineCursor)
1477 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XUndefineCursor");
1478 _glfw.x11.xlib.UngrabPointer = (PFN_XUngrabPointer)
1479 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XUngrabPointer");
1480 _glfw.x11.xlib.UnmapWindow = (PFN_XUnmapWindow)
1481 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XUnmapWindow");
1482 _glfw.x11.xlib.UnsetICFocus = (PFN_XUnsetICFocus)
1483 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XUnsetICFocus");
1484 _glfw.x11.xlib.VisualIDFromVisual = (PFN_XVisualIDFromVisual)
1485 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XVisualIDFromVisual");
1486 _glfw.x11.xlib.WarpPointer = (PFN_XWarpPointer)
1487 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XWarpPointer");
1488 _glfw.x11.xkb.FreeKeyboard = (PFN_XkbFreeKeyboard)
1489 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbFreeKeyboard");
1490 _glfw.x11.xkb.FreeNames = (PFN_XkbFreeNames)
1491 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbFreeNames");
1492 _glfw.x11.xkb.GetMap = (PFN_XkbGetMap)
1493 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbGetMap");
1494 _glfw.x11.xkb.GetNames = (PFN_XkbGetNames)
1495 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbGetNames");
1496 _glfw.x11.xkb.GetState = (PFN_XkbGetState)
1497 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbGetState");
1498 _glfw.x11.xkb.KeycodeToKeysym = (PFN_XkbKeycodeToKeysym)
1499 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbKeycodeToKeysym");
1500 _glfw.x11.xkb.QueryExtension = (PFN_XkbQueryExtension)
1501 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbQueryExtension");
1502 _glfw.x11.xkb.SelectEventDetails = (PFN_XkbSelectEventDetails)
1503 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbSelectEventDetails");
1504 _glfw.x11.xkb.SetDetectableAutoRepeat = (PFN_XkbSetDetectableAutoRepeat)
1505 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XkbSetDetectableAutoRepeat");
1506 _glfw.x11.xrm.DestroyDatabase = (PFN_XrmDestroyDatabase)
1507 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XrmDestroyDatabase");
1508 _glfw.x11.xrm.GetResource = (PFN_XrmGetResource)
1509 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XrmGetResource");
1510 _glfw.x11.xrm.GetStringDatabase = (PFN_XrmGetStringDatabase)
1511 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XrmGetStringDatabase");
1512 _glfw.x11.xrm.UniqueQuark = (PFN_XrmUniqueQuark)
1513 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XrmUniqueQuark");
1514 _glfw.x11.xlib.UnregisterIMInstantiateCallback = (PFN_XUnregisterIMInstantiateCallback)
1515 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "XUnregisterIMInstantiateCallback");
1516 _glfw.x11.xlib.utf8LookupString = (PFN_Xutf8LookupString)
1517 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "Xutf8LookupString");
1518 _glfw.x11.xlib.utf8SetWMProperties = (PFN_Xutf8SetWMProperties)
1519 _glfwPlatformGetModuleSymbol(_glfw.x11.xlib.handle, "Xutf8SetWMProperties");
1520
1521 if (_glfw.x11.xlib.utf8LookupString && _glfw.x11.xlib.utf8SetWMProperties)
1522 _glfw.x11.xlib.utf8 = GLFW_TRUE;
1523
1524 _glfw.x11.screen = DefaultScreen(_glfw.x11.display);
1525 _glfw.x11.root = RootWindow(_glfw.x11.display, _glfw.x11.screen);
1526 _glfw.x11.context = XUniqueContext();
1527
1528 getSystemContentScale(&_glfw.x11.contentScaleX, &_glfw.x11.contentScaleY);
1529
1530 if (!createEmptyEventPipe())
1531 return GLFW_FALSE;
1532
1533 if (!initExtensions())
1534 return GLFW_FALSE;
1535
1536 _glfw.x11.helperWindowHandle = createHelperWindow();
1537 _glfw.x11.hiddenCursorHandle = createHiddenCursor();
1538
1539 if (XSupportsLocale() && _glfw.x11.xlib.utf8)
1540 {
1541 XSetLocaleModifiers("");
1542
1543 // If an IM is already present our callback will be called right away
1544 XRegisterIMInstantiateCallback(_glfw.x11.display,
1545 NULL, NULL, NULL,
1546 inputMethodInstantiateCallback,
1547 NULL);
1548 }
1549
1550 _glfwPollMonitorsX11();
1551 return GLFW_TRUE;
1552 }
1553
1554 void _glfwTerminateX11(void)
1555 {
1556 if (_glfw.x11.helperWindowHandle)
1557 {
1558 if (XGetSelectionOwner(_glfw.x11.display, _glfw.x11.CLIPBOARD) ==
1559 _glfw.x11.helperWindowHandle)
1560 {
1561 _glfwPushSelectionToManagerX11();
1562 }
1563
1564 XDestroyWindow(_glfw.x11.display, _glfw.x11.helperWindowHandle);
1565 _glfw.x11.helperWindowHandle = None;
1566 }
1567
1568 if (_glfw.x11.hiddenCursorHandle)
1569 {
1570 XFreeCursor(_glfw.x11.display, _glfw.x11.hiddenCursorHandle);
1571 _glfw.x11.hiddenCursorHandle = (Cursor) 0;
1572 }
1573
1574 _glfw_free(_glfw.x11.primarySelectionString);
1575 _glfw_free(_glfw.x11.clipboardString);
1576
1577 XUnregisterIMInstantiateCallback(_glfw.x11.display,
1578 NULL, NULL, NULL,
1579 inputMethodInstantiateCallback,
1580 NULL);
1581
1582 if (_glfw.x11.im)
1583 {
1584 XCloseIM(_glfw.x11.im);
1585 _glfw.x11.im = NULL;
1586 }
1587
1588 if (_glfw.x11.display)
1589 {
1590 XCloseDisplay(_glfw.x11.display);
1591 _glfw.x11.display = NULL;
1592 }
1593
1594 if (_glfw.x11.x11xcb.handle)
1595 {
1596 _glfwPlatformFreeModule(_glfw.x11.x11xcb.handle);
1597 _glfw.x11.x11xcb.handle = NULL;
1598 }
1599
1600 if (_glfw.x11.xcursor.handle)
1601 {
1602 _glfwPlatformFreeModule(_glfw.x11.xcursor.handle);
1603 _glfw.x11.xcursor.handle = NULL;
1604 }
1605
1606 if (_glfw.x11.randr.handle)
1607 {
1608 _glfwPlatformFreeModule(_glfw.x11.randr.handle);
1609 _glfw.x11.randr.handle = NULL;
1610 }
1611
1612 if (_glfw.x11.xinerama.handle)
1613 {
1614 _glfwPlatformFreeModule(_glfw.x11.xinerama.handle);
1615 _glfw.x11.xinerama.handle = NULL;
1616 }
1617
1618 if (_glfw.x11.xrender.handle)
1619 {
1620 _glfwPlatformFreeModule(_glfw.x11.xrender.handle);
1621 _glfw.x11.xrender.handle = NULL;
1622 }
1623
1624 if (_glfw.x11.vidmode.handle)
1625 {
1626 _glfwPlatformFreeModule(_glfw.x11.vidmode.handle);
1627 _glfw.x11.vidmode.handle = NULL;
1628 }
1629
1630 if (_glfw.x11.xi.handle)
1631 {
1632 _glfwPlatformFreeModule(_glfw.x11.xi.handle);
1633 _glfw.x11.xi.handle = NULL;
1634 }
1635
1636 _glfwTerminateOSMesa();
1637 // NOTE: These need to be unloaded after XCloseDisplay, as they register
1638 // cleanup callbacks that get called by that function
1639 _glfwTerminateEGL();
1640 _glfwTerminateGLX();
1641
1642 if (_glfw.x11.xlib.handle)
1643 {
1644 _glfwPlatformFreeModule(_glfw.x11.xlib.handle);
1645 _glfw.x11.xlib.handle = NULL;
1646 }
1647
1648 if (_glfw.x11.emptyEventPipe[0] || _glfw.x11.emptyEventPipe[1])
1649 {
1650 close(_glfw.x11.emptyEventPipe[0]);
1651 close(_glfw.x11.emptyEventPipe[1]);
1652 }
1653 }
1654
1655