1*4882a593Smuzhiyun /**
2*4882a593Smuzhiyun * Copyright © 2009 Red Hat, Inc.
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Permission is hereby granted, free of charge, to any person obtaining a
5*4882a593Smuzhiyun * copy of this software and associated documentation files (the "Software"),
6*4882a593Smuzhiyun * to deal in the Software without restriction, including without limitation
7*4882a593Smuzhiyun * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*4882a593Smuzhiyun * and/or sell copies of the Software, and to permit persons to whom the
9*4882a593Smuzhiyun * Software is furnished to do so, subject to the following conditions:
10*4882a593Smuzhiyun *
11*4882a593Smuzhiyun * The above copyright notice and this permission notice (including the next
12*4882a593Smuzhiyun * paragraph) shall be included in all copies or substantial portions of the
13*4882a593Smuzhiyun * Software.
14*4882a593Smuzhiyun *
15*4882a593Smuzhiyun * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*4882a593Smuzhiyun * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*4882a593Smuzhiyun * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*4882a593Smuzhiyun * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*4882a593Smuzhiyun * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*4882a593Smuzhiyun * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21*4882a593Smuzhiyun * DEALINGS IN THE SOFTWARE.
22*4882a593Smuzhiyun */
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #ifdef HAVE_DIX_CONFIG_H
25*4882a593Smuzhiyun #include <dix-config.h>
26*4882a593Smuzhiyun #endif
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun #include <errno.h>
29*4882a593Smuzhiyun #include <stdint.h>
30*4882a593Smuzhiyun #include "extinit.h" /* for XInputExtensionInit */
31*4882a593Smuzhiyun #include "exglobals.h"
32*4882a593Smuzhiyun #include "xkbsrv.h" /* for XkbInitPrivates */
33*4882a593Smuzhiyun #include "xserver-properties.h"
34*4882a593Smuzhiyun #include "syncsrv.h"
35*4882a593Smuzhiyun #include <X11/extensions/XI2.h>
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun #define INSIDE_PROTOCOL_COMMON
38*4882a593Smuzhiyun #include "protocol-common.h"
39*4882a593Smuzhiyun
40*4882a593Smuzhiyun struct devices devices;
41*4882a593Smuzhiyun ScreenRec screen;
42*4882a593Smuzhiyun WindowRec root;
43*4882a593Smuzhiyun WindowRec window;
44*4882a593Smuzhiyun static ClientRec server_client;
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun void *global_userdata;
47*4882a593Smuzhiyun
48*4882a593Smuzhiyun void (*reply_handler) (ClientPtr client, int len, char *data, void *userdata);
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun int enable_GrabButton_wrap = 1;
51*4882a593Smuzhiyun int enable_XISetEventMask_wrap = 1;
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun static void
fake_init_sprite(DeviceIntPtr dev)54*4882a593Smuzhiyun fake_init_sprite(DeviceIntPtr dev)
55*4882a593Smuzhiyun {
56*4882a593Smuzhiyun SpritePtr sprite;
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun sprite = dev->spriteInfo->sprite;
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun sprite->spriteTraceSize = 10;
61*4882a593Smuzhiyun sprite->spriteTrace = calloc(sprite->spriteTraceSize, sizeof(WindowPtr));
62*4882a593Smuzhiyun sprite->spriteTraceGood = 1;
63*4882a593Smuzhiyun sprite->spriteTrace[0] = &root;
64*4882a593Smuzhiyun sprite->hot.x = SPRITE_X;
65*4882a593Smuzhiyun sprite->hot.y = SPRITE_Y;
66*4882a593Smuzhiyun sprite->hotPhys.x = sprite->hot.x;
67*4882a593Smuzhiyun sprite->hotPhys.y = sprite->hot.y;
68*4882a593Smuzhiyun sprite->win = &window;
69*4882a593Smuzhiyun sprite->hotPhys.pScreen = &screen;
70*4882a593Smuzhiyun sprite->physLimits.x1 = 0;
71*4882a593Smuzhiyun sprite->physLimits.y1 = 0;
72*4882a593Smuzhiyun sprite->physLimits.x2 = screen.width;
73*4882a593Smuzhiyun sprite->physLimits.y2 = screen.height;
74*4882a593Smuzhiyun }
75*4882a593Smuzhiyun
76*4882a593Smuzhiyun /* This is essentially CorePointerProc with ScrollAxes added */
77*4882a593Smuzhiyun static int
TestPointerProc(DeviceIntPtr pDev,int what)78*4882a593Smuzhiyun TestPointerProc(DeviceIntPtr pDev, int what)
79*4882a593Smuzhiyun {
80*4882a593Smuzhiyun #define NBUTTONS 10
81*4882a593Smuzhiyun #define NAXES 4
82*4882a593Smuzhiyun BYTE map[NBUTTONS + 1];
83*4882a593Smuzhiyun int i = 0;
84*4882a593Smuzhiyun Atom btn_labels[NBUTTONS] = { 0 };
85*4882a593Smuzhiyun Atom axes_labels[NAXES] = { 0 };
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun switch (what) {
88*4882a593Smuzhiyun case DEVICE_INIT:
89*4882a593Smuzhiyun for (i = 1; i <= NBUTTONS; i++)
90*4882a593Smuzhiyun map[i] = i;
91*4882a593Smuzhiyun
92*4882a593Smuzhiyun btn_labels[0] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_LEFT);
93*4882a593Smuzhiyun btn_labels[1] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_MIDDLE);
94*4882a593Smuzhiyun btn_labels[2] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_RIGHT);
95*4882a593Smuzhiyun btn_labels[3] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_WHEEL_UP);
96*4882a593Smuzhiyun btn_labels[4] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_WHEEL_DOWN);
97*4882a593Smuzhiyun btn_labels[5] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_HWHEEL_LEFT);
98*4882a593Smuzhiyun btn_labels[6] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_HWHEEL_RIGHT);
99*4882a593Smuzhiyun /* don't know about the rest */
100*4882a593Smuzhiyun
101*4882a593Smuzhiyun axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_X);
102*4882a593Smuzhiyun axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_Y);
103*4882a593Smuzhiyun axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_VSCROLL);
104*4882a593Smuzhiyun axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_HSCROLL);
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun if (!InitPointerDeviceStruct
107*4882a593Smuzhiyun ((DevicePtr) pDev, map, NBUTTONS, btn_labels,
108*4882a593Smuzhiyun (PtrCtrlProcPtr) NoopDDA, GetMotionHistorySize(), NAXES,
109*4882a593Smuzhiyun axes_labels)) {
110*4882a593Smuzhiyun ErrorF("Could not initialize device '%s'. Out of memory.\n",
111*4882a593Smuzhiyun pDev->name);
112*4882a593Smuzhiyun return BadAlloc;
113*4882a593Smuzhiyun }
114*4882a593Smuzhiyun pDev->valuator->axisVal[0] = screenInfo.screens[0]->width / 2;
115*4882a593Smuzhiyun pDev->last.valuators[0] = pDev->valuator->axisVal[0];
116*4882a593Smuzhiyun pDev->valuator->axisVal[1] = screenInfo.screens[0]->height / 2;
117*4882a593Smuzhiyun pDev->last.valuators[1] = pDev->valuator->axisVal[1];
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun /* protocol-xiquerydevice.c relies on these increment */
120*4882a593Smuzhiyun SetScrollValuator(pDev, 2, SCROLL_TYPE_VERTICAL, 2.4, SCROLL_FLAG_NONE);
121*4882a593Smuzhiyun SetScrollValuator(pDev, 3, SCROLL_TYPE_HORIZONTAL, 3.5,
122*4882a593Smuzhiyun SCROLL_FLAG_PREFERRED);
123*4882a593Smuzhiyun break;
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun case DEVICE_CLOSE:
126*4882a593Smuzhiyun break;
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun default:
129*4882a593Smuzhiyun break;
130*4882a593Smuzhiyun }
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun return Success;
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun #undef NBUTTONS
135*4882a593Smuzhiyun #undef NAXES
136*4882a593Smuzhiyun }
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun /**
139*4882a593Smuzhiyun * Create and init 2 master devices (VCP + VCK) and two slave devices, one
140*4882a593Smuzhiyun * default mouse, one default keyboard.
141*4882a593Smuzhiyun */
142*4882a593Smuzhiyun struct devices
init_devices(void)143*4882a593Smuzhiyun init_devices(void)
144*4882a593Smuzhiyun {
145*4882a593Smuzhiyun ClientRec client;
146*4882a593Smuzhiyun struct devices local_devices;
147*4882a593Smuzhiyun int ret;
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun /*
150*4882a593Smuzhiyun * Put a unique name in display pointer so that when tests are run in
151*4882a593Smuzhiyun * parallel, their xkbcomp outputs to /tmp/server-<display>.xkm don't
152*4882a593Smuzhiyun * stomp on each other.
153*4882a593Smuzhiyun */
154*4882a593Smuzhiyun #ifdef HAVE_GETPROGNAME
155*4882a593Smuzhiyun display = getprogname();
156*4882a593Smuzhiyun #elif HAVE_DECL_PROGRAM_INVOCATION_SHORT_NAME
157*4882a593Smuzhiyun display = program_invocation_short_name;
158*4882a593Smuzhiyun #endif
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun client = init_client(0, NULL);
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun AllocDevicePair(&client, "Virtual core", &local_devices.vcp, &local_devices.vck,
163*4882a593Smuzhiyun CorePointerProc, CoreKeyboardProc, TRUE);
164*4882a593Smuzhiyun inputInfo.pointer = local_devices.vcp;
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun inputInfo.keyboard = local_devices.vck;
167*4882a593Smuzhiyun ret = ActivateDevice(local_devices.vcp, FALSE);
168*4882a593Smuzhiyun assert(ret == Success);
169*4882a593Smuzhiyun /* This may fail if xkbcomp fails or xkb-config is not found. */
170*4882a593Smuzhiyun ret = ActivateDevice(local_devices.vck, FALSE);
171*4882a593Smuzhiyun assert(ret == Success);
172*4882a593Smuzhiyun EnableDevice(local_devices.vcp, FALSE);
173*4882a593Smuzhiyun EnableDevice(local_devices.vck, FALSE);
174*4882a593Smuzhiyun
175*4882a593Smuzhiyun AllocDevicePair(&client, "", &local_devices.mouse, &local_devices.kbd,
176*4882a593Smuzhiyun TestPointerProc, CoreKeyboardProc, FALSE);
177*4882a593Smuzhiyun ret = ActivateDevice(local_devices.mouse, FALSE);
178*4882a593Smuzhiyun assert(ret == Success);
179*4882a593Smuzhiyun ret = ActivateDevice(local_devices.kbd, FALSE);
180*4882a593Smuzhiyun assert(ret == Success);
181*4882a593Smuzhiyun EnableDevice(local_devices.mouse, FALSE);
182*4882a593Smuzhiyun EnableDevice(local_devices.kbd, FALSE);
183*4882a593Smuzhiyun
184*4882a593Smuzhiyun local_devices.num_devices = 4;
185*4882a593Smuzhiyun local_devices.num_master_devices = 2;
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun fake_init_sprite(local_devices.mouse);
188*4882a593Smuzhiyun fake_init_sprite(local_devices.vcp);
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun return local_devices;
191*4882a593Smuzhiyun }
192*4882a593Smuzhiyun
193*4882a593Smuzhiyun /* Create minimal client, with the given buffer and len as request buffer */
194*4882a593Smuzhiyun ClientRec
init_client(int len,void * data)195*4882a593Smuzhiyun init_client(int len, void *data)
196*4882a593Smuzhiyun {
197*4882a593Smuzhiyun ClientRec client = { 0 };
198*4882a593Smuzhiyun
199*4882a593Smuzhiyun /* we store the privates now and reassign it after the memset. this way
200*4882a593Smuzhiyun * we can share them across multiple test runs and don't have to worry
201*4882a593Smuzhiyun * about freeing them after each test run. */
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun client.index = CLIENT_INDEX;
204*4882a593Smuzhiyun client.clientAsMask = CLIENT_MASK;
205*4882a593Smuzhiyun client.sequence = CLIENT_SEQUENCE;
206*4882a593Smuzhiyun client.req_len = len;
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun client.requestBuffer = data;
209*4882a593Smuzhiyun dixAllocatePrivates(&client.devPrivates, PRIVATE_CLIENT);
210*4882a593Smuzhiyun return client;
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun void
init_window(WindowPtr local_window,WindowPtr parent,int id)214*4882a593Smuzhiyun init_window(WindowPtr local_window, WindowPtr parent, int id)
215*4882a593Smuzhiyun {
216*4882a593Smuzhiyun memset(local_window, 0, sizeof(*local_window));
217*4882a593Smuzhiyun
218*4882a593Smuzhiyun local_window->drawable.id = id;
219*4882a593Smuzhiyun if (parent) {
220*4882a593Smuzhiyun local_window->drawable.x = 30;
221*4882a593Smuzhiyun local_window->drawable.y = 50;
222*4882a593Smuzhiyun local_window->drawable.width = 100;
223*4882a593Smuzhiyun local_window->drawable.height = 200;
224*4882a593Smuzhiyun }
225*4882a593Smuzhiyun local_window->parent = parent;
226*4882a593Smuzhiyun local_window->optional = calloc(1, sizeof(WindowOptRec));
227*4882a593Smuzhiyun assert(local_window->optional);
228*4882a593Smuzhiyun }
229*4882a593Smuzhiyun
230*4882a593Smuzhiyun extern DevPrivateKeyRec miPointerScreenKeyRec;
231*4882a593Smuzhiyun extern DevPrivateKeyRec miPointerPrivKeyRec;
232*4882a593Smuzhiyun
233*4882a593Smuzhiyun /* Needed for the screen setup, otherwise we crash during sprite initialization */
234*4882a593Smuzhiyun static Bool
device_cursor_init(DeviceIntPtr dev,ScreenPtr local_screen)235*4882a593Smuzhiyun device_cursor_init(DeviceIntPtr dev, ScreenPtr local_screen)
236*4882a593Smuzhiyun {
237*4882a593Smuzhiyun return TRUE;
238*4882a593Smuzhiyun }
239*4882a593Smuzhiyun
240*4882a593Smuzhiyun static void
device_cursor_cleanup(DeviceIntPtr dev,ScreenPtr local_screen)241*4882a593Smuzhiyun device_cursor_cleanup(DeviceIntPtr dev, ScreenPtr local_screen)
242*4882a593Smuzhiyun {
243*4882a593Smuzhiyun }
244*4882a593Smuzhiyun
245*4882a593Smuzhiyun static Bool
set_cursor_pos(DeviceIntPtr dev,ScreenPtr local_screen,int x,int y,Bool event)246*4882a593Smuzhiyun set_cursor_pos(DeviceIntPtr dev, ScreenPtr local_screen, int x, int y, Bool event)
247*4882a593Smuzhiyun {
248*4882a593Smuzhiyun return TRUE;
249*4882a593Smuzhiyun }
250*4882a593Smuzhiyun
251*4882a593Smuzhiyun void
init_simple(void)252*4882a593Smuzhiyun init_simple(void)
253*4882a593Smuzhiyun {
254*4882a593Smuzhiyun screenInfo.numScreens = 1;
255*4882a593Smuzhiyun screenInfo.screens[0] = &screen;
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun screen.myNum = 0;
258*4882a593Smuzhiyun screen.id = 100;
259*4882a593Smuzhiyun screen.width = 640;
260*4882a593Smuzhiyun screen.height = 480;
261*4882a593Smuzhiyun screen.DeviceCursorInitialize = device_cursor_init;
262*4882a593Smuzhiyun screen.DeviceCursorCleanup = device_cursor_cleanup;
263*4882a593Smuzhiyun screen.SetCursorPosition = set_cursor_pos;
264*4882a593Smuzhiyun screen.root = &root;
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun dixResetPrivates();
267*4882a593Smuzhiyun InitAtoms();
268*4882a593Smuzhiyun XkbInitPrivates();
269*4882a593Smuzhiyun dixRegisterPrivateKey(&XIClientPrivateKeyRec, PRIVATE_CLIENT,
270*4882a593Smuzhiyun sizeof(XIClientRec));
271*4882a593Smuzhiyun dixRegisterPrivateKey(&miPointerScreenKeyRec, PRIVATE_SCREEN, 0);
272*4882a593Smuzhiyun dixRegisterPrivateKey(&miPointerPrivKeyRec, PRIVATE_DEVICE, 0);
273*4882a593Smuzhiyun XInputExtensionInit();
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun init_window(&root, NULL, ROOT_WINDOW_ID);
276*4882a593Smuzhiyun init_window(&window, &root, CLIENT_WINDOW_ID);
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun serverClient = &server_client;
279*4882a593Smuzhiyun InitClient(serverClient, 0, (void *) NULL);
280*4882a593Smuzhiyun if (!InitClientResources(serverClient)) /* for root resources */
281*4882a593Smuzhiyun FatalError("couldn't init server resources");
282*4882a593Smuzhiyun SyncExtensionInit();
283*4882a593Smuzhiyun
284*4882a593Smuzhiyun devices = init_devices();
285*4882a593Smuzhiyun }
286*4882a593Smuzhiyun
287*4882a593Smuzhiyun void
__wrap_WriteToClient(ClientPtr client,int len,void * data)288*4882a593Smuzhiyun __wrap_WriteToClient(ClientPtr client, int len, void *data)
289*4882a593Smuzhiyun {
290*4882a593Smuzhiyun assert(reply_handler != NULL);
291*4882a593Smuzhiyun
292*4882a593Smuzhiyun (*reply_handler) (client, len, data, global_userdata);
293*4882a593Smuzhiyun }
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun /* dixLookupWindow requires a lot of setup not necessary for this test.
296*4882a593Smuzhiyun * Simple wrapper that returns either one of the fake root window or the
297*4882a593Smuzhiyun * fake client window. If the requested ID is neither of those wanted,
298*4882a593Smuzhiyun * return whatever the real dixLookupWindow does.
299*4882a593Smuzhiyun */
300*4882a593Smuzhiyun int
__wrap_dixLookupWindow(WindowPtr * win,XID id,ClientPtr client,Mask access)301*4882a593Smuzhiyun __wrap_dixLookupWindow(WindowPtr *win, XID id, ClientPtr client, Mask access)
302*4882a593Smuzhiyun {
303*4882a593Smuzhiyun if (id == root.drawable.id) {
304*4882a593Smuzhiyun *win = &root;
305*4882a593Smuzhiyun return Success;
306*4882a593Smuzhiyun }
307*4882a593Smuzhiyun else if (id == window.drawable.id) {
308*4882a593Smuzhiyun *win = &window;
309*4882a593Smuzhiyun return Success;
310*4882a593Smuzhiyun }
311*4882a593Smuzhiyun
312*4882a593Smuzhiyun return __real_dixLookupWindow(win, id, client, access);
313*4882a593Smuzhiyun }
314*4882a593Smuzhiyun
315*4882a593Smuzhiyun extern ClientRec client_window;
316*4882a593Smuzhiyun
317*4882a593Smuzhiyun int
__wrap_dixLookupClient(ClientPtr * pClient,XID rid,ClientPtr client,Mask access)318*4882a593Smuzhiyun __wrap_dixLookupClient(ClientPtr *pClient, XID rid, ClientPtr client,
319*4882a593Smuzhiyun Mask access)
320*4882a593Smuzhiyun {
321*4882a593Smuzhiyun if (rid == ROOT_WINDOW_ID)
322*4882a593Smuzhiyun return BadWindow;
323*4882a593Smuzhiyun
324*4882a593Smuzhiyun if (rid == CLIENT_WINDOW_ID) {
325*4882a593Smuzhiyun *pClient = &client_window;
326*4882a593Smuzhiyun return Success;
327*4882a593Smuzhiyun }
328*4882a593Smuzhiyun
329*4882a593Smuzhiyun return __real_dixLookupClient(pClient, rid, client, access);
330*4882a593Smuzhiyun }
331