2 * Copyright © 2009 Red Hat, Inc.
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21 * DEALINGS IN THE SOFTWARE.
29 extern void print_classes_xi2(Display
*, XIAnyClassInfo
**classes
,
32 static Window
create_win(Display
*dpy
)
34 Window win
= XCreateSimpleWindow(dpy
, DefaultRootWindow(dpy
), 0, 0, 200,
35 200, 0, 0, WhitePixel(dpy
, 0));
36 Window subwindow
= XCreateSimpleWindow(dpy
, win
, 50, 50, 50, 50, 0, 0,
39 XMapWindow(dpy
, subwindow
);
40 XSelectInput(dpy
, win
, ExposureMask
);
44 static void print_deviceevent(XIDeviceEvent
* event
)
49 printf(" device: %d (%d)\n", event
->deviceid
, event
->sourceid
);
50 printf(" detail: %d\n", event
->detail
);
51 switch(event
->evtype
) {
54 printf(" flags: %s\n", (event
->flags
& XIKeyRepeat
) ? "repeat" : "");
58 case XI_ButtonRelease
:
60 printf(" flags: %s\n", (event
->flags
& XIPointerEmulated
) ? "emulated" : "");
65 printf(" root: %.2f/%.2f\n", event
->root_x
, event
->root_y
);
66 printf(" event: %.2f/%.2f\n", event
->event_x
, event
->event_y
);
69 for (i
= 0; i
< event
->buttons
.mask_len
* 8; i
++)
70 if (XIMaskIsSet(event
->buttons
.mask
, i
))
74 printf(" modifiers: locked %#x latched %#x base %#x effective: %#x\n",
75 event
->mods
.locked
, event
->mods
.latched
,
76 event
->mods
.base
, event
->mods
.effective
);
77 printf(" group: locked %#x latched %#x base %#x effective: %#x\n",
78 event
->group
.locked
, event
->group
.latched
,
79 event
->group
.base
, event
->group
.effective
);
80 printf(" valuators:\n");
82 val
= event
->valuators
.values
;
83 for (i
= 0; i
< event
->valuators
.mask_len
* 8; i
++)
84 if (XIMaskIsSet(event
->valuators
.mask
, i
))
85 printf(" %i: %.2f\n", i
, *val
++);
87 printf(" windows: root 0x%lx event 0x%lx child 0x%lx\n",
88 event
->root
, event
->event
, event
->child
);
91 static void print_devicechangedevent(Display
*dpy
, XIDeviceChangedEvent
*event
)
93 printf(" device: %d (%d)\n", event
->deviceid
, event
->sourceid
);
94 printf(" reason: %s\n", (event
->reason
== XISlaveSwitch
) ? "SlaveSwitch" :
96 print_classes_xi2(dpy
, event
->classes
, event
->num_classes
);
99 static void print_hierarchychangedevent(XIHierarchyEvent
*event
)
102 printf(" Changes happened: %s %s %s %s %s %s %s %s\n",
103 (event
->flags
& XIMasterAdded
) ? "[new master]" : "",
104 (event
->flags
& XIMasterRemoved
) ? "[master removed]" : "",
105 (event
->flags
& XISlaveAdded
) ? "[new slave]" : "",
106 (event
->flags
& XISlaveRemoved
) ? "[slave removed]" : "",
107 (event
->flags
& XISlaveAttached
) ? "[slave attached]" : "",
108 (event
->flags
& XISlaveDetached
) ? "[slave detached]" : "",
109 (event
->flags
& XIDeviceEnabled
) ? "[device enabled]" : "",
110 (event
->flags
& XIDeviceDisabled
) ? "[device disabled]" : "");
112 for (i
= 0; i
< event
->num_info
; i
++)
114 char *use
= "<undefined>";
115 switch(event
->info
[i
].use
)
117 case XIMasterPointer
: use
= "master pointer"; break;
118 case XIMasterKeyboard
: use
= "master keyboard"; break;
119 case XISlavePointer
: use
= "slave pointer"; break;
120 case XISlaveKeyboard
: use
= "slave keyboard"; break;
121 case XIFloatingSlave
: use
= "floating slave"; break;
125 printf(" device %d [%s (%d)] is %s\n",
126 event
->info
[i
].deviceid
,
128 event
->info
[i
].attachment
,
129 (event
->info
[i
].enabled
) ? "enabled" : "disabled");
130 if (event
->info
[i
].flags
)
132 printf(" changes: %s %s %s %s %s %s %s %s\n",
133 (event
->info
[i
].flags
& XIMasterAdded
) ? "[new master]" : "",
134 (event
->info
[i
].flags
& XIMasterRemoved
) ? "[master removed]" : "",
135 (event
->info
[i
].flags
& XISlaveAdded
) ? "[new slave]" : "",
136 (event
->info
[i
].flags
& XISlaveRemoved
) ? "[slave removed]" : "",
137 (event
->info
[i
].flags
& XISlaveAttached
) ? "[slave attached]" : "",
138 (event
->info
[i
].flags
& XISlaveDetached
) ? "[slave detached]" : "",
139 (event
->info
[i
].flags
& XIDeviceEnabled
) ? "[device enabled]" : "",
140 (event
->info
[i
].flags
& XIDeviceDisabled
) ? "[device disabled]" : "");
145 static void print_rawevent(XIRawEvent
*event
)
148 double *val
, *raw_val
;
150 printf(" device: %d\n", event
->deviceid
);
151 printf(" detail: %d\n", event
->detail
);
152 printf(" valuators:\n");
154 switch(event
->evtype
) {
155 case XI_RawButtonPress
:
156 case XI_RawButtonRelease
:
158 printf(" flags: %s\n", (event
->flags
& XIPointerEmulated
) ? "emulated" : "");
163 val
= event
->valuators
.values
;
164 raw_val
= event
->raw_values
;
165 for (i
= 0; i
< event
->valuators
.mask_len
* 8; i
++)
166 if (XIMaskIsSet(event
->valuators
.mask
, i
))
167 printf(" %2d: %.2f (%.2f)\n", i
, *val
++, *raw_val
++);
171 static void print_enterleave(XILeaveEvent
* event
)
173 char *mode
= "<undefined>",
174 *detail
= "<undefined>";
177 printf(" device: %d (%d)\n", event
->deviceid
, event
->sourceid
);
178 printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n",
179 event
->root
, event
->event
, event
->child
);
182 case XINotifyNormal
: mode
= "NotifyNormal"; break;
183 case XINotifyGrab
: mode
= "NotifyGrab"; break;
184 case XINotifyUngrab
: mode
= "NotifyUngrab"; break;
185 case XINotifyWhileGrabbed
: mode
= "NotifyWhileGrabbed"; break;
186 case XINotifyPassiveGrab
: mode
= "NotifyPassiveGrab"; break;
187 case XINotifyPassiveUngrab
:mode
= "NotifyPassiveUngrab"; break;
189 switch (event
->detail
)
191 case XINotifyAncestor
: detail
= "NotifyAncestor"; break;
192 case XINotifyVirtual
: detail
= "NotifyVirtual"; break;
193 case XINotifyInferior
: detail
= "NotifyInferior"; break;
194 case XINotifyNonlinear
: detail
= "NotifyNonlinear"; break;
195 case XINotifyNonlinearVirtual
: detail
= "NotifyNonlinearVirtual"; break;
196 case XINotifyPointer
: detail
= "NotifyPointer"; break;
197 case XINotifyPointerRoot
: detail
= "NotifyPointerRoot"; break;
198 case XINotifyDetailNone
: detail
= "NotifyDetailNone"; break;
200 printf(" mode: %s (detail %s)\n", mode
, detail
);
201 printf(" flags: %s %s\n", event
->focus
? "[focus]" : "",
202 event
->same_screen
? "[same screen]" : "");
204 for (i
= 0; i
< event
->buttons
.mask_len
* 8; i
++)
205 if (XIMaskIsSet(event
->buttons
.mask
, i
))
209 printf(" modifiers: locked %#x latched %#x base %#x effective: %#x\n",
210 event
->mods
.locked
, event
->mods
.latched
,
211 event
->mods
.base
, event
->mods
.effective
);
212 printf(" group: locked %#x latched %#x base %#x effective: %#x\n",
213 event
->group
.locked
, event
->group
.latched
,
214 event
->group
.base
, event
->group
.effective
);
216 printf(" root x/y: %.2f / %.2f\n", event
->root_x
, event
->root_y
);
217 printf(" event x/y: %.2f / %.2f\n", event
->event_x
, event
->event_y
);
221 static void print_propertyevent(Display
*display
, XIPropertyEvent
* event
)
226 if (event
->what
== XIPropertyDeleted
)
228 else if (event
->what
== XIPropertyCreated
)
231 changed
= "modified";
232 name
= XGetAtomName(display
, event
->property
);
233 printf(" property: %ld '%s'\n", event
->property
, name
);
234 printf(" changed: %s\n", changed
);
239 test_sync_grab(Display
*display
, Window win
)
245 /* Select for motion events */
246 mask
.deviceid
= XIAllDevices
;
248 mask
.mask
= calloc(2, sizeof(char));
249 XISetMask(mask
.mask
, XI_ButtonPress
);
251 if ((rc
= XIGrabDevice(display
, 2, win
, CurrentTime
, None
, GrabModeSync
,
252 GrabModeAsync
, False
, &mask
)) != GrabSuccess
)
254 fprintf(stderr
, "Grab failed with %d\n", rc
);
259 XSync(display
, True
);
260 XIAllowEvents(display
, 2, SyncPointer
, CurrentTime
);
263 printf("Holding sync grab for %d button presses.\n", loop
);
269 XNextEvent(display
, (XEvent
*)&ev
);
270 if (ev
.type
== GenericEvent
&& ev
.extension
== xi_opcode
)
272 XIDeviceEvent
*event
= (XIDeviceEvent
*)&ev
;
273 print_deviceevent(event
);
274 XIAllowEvents(display
, 2, SyncPointer
, CurrentTime
);
278 XIUngrabDevice(display
, 2, CurrentTime
);
282 static const char* type_to_name(int evtype
)
287 case XI_DeviceChanged
: name
= "DeviceChanged"; break;
288 case XI_KeyPress
: name
= "KeyPress"; break;
289 case XI_KeyRelease
: name
= "KeyRelease"; break;
290 case XI_ButtonPress
: name
= "ButtonPress"; break;
291 case XI_ButtonRelease
: name
= "ButtonRelease"; break;
292 case XI_Motion
: name
= "Motion"; break;
293 case XI_Enter
: name
= "Enter"; break;
294 case XI_Leave
: name
= "Leave"; break;
295 case XI_FocusIn
: name
= "FocusIn"; break;
296 case XI_FocusOut
: name
= "FocusOut"; break;
297 case XI_HierarchyChanged
: name
= "HierarchyChanged"; break;
298 case XI_PropertyEvent
: name
= "PropertyEvent"; break;
299 case XI_RawKeyPress
: name
= "RawKeyPress"; break;
300 case XI_RawKeyRelease
: name
= "RawKeyRelease"; break;
301 case XI_RawButtonPress
: name
= "RawButtonPress"; break;
302 case XI_RawButtonRelease
: name
= "RawButtonRelease"; break;
303 case XI_RawMotion
: name
= "RawMotion"; break;
305 name
= "unknown event type"; break;
312 test_xi2(Display
*display
,
322 list(display
, argc
, argv
, name
, desc
);
325 info
= xi2_find_device_info(display
, argv
[0]);
326 deviceid
= info
->deviceid
;
328 win
= create_win(display
);
330 /* Select for motion events */
331 mask
.deviceid
= (deviceid
== -1) ? XIAllDevices
: deviceid
;
332 mask
.mask_len
= XIMaskLen(XI_RawMotion
);
333 mask
.mask
= calloc(mask
.mask_len
, sizeof(char));
334 XISetMask(mask
.mask
, XI_ButtonPress
);
335 XISetMask(mask
.mask
, XI_ButtonRelease
);
336 XISetMask(mask
.mask
, XI_KeyPress
);
337 XISetMask(mask
.mask
, XI_KeyRelease
);
338 XISetMask(mask
.mask
, XI_Motion
);
339 XISetMask(mask
.mask
, XI_DeviceChanged
);
340 XISetMask(mask
.mask
, XI_Enter
);
341 XISetMask(mask
.mask
, XI_Leave
);
342 XISetMask(mask
.mask
, XI_FocusIn
);
343 XISetMask(mask
.mask
, XI_FocusOut
);
344 if (mask
.deviceid
== XIAllDevices
)
345 XISetMask(mask
.mask
, XI_HierarchyChanged
);
346 XISetMask(mask
.mask
, XI_PropertyEvent
);
347 XISelectEvents(display
, win
, &mask
, 1);
348 XMapWindow(display
, win
);
349 XSync(display
, False
);
352 XIGrabModifiers modifiers
[] = {{0, 0}, {0, 0x10}, {0, 0x1}, {0, 0x11}};
353 int nmods
= sizeof(modifiers
)/sizeof(modifiers
[0]);
356 memset(mask
.mask
, 0, mask
.mask_len
);
357 XISetMask(mask
.mask
, XI_KeyPress
);
358 XISetMask(mask
.mask
, XI_KeyRelease
);
359 XISetMask(mask
.mask
, XI_ButtonPress
);
360 XISetMask(mask
.mask
, XI_ButtonRelease
);
361 XISetMask(mask
.mask
, XI_Motion
);
362 XIGrabButton(display
, 2, 1, win
, None
, GrabModeAsync
, GrabModeAsync
,
363 False
, &mask
, nmods
, modifiers
);
364 XIGrabKeycode(display
, 3, 24 /* q */, win
, GrabModeAsync
, GrabModeAsync
,
365 False
, &mask
, nmods
, modifiers
);
366 XIUngrabButton(display
, 3, 1, win
, nmods
- 2, &modifiers
[2]);
367 XIUngrabKeycode(display
, 3, 24 /* q */, win
, nmods
- 2, &modifiers
[2]);
370 mask
.deviceid
= (deviceid
== -1) ? XIAllMasterDevices
: deviceid
;
371 memset(mask
.mask
, 0, mask
.mask_len
);
372 XISetMask(mask
.mask
, XI_RawKeyPress
);
373 XISetMask(mask
.mask
, XI_RawKeyRelease
);
374 XISetMask(mask
.mask
, XI_RawButtonPress
);
375 XISetMask(mask
.mask
, XI_RawButtonRelease
);
376 XISetMask(mask
.mask
, XI_RawMotion
);
377 XISelectEvents(display
, DefaultRootWindow(display
), &mask
, 1);
383 XMaskEvent(display
, ExposureMask
, &event
);
384 XSelectInput(display
, win
, 0);
388 test_sync_grab(display, win);
394 XGenericEventCookie
*cookie
= (XGenericEventCookie
*)&ev
.xcookie
;
395 XNextEvent(display
, (XEvent
*)&ev
);
397 if (XGetEventData(display
, cookie
) &&
398 cookie
->type
== GenericEvent
&&
399 cookie
->extension
== xi_opcode
)
401 printf("EVENT type %d (%s)\n", cookie
->evtype
, type_to_name(cookie
->evtype
));
402 switch (cookie
->evtype
)
404 case XI_DeviceChanged
:
405 print_devicechangedevent(display
, cookie
->data
);
407 case XI_HierarchyChanged
:
408 print_hierarchychangedevent(cookie
->data
);
411 case XI_RawKeyRelease
:
412 case XI_RawButtonPress
:
413 case XI_RawButtonRelease
:
415 print_rawevent(cookie
->data
);
421 print_enterleave(cookie
->data
);
423 case XI_PropertyEvent
:
424 print_propertyevent(display
, cookie
->data
);
427 print_deviceevent(cookie
->data
);
432 XFreeEventData(display
, cookie
);
435 XDestroyWindow(display
, win
);