2 X11 support for the Midnight Commander.
4 Copyright (C) 2005-2023
5 Free Software Foundation, Inc.
8 Roland Illig <roland.illig@gmx.de>, 2005.
10 This file is part of the Midnight Commander.
12 The Midnight Commander is free software: you can redistribute it
13 and/or modify it under the terms of the GNU General Public License as
14 published by the Free Software Foundation, either version 3 of the License,
15 or (at your option) any later version.
17 The Midnight Commander is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 GNU General Public License for more details.
22 You should have received a copy of the GNU General Public License
23 along with this program. If not, see <http://www.gnu.org/licenses/>.
27 * \brief Source: X11 support
28 * \warning This code uses setjmp() and longjmp(). Before you modify _anything_ here,
29 * please read the relevant sections of the C standard.
40 #include "lib/global.h"
43 /*** global variables ****************************************************************************/
45 /*** file scope macro definitions ****************************************************************/
48 #define func_XOpenDisplay XOpenDisplay
49 #define func_XCloseDisplay XCloseDisplay
50 #define func_XSetErrorHandler XSetErrorHandler
51 #define func_XSetIOErrorHandler XSetIOErrorHandler
52 #define func_XQueryPointer XQueryPointer
55 /*** file scope type declarations ****************************************************************/
57 typedef int (*mc_XErrorHandler_callback
) (Display
*, XErrorEvent
*);
58 typedef int (*mc_XIOErrorHandler_callback
) (Display
*);
60 /*** forward declarations (file scope functions) *************************************************/
62 /*** file scope variables ************************************************************************/
65 static Display
*(*func_XOpenDisplay
) (_Xconst
char *);
66 static int (*func_XCloseDisplay
) (Display
*);
67 static mc_XErrorHandler_callback (*func_XSetErrorHandler
) (mc_XErrorHandler_callback
);
68 static mc_XIOErrorHandler_callback (*func_XSetIOErrorHandler
) (mc_XIOErrorHandler_callback
);
69 static Bool (*func_XQueryPointer
) (Display
*, Window
, Window
*, Window
*,
70 int *, int *, int *, int *, unsigned int *);
72 static GModule
*x11_module
;
75 static gboolean handlers_installed
= FALSE
;
77 /* This flag is set as soon as an X11 error is reported. Usually that
78 * means that the DISPLAY is not available anymore. We do not try to
79 * reconnect, as that would violate the X11 protocol. */
80 static gboolean lost_connection
= FALSE
;
82 static jmp_buf x11_exception
; /* FIXME: get a better name */
83 static gboolean longjmp_allowed
= FALSE
;
85 /* --------------------------------------------------------------------------------------------- */
86 /*** file scope functions ************************************************************************/
87 /* --------------------------------------------------------------------------------------------- */
90 x_io_error_handler (Display
* dpy
)
94 lost_connection
= TRUE
;
97 longjmp_allowed
= FALSE
;
98 longjmp (x11_exception
, 1);
103 /* --------------------------------------------------------------------------------------------- */
106 x_error_handler (Display
* dpy
, XErrorEvent
* ee
)
109 (void) func_XCloseDisplay (dpy
);
110 return x_io_error_handler (dpy
);
113 /* --------------------------------------------------------------------------------------------- */
116 install_error_handlers (void)
118 if (handlers_installed
)
121 (void) func_XSetErrorHandler (x_error_handler
);
122 (void) func_XSetIOErrorHandler (x_io_error_handler
);
123 handlers_installed
= TRUE
;
126 /* --------------------------------------------------------------------------------------------- */
132 gchar
*x11_module_fname
;
137 if (x11_module
!= NULL
)
140 x11_module_fname
= g_module_build_path (NULL
, "X11");
141 x11_module
= g_module_open (x11_module_fname
, G_MODULE_BIND_LAZY
);
142 if (x11_module
== NULL
)
143 x11_module
= g_module_open ("libX11.so.6", G_MODULE_BIND_LAZY
);
145 g_free (x11_module_fname
);
147 if (x11_module
== NULL
)
150 if (!g_module_symbol (x11_module
, "XOpenDisplay", (void *) &func_XOpenDisplay
))
152 if (!g_module_symbol (x11_module
, "XCloseDisplay", (void *) &func_XCloseDisplay
))
154 if (!g_module_symbol (x11_module
, "XQueryPointer", (void *) &func_XQueryPointer
))
156 if (!g_module_symbol (x11_module
, "XSetErrorHandler", (void *) &func_XSetErrorHandler
))
158 if (!g_module_symbol (x11_module
, "XSetIOErrorHandler", (void *) &func_XSetIOErrorHandler
))
161 install_error_handlers ();
165 func_XOpenDisplay
= 0;
166 func_XCloseDisplay
= 0;
167 func_XQueryPointer
= 0;
168 func_XSetErrorHandler
= 0;
169 func_XSetIOErrorHandler
= 0;
170 g_module_close (x11_module
);
174 install_error_handlers ();
175 return !(lost_connection
);
179 /* --------------------------------------------------------------------------------------------- */
180 /*** public functions ****************************************************************************/
181 /* --------------------------------------------------------------------------------------------- */
184 mc_XOpenDisplay (const char *displayname
)
186 if (x11_available ())
188 if (setjmp (x11_exception
) == 0)
192 /* cppcheck-suppress redundantAssignment */
193 longjmp_allowed
= TRUE
;
195 retval
= func_XOpenDisplay (displayname
);
197 /* cppcheck-suppress redundantAssignment */
198 longjmp_allowed
= FALSE
;
205 /* --------------------------------------------------------------------------------------------- */
208 mc_XCloseDisplay (Display
* display
)
210 if (x11_available ())
212 if (setjmp (x11_exception
) == 0)
216 /* cppcheck-suppress redundantAssignment */
217 longjmp_allowed
= TRUE
;
219 retval
= func_XCloseDisplay (display
);
221 /* cppcheck-suppress redundantAssignment */
222 longjmp_allowed
= FALSE
;
230 /* --------------------------------------------------------------------------------------------- */
233 mc_XQueryPointer (Display
* display
, Window win
, Window
* root_return
,
234 Window
* child_return
, int *root_x_return
, int *root_y_return
,
235 int *win_x_return
, int *win_y_return
, unsigned int *mask_return
)
239 if (x11_available ())
241 if (setjmp (x11_exception
) == 0)
243 /* cppcheck-suppress redundantAssignment */
244 longjmp_allowed
= TRUE
;
246 retval
= func_XQueryPointer (display
, win
, root_return
,
247 child_return
, root_x_return
, root_y_return
,
248 win_x_return
, win_y_return
, mask_return
);
250 /* cppcheck-suppress redundantAssignment */
251 longjmp_allowed
= FALSE
;
257 *child_return
= None
;
266 /* --------------------------------------------------------------------------------------------- */