-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathgen-x11
executable file
·204 lines (180 loc) · 9.77 KB
/
gen-x11
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
#!/usr/bin/env python
import textwrap
import re
# ./build 2>&1 | grep -i 'undefined reference' | awk -F'`|'"'" '{print $2}' | sort -u
FUNCTIONS = {
"X11": [
"Display *XOpenDisplay(const char *display_name)",
"int XCloseDisplay(Display *display)",
"int XFlush(Display *display)",
"int XNextEvent(Display *display, XEvent *event_return)",
"int XPending(Display *display)",
"Status XInitThreads()",
"Bool XFilterEvent(XEvent *event, Window w)",
"Bool XkbSetDetectableAutoRepeat(Display *display, Bool detectable, Bool *supported_rtrn)",
"int XSync(Display *display, Bool discard)",
"void XrmInitialize()",
"int XChangeProperty(Display *display, Window w, Atom property, Atom type, int format, int mode, const unsigned char *data, int nelements)",
"Status XSendEvent(Display *display, Window w, Bool propagate, long event_mask, XEvent *event_send)",
"int XFree(void *data)",
"XErrorHandler XSetErrorHandler(XErrorHandler handler)",
"int XConvertSelection(Display *display, Atom selection, Atom target, Atom property, Window requestor, Time time)",
"int XLookupString(XKeyEvent *event_struct, char *buffer_return, int bytes_buffer, KeySym *keysym_return, XComposeStatus *status_in_out)",
"Bool XGetEventData(Display *display, XGenericEventCookie *cookie)",
"void XFreeEventData(Display *display, XGenericEventCookie *cookie)",
"int XGetWindowProperty(Display *display, Window w, Atom property, long long_offset, long long_length, Bool delete, Atom req_type, Atom *actual_type_return, int *actual_format_return, unsigned long *nitems_return, unsigned long *bytes_after_return, unsigned char **prop_return)",
"int XMapWindow(Display *display, Window w)",
"int XUnmapWindow(Display *display, Window w)",
"int XRaiseWindow(Display *display, Window w)",
"Status XGetWindowAttributes(Display *display, Window w, XWindowAttributes *window_attributes_return)",
"XSizeHints *XAllocSizeHints()",
"Bool XCheckTypedWindowEvent(Display *display, Window w, int event_type, XEvent *event_return)",
"Colormap XCreateColormap(Display *display, Window w, Visual *visual, int alloc)",
"Cursor XCreateFontCursor(Display *display, unsigned int shape)",
"Window XCreateWindow(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)",
"int XWarpPointer(Display *display, Window src_w, Window dest_w, int src_x, int src_y, unsigned int src_width, unsigned int src_height, int dest_x, int dest_y)",
"int XDefineCursor(Display *display, Window w, Cursor cursor)",
"int XDestroyWindow(Display *display, Window w)",
"int XFreeColormap(Display *display, Colormap colormap)",
"int XFreeCursor(Display *display, Cursor cursor)",
"KeySym *XGetKeyboardMapping(Display *display, unsigned int first_keycode, int keycode_count, int *keysyms_per_keycode_return)",
"Window XGetSelectionOwner(Display *display, Atom selection)",
"int XGrabPointer(Display *display, Window grab_window, Bool owner_events, unsigned int event_mask, int pointer_mode, int keyboard_mode, Window confine_to, Cursor cursor, Time time)",
"Atom XInternAtom(Display *display, const char *atom_name, Bool only_if_exists)",
"Status XInternAtoms(Display *display, char **names, int count, Bool only_if_exists, Atom *atoms_return)",
"int XSetSelectionOwner(Display *display, Atom selection, Window owner, Time time)",
"void XSetWMNormalHints(Display *display, Window w, XSizeHints *hints)",
"Status XSetWMProtocols(Display *display, Window w, Atom *protocols, int count)",
"int XUndefineCursor(Display *display, Window w)",
"int XUngrabPointer(Display *display, Time time)",
"void Xutf8SetWMProperties(Display *display, Window w, const char *window_name, const char *icon_name, char **argv, int argc, XSizeHints *normal_hints, XWMHints *wm_hints, XClassHint *class_hints)",
"void XkbFreeKeyboard(XkbDescPtr xkb, unsigned int which, Bool free_all)",
"void XkbFreeNames(XkbDescPtr xkb, unsigned int which, Bool free_map)",
"char *XResourceManagerString(Display *display)",
"void XrmDestroyDatabase(XrmDatabase database)",
"Bool XrmGetResource(XrmDatabase database, const char *str_name, const char *str_class, char **str_type_return, XrmValue *value_return)",
"XkbDescPtr XkbGetMap(Display *display, unsigned int which, unsigned int device_spec)",
"Status XkbGetNames(Display *dpy, unsigned int which, XkbDescPtr xkb)",
"XrmDatabase XrmGetStringDatabase(const char *data)",
"Bool XQueryExtension(Display *display, const char* name, int *major_opcode_return, int* first_event_return, int* first_error_return)",
],
"Xcursor": [
"int XcursorGetDefaultSize(Display *dpy)",
"char *XcursorGetTheme(Display *dpy)",
"XcursorImage *XcursorImageCreate(int width, int height)",
"void XcursorImageDestroy(XcursorImage *image)",
"Cursor XcursorImageLoadCursor(Display *dpy, const XcursorImage *image)",
"XcursorImage *XcursorLibraryLoadImage(const char *library, const char *theme, int size)",
],
"Xi": [
"Status XIQueryVersion(Display* dpy, int* major_version_inout, int* minor_version_inout)",
"int XISelectEvents(Display* dpy, Window win, XIEventMask *masks, int num_masks)",
],
}
def main():
lib_specs = {
lib: [
parse_c_signature(function)
for function in functions
]
for lib, functions in FUNCTIONS.items()
}
with open("x11.c", "w") as f:
f.write(textwrap.dedent("""
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xresource.h>
#include <X11/Xcursor/Xcursor.h>
#include <X11/extensions/XKBstr.h>
#include <X11/extensions/XInput2.h>
#include <stddef.h>
#include <dlfcn.h>
#include <assert.h>
""").lstrip())
# Function pointers
for lib, functions in lib_specs.items():
f.write(f"\n// lib{lib}\n");
f.write(f"static void* lib{lib} = NULL;\n");
for function in functions:
f.write(
f"static {function['return_type']} "
f"(*proc_{function['name']})("
f"{arg_list(function['args'])}"
") = NULL;\n"
)
# Loader
for lib, function_specs in lib_specs.items():
f.write(textwrap.dedent(f"""
static void load_{lib}_procs(void) {{
lib{lib} = cosmo_dlopen("lib{lib}.so", RTLD_NOW | RTLD_GLOBAL);
"""))
for function_spec in function_specs:
name = function_spec['name']
f.write(f' proc_{name} = cosmo_dltramp(cosmo_dlsym(lib{lib}, "{name}"));\n')
f.write(f' assert(proc_{name} != NULL && "Could not load {name}");\n')
f.write("}\n")
# Shim
for lib, function_specs in lib_specs.items():
for function_spec in function_specs:
name = function_spec['name']
return_type = function_spec['return_type']
params = function_spec['args']
f.write(textwrap.dedent(f"""
{return_type} {name}({arg_list(params)}) {{
if (lib{lib} == NULL) {{ load_{lib}_procs(); }}
{ 'return ' if return_type != 'void' else '' }proc_{name}({forward_list(params)});
}}
"""))
def parse_c_signature(signature):
# Define a regex pattern to match a C function signature
pattern = r"(?P<return_type>[a-zA-Z_][a-zA-Z0-9_\s\*]+\s+\**)(?P<name>[a-zA-Z_][a-zA-Z0-9_]*)\s*\((?P<args>.*)\)"
# Match the pattern with the provided signature
match = re.match(pattern, signature.strip())
assert match, "Could not parse signature"
# Extract components from named groups
return_type = match.group("return_type").strip()
name = match.group("name").strip()
args = match.group("args").strip()
# Process arguments into a list, handling empty argument lists
args_list = [arg.strip() for arg in args.split(",")] if args else []
# Further split each argument into type and name
processed_args = []
for arg in args_list:
if arg: # Only process non-empty arguments
# Separate type and name for each argument
parts = arg.rsplit(" ", 1) # Split from the right to handle cases like "int *ptr"
if len(parts) == 2:
arg_type, arg_name = parts[0].strip(), parts[1].strip()
else:
# Handle cases with unnamed parameters or pointer-only types
arg_type, arg_name = parts[0].strip(), None
# Pointer arg
if arg_name.startswith('*'):
new_arg_name = arg_name.lstrip('*')
num_stars = len(arg_name) - len(new_arg_name)
arg_name = new_arg_name
arg_type = arg_type + '*' * num_stars
processed_args.append({"type": arg_type, "name": arg_name})
return {
"return_type": return_type,
"name": name,
"args": processed_args
}
def arg_list(params):
if len(params) > 0:
return ', '.join([
f"{param['type']} {param['name']}"
for param in params
])
else:
return 'void'
def forward_list(params):
if len(params) > 0:
return ', '.join([
f"{param['name']}"
for param in params
])
else:
return ''
if __name__ == "__main__":
main()