2009-01-24 17:57:26 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2009 Marco Peereboom <marco@peereboom.us>
|
|
|
|
* Copyright (c) 2009 Ryan McBride <mcbride@countersiege.com>
|
2018-01-19 04:14:15 +08:00
|
|
|
* Copyright (c) 2011-2018 Reginald Kennedy <rk@rejii.com>
|
2009-01-24 17:57:26 +00:00
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2005-2007 Carsten Haitzler
|
|
|
|
* Copyright (C) 2006-2007 Kim Woelders
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to
|
|
|
|
* deal in the Software without restriction, including without limitation the
|
|
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies of the Software, its documentation and marketing & publicity
|
|
|
|
* materials, and acknowledgment shall be given in the documentation, materials
|
|
|
|
* and software packages that this Software was used.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Basic hack mechanism (dlopen etc.) taken from e_hack.c in e17.
|
|
|
|
*/
|
|
|
|
#include <string.h>
|
2015-05-21 02:40:54 +08:00
|
|
|
#include <stdbool.h>
|
2009-01-24 17:57:26 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/X.h>
|
|
|
|
#include <X11/Xatom.h>
|
2009-02-07 19:49:58 +00:00
|
|
|
#include <X11/Intrinsic.h>
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2009-02-07 19:49:58 +00:00
|
|
|
/* dlopened libs so we can find the symbols in the real one to call them */
|
|
|
|
static void *lib_xlib = NULL;
|
|
|
|
static void *lib_xtlib = NULL;
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2015-05-21 02:40:54 +08:00
|
|
|
static bool xterm = false;
|
2012-07-28 04:48:27 +08:00
|
|
|
static Display *display = NULL;
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
static Atom swmws = None, swmpid = None;
|
|
|
|
|
|
|
|
void set_property(Display *, Window, Atom, char *);
|
|
|
|
Atom get_atom_from_string(Display *, char *);
|
2012-08-29 08:35:50 +08:00
|
|
|
|
2019-10-18 08:08:00 +08:00
|
|
|
#if defined(_GNU_SOURCE) && !defined(__CYGWIN__)
|
2015-05-21 02:40:54 +08:00
|
|
|
#define DLOPEN(s) RTLD_NEXT
|
|
|
|
#else
|
|
|
|
#define DLOPEN(s) dlopen((s), RTLD_GLOBAL | RTLD_LAZY)
|
|
|
|
#endif
|
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
typedef Atom (XIA)(Display *_display, char *atom_name, Bool only_if_exists);
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
Atom
|
|
|
|
get_atom_from_string(Display *dpy, char *name)
|
|
|
|
{
|
|
|
|
Atom atom = None;
|
|
|
|
static XIA *xia = NULL;
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
if (lib_xlib == NULL)
|
|
|
|
lib_xlib = DLOPEN("libX11.so");
|
|
|
|
if (lib_xlib) {
|
|
|
|
if (xia == NULL)
|
|
|
|
xia = (XIA *) dlsym(lib_xlib, "XInternAtom");
|
|
|
|
}
|
|
|
|
if (xia == NULL) {
|
|
|
|
fprintf(stderr, "libswmhack.so: ERROR: %s\n", dlerror());
|
2018-03-28 02:33:00 +08:00
|
|
|
return (atom);
|
2018-01-19 04:14:15 +08:00
|
|
|
}
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
atom = (*xia)(dpy, name, False);
|
2018-03-28 02:33:00 +08:00
|
|
|
return (atom);
|
2009-01-24 17:57:26 +00:00
|
|
|
}
|
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
typedef int (XCP)(Display *_display, Window w, Atom property, Atom type,
|
|
|
|
int format, int mode, unsigned char *data, int nelements);
|
2011-06-03 22:51:54 +00:00
|
|
|
|
2009-01-24 17:57:26 +00:00
|
|
|
#define SWM_PROPLEN (16)
|
|
|
|
void
|
2018-01-19 04:14:15 +08:00
|
|
|
set_property(Display *dpy, Window id, Atom atom, char *val)
|
2009-01-24 17:57:26 +00:00
|
|
|
{
|
|
|
|
char prop[SWM_PROPLEN];
|
2011-09-19 22:01:00 -05:00
|
|
|
static XCP *xcp = NULL;
|
2011-06-03 22:51:54 +00:00
|
|
|
|
2015-05-21 02:40:54 +08:00
|
|
|
if (lib_xlib == NULL)
|
|
|
|
lib_xlib = DLOPEN("libX11.so");
|
|
|
|
if (lib_xlib) {
|
|
|
|
if (xcp == NULL)
|
|
|
|
xcp = (XCP *) dlsym(lib_xlib, "XChangeProperty");
|
|
|
|
}
|
2018-01-19 04:14:15 +08:00
|
|
|
if (xcp == NULL) {
|
2015-05-06 20:36:37 +02:00
|
|
|
fprintf(stderr, "libswmhack.so: ERROR: %s\n", dlerror());
|
|
|
|
return;
|
|
|
|
}
|
2011-06-03 22:51:54 +00:00
|
|
|
|
2009-01-24 17:57:26 +00:00
|
|
|
/* Try to update the window's workspace property */
|
2011-09-19 22:01:00 -05:00
|
|
|
if (atom)
|
2009-01-24 17:57:26 +00:00
|
|
|
if (snprintf(prop, SWM_PROPLEN, "%s", val) < SWM_PROPLEN)
|
2011-06-03 22:51:54 +00:00
|
|
|
(*xcp)(dpy, id, atom, XA_STRING,
|
2012-02-13 23:32:31 +08:00
|
|
|
8, PropModeReplace, (unsigned char *)prop,
|
|
|
|
strlen((char *)prop));
|
2009-01-24 17:57:26 +00:00
|
|
|
}
|
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
typedef Display *(ODF)(register _Xconst char *_display);
|
|
|
|
|
|
|
|
/* XOpenDisplay intercept hack */
|
|
|
|
Display *
|
|
|
|
XOpenDisplay(register _Xconst char *_display)
|
|
|
|
{
|
|
|
|
static ODF *func = NULL;
|
|
|
|
|
|
|
|
if (lib_xlib == NULL)
|
|
|
|
lib_xlib = DLOPEN("libX11.so");
|
|
|
|
if (lib_xlib && func == NULL)
|
|
|
|
func = (ODF *) dlsym(lib_xlib, "XOpenDisplay");
|
|
|
|
if (func == NULL) {
|
|
|
|
fprintf(stderr, "libswmhack.so: ERROR: %s\n", dlerror());
|
2018-03-28 02:33:00 +08:00
|
|
|
return (None);
|
2018-01-19 04:14:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
display = (*func) (_display);
|
|
|
|
|
|
|
|
/* Preload atoms to prevent deadlock. */
|
|
|
|
if (swmws == None)
|
|
|
|
swmws = get_atom_from_string(display, "_SWM_WS");
|
|
|
|
if (swmpid == None)
|
|
|
|
swmpid = get_atom_from_string(display, "_SWM_PID");
|
|
|
|
|
2018-03-28 02:33:00 +08:00
|
|
|
return (display);
|
2018-01-19 04:14:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
typedef Window (CWF)(Display * _display, Window _parent, int _x, int _y,
|
|
|
|
unsigned int _width, unsigned int _height, unsigned int _border_width,
|
|
|
|
int _depth, unsigned int _class, Visual * _visual, unsigned long _valuemask,
|
|
|
|
XSetWindowAttributes * _attributes);
|
2009-01-24 17:57:26 +00:00
|
|
|
|
|
|
|
/* XCreateWindow intercept hack */
|
|
|
|
Window
|
2018-01-19 04:14:15 +08:00
|
|
|
XCreateWindow(Display *dpy, Window parent, int x, int y, unsigned int width,
|
|
|
|
unsigned int height, unsigned int border_width, int depth,
|
|
|
|
unsigned int clss, Visual * visual, unsigned long valuemask,
|
|
|
|
XSetWindowAttributes * attributes)
|
2009-01-24 17:57:26 +00:00
|
|
|
{
|
|
|
|
static CWF *func = NULL;
|
|
|
|
char *env;
|
|
|
|
Window id;
|
|
|
|
|
2015-05-21 02:40:54 +08:00
|
|
|
if (lib_xlib == NULL)
|
|
|
|
lib_xlib = DLOPEN("libX11.so");
|
2018-01-19 04:14:15 +08:00
|
|
|
if (lib_xlib && func == NULL)
|
2009-01-24 17:57:26 +00:00
|
|
|
func = (CWF *) dlsym(lib_xlib, "XCreateWindow");
|
2015-05-21 02:40:54 +08:00
|
|
|
if (func == NULL) {
|
2015-05-06 20:36:37 +02:00
|
|
|
fprintf(stderr, "libswmhack.so: ERROR: %s\n", dlerror());
|
2018-03-28 02:33:00 +08:00
|
|
|
return (None);
|
2015-05-06 20:36:37 +02:00
|
|
|
}
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2012-08-24 23:33:15 +08:00
|
|
|
id = (*func) (dpy, parent, x, y, width, height, border_width,
|
2009-01-24 17:57:26 +00:00
|
|
|
depth, clss, visual, valuemask, attributes);
|
|
|
|
|
|
|
|
if (id) {
|
2015-05-21 02:40:54 +08:00
|
|
|
if ((env = getenv("_SWM_WS")) != NULL)
|
2018-01-19 04:14:15 +08:00
|
|
|
set_property(dpy, id, swmws, env);
|
2009-01-24 17:57:26 +00:00
|
|
|
if ((env = getenv("_SWM_PID")) != NULL)
|
2018-01-19 04:14:15 +08:00
|
|
|
set_property(dpy, id, swmpid, env);
|
2019-11-22 21:01:20 +01:00
|
|
|
if (getenv("_SWM_XTERM_FONTADJ") != NULL) {
|
2009-02-07 19:49:58 +00:00
|
|
|
unsetenv("_SWM_XTERM_FONTADJ");
|
2015-05-21 02:40:54 +08:00
|
|
|
xterm = true;
|
2009-02-07 19:49:58 +00:00
|
|
|
}
|
2009-01-24 17:57:26 +00:00
|
|
|
}
|
|
|
|
return (id);
|
|
|
|
}
|
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
typedef Window (CSWF)(Display * _display, Window _parent, int _x, int _y,
|
|
|
|
unsigned int _width, unsigned int _height, unsigned int _border_width,
|
|
|
|
unsigned long _border, unsigned long _background);
|
2009-01-24 17:57:26 +00:00
|
|
|
|
|
|
|
/* XCreateSimpleWindow intercept hack */
|
|
|
|
Window
|
2012-07-28 04:48:27 +08:00
|
|
|
XCreateSimpleWindow(Display *dpy, Window parent, int x, int y,
|
2018-01-19 04:14:15 +08:00
|
|
|
unsigned int width, unsigned int height, unsigned int border_width,
|
2009-01-24 17:57:26 +00:00
|
|
|
unsigned long border, unsigned long background)
|
|
|
|
{
|
|
|
|
static CSWF *func = NULL;
|
|
|
|
char *env;
|
|
|
|
Window id;
|
|
|
|
|
2015-05-21 02:40:54 +08:00
|
|
|
if (lib_xlib == NULL)
|
|
|
|
lib_xlib = DLOPEN("libX11.so");
|
|
|
|
if (lib_xlib && func == NULL)
|
2009-01-24 17:57:26 +00:00
|
|
|
func = (CSWF *) dlsym(lib_xlib, "XCreateSimpleWindow");
|
2015-05-21 02:40:54 +08:00
|
|
|
if (func == NULL) {
|
2015-05-06 20:36:37 +02:00
|
|
|
fprintf(stderr, "libswmhack.so: ERROR: %s\n", dlerror());
|
2018-03-28 02:33:00 +08:00
|
|
|
return (None);
|
2015-05-06 20:36:37 +02:00
|
|
|
}
|
2009-01-24 17:57:26 +00:00
|
|
|
|
2012-07-28 04:48:27 +08:00
|
|
|
id = (*func) (dpy, parent, x, y, width, height,
|
2009-01-24 17:57:26 +00:00
|
|
|
border_width, border, background);
|
|
|
|
|
|
|
|
if (id) {
|
2015-05-21 02:40:54 +08:00
|
|
|
if ((env = getenv("_SWM_WS")) != NULL)
|
2018-01-19 04:14:15 +08:00
|
|
|
set_property(dpy, id, swmws, env);
|
2009-01-24 17:57:26 +00:00
|
|
|
if ((env = getenv("_SWM_PID")) != NULL)
|
2018-01-19 04:14:15 +08:00
|
|
|
set_property(dpy, id, swmpid, env);
|
2019-11-22 21:01:20 +01:00
|
|
|
if (getenv("_SWM_XTERM_FONTADJ") != NULL) {
|
2009-02-07 19:49:58 +00:00
|
|
|
unsetenv("_SWM_XTERM_FONTADJ");
|
2015-05-21 02:40:54 +08:00
|
|
|
xterm = true;
|
2009-02-07 19:49:58 +00:00
|
|
|
}
|
2009-01-24 17:57:26 +00:00
|
|
|
}
|
|
|
|
return (id);
|
|
|
|
}
|
|
|
|
|
2018-01-19 04:14:15 +08:00
|
|
|
typedef void (ANEF)(XtAppContext app_context, XEvent *event_return);
|
2009-02-07 19:49:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* XtAppNextEvent Intercept Hack
|
|
|
|
* Normally xterm rejects "synthetic" (XSendEvent) events to prevent spoofing.
|
|
|
|
* We don't want to disable this completely, it's insecure. But hook here
|
|
|
|
* and allow these mostly harmless ones that we use to adjust fonts.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
XtAppNextEvent(XtAppContext app_context, XEvent *event_return)
|
|
|
|
{
|
|
|
|
static ANEF *func = NULL;
|
2012-07-20 01:01:04 +08:00
|
|
|
static KeyCode kp_add = 0, kp_subtract = 0;
|
2009-02-07 19:49:58 +00:00
|
|
|
|
2015-05-21 02:40:54 +08:00
|
|
|
if (lib_xtlib == NULL)
|
|
|
|
lib_xtlib = DLOPEN("libXt.so");
|
|
|
|
if (lib_xtlib && func == NULL) {
|
2009-02-07 19:49:58 +00:00
|
|
|
func = (ANEF *) dlsym(lib_xtlib, "XtAppNextEvent");
|
2015-05-21 02:40:54 +08:00
|
|
|
if (display) {
|
2012-07-28 04:48:27 +08:00
|
|
|
kp_add = XKeysymToKeycode(display, XK_KP_Add);
|
|
|
|
kp_subtract = XKeysymToKeycode(display, XK_KP_Subtract);
|
2009-02-07 19:49:58 +00:00
|
|
|
}
|
|
|
|
}
|
2015-05-21 02:40:54 +08:00
|
|
|
if (func == NULL) {
|
2015-05-06 20:36:37 +02:00
|
|
|
fprintf(stderr, "libswmhack.so: ERROR: %s\n", dlerror());
|
|
|
|
return;
|
|
|
|
}
|
2009-02-07 19:49:58 +00:00
|
|
|
|
|
|
|
(*func) (app_context, event_return);
|
|
|
|
|
|
|
|
/* Return here if it's not an Xterm. */
|
|
|
|
if (!xterm)
|
|
|
|
return;
|
2015-05-21 02:40:54 +08:00
|
|
|
|
2009-02-07 19:49:58 +00:00
|
|
|
/* Allow spoofing of font change keystrokes. */
|
2015-05-21 02:40:54 +08:00
|
|
|
if ((event_return->type == KeyPress ||
|
2009-02-07 19:49:58 +00:00
|
|
|
event_return->type == KeyRelease) &&
|
|
|
|
event_return->xkey.state == ShiftMask &&
|
|
|
|
(event_return->xkey.keycode == kp_add ||
|
|
|
|
event_return->xkey.keycode == kp_subtract))
|
|
|
|
event_return->xkey.send_event = 0;
|
|
|
|
}
|