forked from FEX-Emu/FEX
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlibxcb_Host.cpp
116 lines (93 loc) · 3.06 KB
/
libxcb_Host.cpp
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
/*
$info$
tags: thunklibs|xcb
$end_info$
*/
#include <chrono>
#include <cstring>
#include <malloc.h>
#include <thread>
#include <stdio.h>
#include <unistd.h>
#include <syscall.h>
#include <sys/syscall.h>
#include <xcb/xcb.h>
#include <xcb/xcbext.h>
#include "common/CrossArchEvent.h"
#include "common/Host.h"
#include <dlfcn.h>
#include <unordered_map>
#include "WorkEventData.h"
#include "thunkgen_host_libxcb.inl"
static void fexfn_impl_libxcb_FEX_xcb_init_extension(xcb_connection_t*, xcb_extension_t*);
static size_t fexfn_impl_libxcb_FEX_usable_size(void*);
static void fexfn_impl_libxcb_FEX_free_on_host(void*);
static void fexfn_impl_libxcb_FEX_GiveEvents(CrossArchEvent*, CrossArchEvent*, CBWork*);
static int fexfn_impl_libxcb_xcb_take_socket(xcb_connection_t * a_0, fex_guest_function_ptr a_1, void * a_2, int a_3, uint64_t * a_4);
struct xcb_take_socket_CB_args {
xcb_connection_t * conn;
fex_guest_function_ptr CBFunction;
void *closure;
};
CrossArchEvent *WaitForWork{};
CrossArchEvent *WorkDone{};
CBWork *Work{};
static std::unordered_map<xcb_connection_t*, xcb_take_socket_CB_args> CBArgs{};
static size_t fexfn_impl_libxcb_FEX_usable_size(void *a_0){
return malloc_usable_size(a_0);
}
static void fexfn_impl_libxcb_FEX_free_on_host(void *a_0){
free(a_0);
}
static void fexfn_impl_libxcb_FEX_GiveEvents(CrossArchEvent* a_0, CrossArchEvent* a_1, CBWork* a_2){
WaitForWork = a_0;
WorkDone = a_1;
Work = a_2;
}
static void xcb_take_socket_cb(void *closure) {
xcb_take_socket_CB_args *Args = (xcb_take_socket_CB_args *)closure;
// Signalling to the guest thread like this allows us to call the callback function from any thread without
// creating spurious thread objects inside of FEX
Work->cb = Args->CBFunction;
Work->argsv = Args->closure;
// Tell the thread it has work
NotifyWorkFunc(WaitForWork);
// Wait for the work to be done
WaitForWorkFunc(WorkDone);
}
static int fexfn_impl_libxcb_xcb_take_socket(xcb_connection_t * a_0, fex_guest_function_ptr a_1, void * a_2, int a_3, uint64_t * a_4){
xcb_take_socket_CB_args Args{};
Args.conn = a_0;
Args.CBFunction = a_1;
Args.closure = a_2;
auto Res = CBArgs.insert_or_assign(a_0, Args);
return fexldr_ptr_libxcb_xcb_take_socket
(a_0,
xcb_take_socket_cb,
&Res.first->second,
a_3,
a_4);
}
static void fexfn_impl_libxcb_FEX_xcb_init_extension(xcb_connection_t * a_0, xcb_extension_t * a_1){
xcb_extension_t *ext{};
if (strcmp(a_1->name, "BIG-REQUESTS") == 0) {
ext = (xcb_extension_t *)dlsym(fexldr_ptr_libxcb_so, "xcb_big_requests_id");
}
else if (strcmp(a_1->name, "XC-MISC") == 0) {
ext = (xcb_extension_t *)dlsym(fexldr_ptr_libxcb_so, "xcb_xc_misc_id");
}
else {
fprintf(stderr, "Unknown xcb extension '%s'\n", a_1->name);
__builtin_trap();
return;
}
if (!ext) {
fprintf(stderr, "Couldn't find extension symbol: '%s'\n", a_1->name);
__builtin_trap();
return;
}
[[maybe_unused]] auto res = fexldr_ptr_libxcb_xcb_get_extension_data(a_0, ext);
// Copy over the global id
a_1->global_id = ext->global_id;
}
EXPORTS(libxcb)