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 printf(" flags: %s\n", (event
->flags
& XIKeyRepeat
) ? "repeat" : "");
53 printf(" root: %.2f/%.2f\n", event
->root_x
, event
->root_y
);
54 printf(" event: %.2f/%.2f\n", event
->event_x
, event
->event_y
);
57 for (i
= 0; i
< event
->buttons
.mask_len
* 8; i
++)
58 if (XIMaskIsSet(event
->buttons
.mask
, i
))
62 printf(" modifiers: locked %#x latched %#x base %#x effective: %#x\n",
63 event
->mods
.locked
, event
->mods
.latched
,
64 event
->mods
.base
, event
->mods
.effective
);
65 printf(" group: locked %#x latched %#x base %#x effective: %#x\n",
66 event
->group
.locked
, event
->group
.latched
,
67 event
->group
.base
, event
->group
.effective
);
68 printf(" valuators:");
70 val
= event
->valuators
.values
;
71 for (i
= 0; i
< event
->valuators
.mask_len
* 8; i
++)
72 if (XIMaskIsSet(event
->valuators
.mask
, i
))
73 printf(" %.2f", *val
++);
76 printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n",
77 event
->root
, event
->event
, event
->child
);
80 static void print_devicechangedevent(Display
*dpy
, XIDeviceChangedEvent
*event
)
82 printf(" device: %d (%d)\n", event
->deviceid
, event
->sourceid
);
83 printf(" reason: %s\n", (event
->reason
== XISlaveSwitch
) ? "SlaveSwitch" :
85 print_classes_xi2(dpy
, event
->classes
, event
->num_classes
);
88 static void print_hierarchychangedevent(XIHierarchyEvent
*event
)
91 printf(" Changes happened: %s %s %s %s %s %s %s %s\n",
92 (event
->flags
& XIMasterAdded
) ? "[new master]" : "",
93 (event
->flags
& XIMasterRemoved
) ? "[master removed]" : "",
94 (event
->flags
& XISlaveAdded
) ? "[new slave]" : "",
95 (event
->flags
& XISlaveRemoved
) ? "[slave removed]" : "",
96 (event
->flags
& XISlaveAttached
) ? "[slave attached]" : "",
97 (event
->flags
& XISlaveDetached
) ? "[slave detached]" : "",
98 (event
->flags
& XIDeviceEnabled
) ? "[device enabled]" : "",
99 (event
->flags
& XIDeviceDisabled
) ? "[device disabled]" : "");
101 for (i
= 0; i
< event
->num_info
; i
++)
104 switch(event
->info
[i
].use
)
106 case XIMasterPointer
: use
= "master pointer"; break;
107 case XIMasterKeyboard
: use
= "master keyboard"; break;
108 case XISlavePointer
: use
= "slave pointer"; break;
109 case XISlaveKeyboard
: use
= "slave keyboard"; break;
110 case XIFloatingSlave
: use
= "floating slave"; break;
114 printf(" device %d [%s (%d)] is %s\n",
115 event
->info
[i
].deviceid
,
117 event
->info
[i
].attachment
,
118 (event
->info
[i
].enabled
) ? "enabled" : "disabled");
119 if (event
->info
[i
].flags
)
121 printf(" changes: %s %s %s %s %s %s %s %s\n",
122 (event
->info
[i
].flags
& XIMasterAdded
) ? "[new master]" : "",
123 (event
->info
[i
].flags
& XIMasterRemoved
) ? "[master removed]" : "",
124 (event
->info
[i
].flags
& XISlaveAdded
) ? "[new slave]" : "",
125 (event
->info
[i
].flags
& XISlaveRemoved
) ? "[slave removed]" : "",
126 (event
->info
[i
].flags
& XISlaveAttached
) ? "[slave attached]" : "",
127 (event
->info
[i
].flags
& XISlaveDetached
) ? "[slave detached]" : "",
128 (event
->info
[i
].flags
& XIDeviceEnabled
) ? "[device enabled]" : "",
129 (event
->info
[i
].flags
& XIDeviceDisabled
) ? "[device disabled]" : "");
134 static void print_rawevent(XIRawEvent
*event
)
137 double *val
, *raw_val
;
139 printf(" device: %d\n", event
->deviceid
);
140 printf(" detail: %d\n", event
->detail
);
141 printf(" valuators:\n");
143 val
= event
->valuators
.values
;
144 raw_val
= event
->raw_values
;
145 for (i
= 0; i
< event
->valuators
.mask_len
* 8; i
++)
146 if (XIMaskIsSet(event
->valuators
.mask
, i
))
147 printf(" %2d: %.2f (%.2f)\n", i
, *val
++, *raw_val
++);
151 static void print_enterleave(XILeaveEvent
* event
)
156 printf(" device: %d\n", event
->deviceid
);
157 printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n",
158 event
->root
, event
->event
, event
->child
);
161 case XINotifyNormal
: mode
= "NotifyNormal"; break;
162 case XINotifyGrab
: mode
= "NotifyGrab"; break;
163 case XINotifyUngrab
: mode
= "NotifyUngrab"; break;
164 case XINotifyWhileGrabbed
: mode
= "NotifyWhileGrabbed"; break;
165 case XINotifyPassiveGrab
: mode
= "NotifyPassiveGrab"; break;
166 case XINotifyPassiveUngrab
:mode
= "NotifyPassiveUngrab"; break;
168 switch (event
->detail
)
170 case XINotifyAncestor
: detail
= "NotifyAncestor"; break;
171 case XINotifyVirtual
: detail
= "NotifyVirtual"; break;
172 case XINotifyInferior
: detail
= "NotifyInferior"; break;
173 case XINotifyNonlinear
: detail
= "NotifyNonlinear"; break;
174 case XINotifyNonlinearVirtual
: detail
= "NotifyNonlinearVirtual"; break;
175 case XINotifyPointer
: detail
= "NotifyPointer"; break;
176 case XINotifyPointerRoot
: detail
= "NotifyPointerRoot"; break;
177 case XINotifyDetailNone
: detail
= "NotifyDetailNone"; break;
179 printf(" mode: %s (detail %s)\n", mode
, detail
);
180 printf(" flags: %s %s\n", event
->focus
? "[focus]" : "",
181 event
->same_screen
? "[same screen]" : "");
183 for (i
= 0; i
< event
->buttons
.mask_len
* 8; i
++)
184 if (XIMaskIsSet(event
->buttons
.mask
, i
))
188 printf(" modifiers: locked %#x latched %#x base %#x effective: %#x\n",
189 event
->mods
.locked
, event
->mods
.latched
,
190 event
->mods
.base
, event
->mods
.effective
);
191 printf(" group: locked %#x latched %#x base %#x effective: %#x\n",
192 event
->group
.locked
, event
->group
.latched
,
193 event
->group
.base
, event
->group
.effective
);
195 printf(" root x/y: %.2f / %.2f\n", event
->root_x
, event
->root_y
);
196 printf(" event x/y: %.2f / %.2f\n", event
->event_x
, event
->event_y
);
200 static void print_propertyevent(Display
*display
, XIPropertyEvent
* event
)
205 if (event
->what
== XIPropertyDeleted
)
207 else if (event
->what
== XIPropertyCreated
)
210 changed
= "modified";
211 name
= XGetAtomName(display
, event
->property
);
212 printf(" property: %ld '%s'\n", event
->property
, name
);
213 printf(" changed: %s\n", changed
);
218 test_sync_grab(Display
*display
, Window win
)
224 /* Select for motion events */
225 mask
.deviceid
= XIAllDevices
;
227 mask
.mask
= calloc(2, sizeof(char));
228 XISetMask(mask
.mask
, XI_ButtonPress
);
230 if ((rc
= XIGrabDevice(display
, 2, win
, CurrentTime
, None
, GrabModeSync
,
231 GrabModeAsync
, False
, &mask
)) != GrabSuccess
)
233 fprintf(stderr
, "Grab failed with %d\n", rc
);
238 XSync(display
, True
);
239 XIAllowEvents(display
, 2, SyncPointer
, CurrentTime
);
242 printf("Holding sync grab for %d button presses.\n", loop
);
248 XNextEvent(display
, (XEvent
*)&ev
);
249 if (ev
.type
== GenericEvent
&& ev
.extension
== xi_opcode
)
251 XIDeviceEvent
*event
= (XIDeviceEvent
*)&ev
;
252 print_deviceevent(event
);
253 XIAllowEvents(display
, 2, SyncPointer
, CurrentTime
);
257 XIUngrabDevice(display
, 2, CurrentTime
);
262 test_xi2(Display
*display
,
271 list(display
, argc
, argv
, name
, desc
);
272 win
= create_win(display
);
274 /* Select for motion events */
275 mask
.deviceid
= XIAllDevices
;
276 mask
.mask_len
= XIMaskLen(XI_RawMotion
);
277 mask
.mask
= calloc(mask
.mask_len
, sizeof(char));
278 XISetMask(mask
.mask
, XI_ButtonPress
);
279 XISetMask(mask
.mask
, XI_ButtonRelease
);
280 XISetMask(mask
.mask
, XI_KeyPress
);
281 XISetMask(mask
.mask
, XI_KeyRelease
);
282 XISetMask(mask
.mask
, XI_Motion
);
283 XISetMask(mask
.mask
, XI_DeviceChanged
);
284 XISetMask(mask
.mask
, XI_Enter
);
285 XISetMask(mask
.mask
, XI_Leave
);
286 XISetMask(mask
.mask
, XI_FocusIn
);
287 XISetMask(mask
.mask
, XI_FocusOut
);
288 XISetMask(mask
.mask
, XI_HierarchyChanged
);
289 XISetMask(mask
.mask
, XI_PropertyEvent
);
290 XISelectEvents(display
, win
, &mask
, 1);
291 XMapWindow(display
, win
);
292 XSync(display
, False
);
295 XIGrabModifiers modifiers
[] = {{0, 0}, {0, 0x10}, {0, 0x1}, {0, 0x11}};
296 int nmods
= sizeof(modifiers
)/sizeof(modifiers
[0]);
299 memset(mask
.mask
, 0, 2);
300 XISetMask(mask
.mask
, XI_KeyPress
);
301 XISetMask(mask
.mask
, XI_KeyRelease
);
302 XISetMask(mask
.mask
, XI_ButtonPress
);
303 XISetMask(mask
.mask
, XI_ButtonRelease
);
304 XISetMask(mask
.mask
, XI_Motion
);
305 XIGrabButton(display
, 2, 1, win
, None
, GrabModeAsync
, GrabModeAsync
,
306 False
, &mask
, nmods
, modifiers
);
307 XIGrabKeycode(display
, 3, 24 /* q */, win
, GrabModeAsync
, GrabModeAsync
,
308 False
, &mask
, nmods
, modifiers
);
309 XIUngrabButton(display
, 3, 1, win
, nmods
- 2, &modifiers
[2]);
310 XIUngrabKeycode(display
, 3, 24 /* q */, win
, nmods
- 2, &modifiers
[2]);
313 mask
.deviceid
= XIAllMasterDevices
;
314 memset(mask
.mask
, 0, 2);
315 XISetMask(mask
.mask
, XI_RawKeyPress
);
316 XISetMask(mask
.mask
, XI_RawKeyRelease
);
317 XISetMask(mask
.mask
, XI_RawButtonPress
);
318 XISetMask(mask
.mask
, XI_RawButtonRelease
);
319 XISetMask(mask
.mask
, XI_RawMotion
);
320 XISelectEvents(display
, DefaultRootWindow(display
), &mask
, 1);
326 XMaskEvent(display
, ExposureMask
, &event
);
327 XSelectInput(display
, win
, 0);
331 test_sync_grab(display, win);
337 XGenericEventCookie
*cookie
= (XGenericEventCookie
*)&ev
.xcookie
;
338 XNextEvent(display
, (XEvent
*)&ev
);
340 if (XGetEventData(display
, cookie
) &&
341 cookie
->type
== GenericEvent
&&
342 cookie
->extension
== xi_opcode
)
344 printf("EVENT type %d\n", cookie
->evtype
);
345 switch (cookie
->evtype
)
347 case XI_DeviceChanged
:
348 print_devicechangedevent(display
, cookie
->data
);
350 case XI_HierarchyChanged
:
351 print_hierarchychangedevent(cookie
->data
);
354 case XI_RawKeyRelease
:
355 case XI_RawButtonPress
:
356 case XI_RawButtonRelease
:
358 print_rawevent(cookie
->data
);
364 print_enterleave(cookie
->data
);
366 case XI_PropertyEvent
:
367 print_propertyevent(display
, cookie
->data
);
370 print_deviceevent(cookie
->data
);
375 XFreeEventData(display
, cookie
);
378 XDestroyWindow(display
, win
);