test_xi2.c revision 53719b08
153719b08Smrg/* 253719b08Smrg * Copyright © 2009 Red Hat, Inc. 353719b08Smrg * 453719b08Smrg * Permission is hereby granted, free of charge, to any person obtaining a 553719b08Smrg * copy of this software and associated documentation files (the "Software"), 653719b08Smrg * to deal in the Software without restriction, including without limitation 753719b08Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 853719b08Smrg * and/or sell copies of the Software, and to permit persons to whom the 953719b08Smrg * Software is furnished to do so, subject to the following conditions: 1053719b08Smrg * 1153719b08Smrg * The above copyright notice and this permission notice (including the next 1253719b08Smrg * paragraph) shall be included in all copies or substantial portions of the 1353719b08Smrg * Software. 1453719b08Smrg * 1553719b08Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 1653719b08Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 1753719b08Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 1853719b08Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 1953719b08Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 2053719b08Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 2153719b08Smrg * DEALINGS IN THE SOFTWARE. 2253719b08Smrg * 2353719b08Smrg */ 2453719b08Smrg 2553719b08Smrg 2653719b08Smrg#include "xinput.h" 2753719b08Smrg#include <string.h> 2853719b08Smrg 2953719b08Smrgextern void print_classes_xi2(Display*, XIAnyClassInfo **classes, 3053719b08Smrg int num_classes); 3153719b08Smrg 3253719b08Smrgstatic Window create_win(Display *dpy) 3353719b08Smrg{ 3453719b08Smrg Window win = XCreateSimpleWindow(dpy, DefaultRootWindow(dpy), 0, 0, 200, 3553719b08Smrg 200, 0, 0, WhitePixel(dpy, 0)); 3653719b08Smrg Window subwindow = XCreateSimpleWindow(dpy, win, 50, 50, 50, 50, 0, 0, 3753719b08Smrg BlackPixel(dpy, 0)); 3853719b08Smrg 3953719b08Smrg XMapWindow(dpy, subwindow); 4053719b08Smrg XSelectInput(dpy, win, ExposureMask); 4153719b08Smrg return win; 4253719b08Smrg} 4353719b08Smrg 4453719b08Smrgstatic void print_deviceevent(XIDeviceEvent* event) 4553719b08Smrg{ 4653719b08Smrg double *val; 4753719b08Smrg int i; 4853719b08Smrg 4953719b08Smrg printf(" device: %d (%d)\n", event->deviceid, event->sourceid); 5053719b08Smrg printf(" detail: %d\n", event->detail); 5153719b08Smrg printf(" flags: %s\n", (event->flags & XIKeyRepeat) ? "repeat" : ""); 5253719b08Smrg 5353719b08Smrg printf(" root: %.2f/%.2f\n", event->root_x, event->root_y); 5453719b08Smrg printf(" event: %.2f/%.2f\n", event->event_x, event->event_y); 5553719b08Smrg 5653719b08Smrg printf(" buttons:"); 5753719b08Smrg for (i = 0; i < event->buttons.mask_len * 8; i++) 5853719b08Smrg if (XIMaskIsSet(event->buttons.mask, i)) 5953719b08Smrg printf(" %d", i); 6053719b08Smrg printf("\n"); 6153719b08Smrg 6253719b08Smrg printf(" modifiers: locked %#x latched %#x base %#x effective: %#x\n", 6353719b08Smrg event->mods.locked, event->mods.latched, 6453719b08Smrg event->mods.base, event->mods.effective); 6553719b08Smrg printf(" group: locked %#x latched %#x base %#x effective: %#x\n", 6653719b08Smrg event->group.locked, event->group.latched, 6753719b08Smrg event->group.base, event->group.effective); 6853719b08Smrg printf(" valuators:"); 6953719b08Smrg 7053719b08Smrg val = event->valuators.values; 7153719b08Smrg for (i = 0; i < event->valuators.mask_len * 8; i++) 7253719b08Smrg if (XIMaskIsSet(event->valuators.mask, i)) 7353719b08Smrg printf(" %.2f", *val++); 7453719b08Smrg printf("\n"); 7553719b08Smrg 7653719b08Smrg printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n", 7753719b08Smrg event->root, event->event, event->child); 7853719b08Smrg} 7953719b08Smrg 8053719b08Smrgstatic void print_devicechangedevent(Display *dpy, XIDeviceChangedEvent *event) 8153719b08Smrg{ 8253719b08Smrg printf(" device: %d (%d)\n", event->deviceid, event->sourceid); 8353719b08Smrg printf(" reason: %s\n", (event->reason == XISlaveSwitch) ? "SlaveSwitch" : 8453719b08Smrg "DeviceChanged"); 8553719b08Smrg print_classes_xi2(dpy, event->classes, event->num_classes); 8653719b08Smrg} 8753719b08Smrg 8853719b08Smrgstatic void print_hierarchychangedevent(XIHierarchyEvent *event) 8953719b08Smrg{ 9053719b08Smrg int i; 9153719b08Smrg printf(" Changes happened: %s %s %s %s %s %s %s %s\n", 9253719b08Smrg (event->flags & XIMasterAdded) ? "[new master]" : "", 9353719b08Smrg (event->flags & XIMasterRemoved) ? "[master removed]" : "", 9453719b08Smrg (event->flags & XISlaveAdded) ? "[new slave]" : "", 9553719b08Smrg (event->flags & XISlaveRemoved) ? "[slave removed]" : "", 9653719b08Smrg (event->flags & XISlaveAttached) ? "[slave attached]" : "", 9753719b08Smrg (event->flags & XISlaveDetached) ? "[slave detached]" : "", 9853719b08Smrg (event->flags & XIDeviceEnabled) ? "[device enabled]" : "", 9953719b08Smrg (event->flags & XIDeviceDisabled) ? "[device disabled]" : ""); 10053719b08Smrg 10153719b08Smrg for (i = 0; i < event->num_info; i++) 10253719b08Smrg { 10353719b08Smrg char *use; 10453719b08Smrg switch(event->info[i].use) 10553719b08Smrg { 10653719b08Smrg case XIMasterPointer: use = "master pointer"; break; 10753719b08Smrg case XIMasterKeyboard: use = "master keyboard"; break; 10853719b08Smrg case XISlavePointer: use = "slave pointer"; break; 10953719b08Smrg case XISlaveKeyboard: use = "slave keyboard"; break; 11053719b08Smrg case XIFloatingSlave: use = "floating slave"; break; 11153719b08Smrg break; 11253719b08Smrg } 11353719b08Smrg 11453719b08Smrg printf(" device %d [%s (%d)] is %s\n", 11553719b08Smrg event->info[i].deviceid, 11653719b08Smrg use, 11753719b08Smrg event->info[i].attachment, 11853719b08Smrg (event->info[i].enabled) ? "enabled" : "disabled"); 11953719b08Smrg if (event->info[i].flags) 12053719b08Smrg { 12153719b08Smrg printf(" changes: %s %s %s %s %s %s %s %s\n", 12253719b08Smrg (event->info[i].flags & XIMasterAdded) ? "[new master]" : "", 12353719b08Smrg (event->info[i].flags & XIMasterRemoved) ? "[master removed]" : "", 12453719b08Smrg (event->info[i].flags & XISlaveAdded) ? "[new slave]" : "", 12553719b08Smrg (event->info[i].flags & XISlaveRemoved) ? "[slave removed]" : "", 12653719b08Smrg (event->info[i].flags & XISlaveAttached) ? "[slave attached]" : "", 12753719b08Smrg (event->info[i].flags & XISlaveDetached) ? "[slave detached]" : "", 12853719b08Smrg (event->info[i].flags & XIDeviceEnabled) ? "[device enabled]" : "", 12953719b08Smrg (event->info[i].flags & XIDeviceDisabled) ? "[device disabled]" : ""); 13053719b08Smrg } 13153719b08Smrg } 13253719b08Smrg} 13353719b08Smrg 13453719b08Smrgstatic void print_rawevent(XIRawEvent *event) 13553719b08Smrg{ 13653719b08Smrg int i; 13753719b08Smrg double *val, *raw_val; 13853719b08Smrg 13953719b08Smrg printf(" device: %d\n", event->deviceid); 14053719b08Smrg printf(" detail: %d\n", event->detail); 14153719b08Smrg printf(" valuators:\n"); 14253719b08Smrg 14353719b08Smrg val = event->valuators.values; 14453719b08Smrg raw_val = event->raw_values; 14553719b08Smrg for (i = 0; i < event->valuators.mask_len * 8; i++) 14653719b08Smrg if (XIMaskIsSet(event->valuators.mask, i)) 14753719b08Smrg printf(" %2d: %.2f (%.2f)\n", i, *val++, *raw_val++); 14853719b08Smrg printf("\n"); 14953719b08Smrg} 15053719b08Smrg 15153719b08Smrgstatic void print_enterleave(XILeaveEvent* event) 15253719b08Smrg{ 15353719b08Smrg char *mode, *detail; 15453719b08Smrg int i; 15553719b08Smrg 15653719b08Smrg printf(" device: %d\n", event->deviceid); 15753719b08Smrg printf(" windows: root 0x%lx event 0x%lx child 0x%ld\n", 15853719b08Smrg event->root, event->event, event->child); 15953719b08Smrg switch(event->mode) 16053719b08Smrg { 16153719b08Smrg case XINotifyNormal: mode = "NotifyNormal"; break; 16253719b08Smrg case XINotifyGrab: mode = "NotifyGrab"; break; 16353719b08Smrg case XINotifyUngrab: mode = "NotifyUngrab"; break; 16453719b08Smrg case XINotifyWhileGrabbed: mode = "NotifyWhileGrabbed"; break; 16553719b08Smrg case XINotifyPassiveGrab: mode = "NotifyPassiveGrab"; break; 16653719b08Smrg case XINotifyPassiveUngrab:mode = "NotifyPassiveUngrab"; break; 16753719b08Smrg } 16853719b08Smrg switch (event->detail) 16953719b08Smrg { 17053719b08Smrg case XINotifyAncestor: detail = "NotifyAncestor"; break; 17153719b08Smrg case XINotifyVirtual: detail = "NotifyVirtual"; break; 17253719b08Smrg case XINotifyInferior: detail = "NotifyInferior"; break; 17353719b08Smrg case XINotifyNonlinear: detail = "NotifyNonlinear"; break; 17453719b08Smrg case XINotifyNonlinearVirtual: detail = "NotifyNonlinearVirtual"; break; 17553719b08Smrg case XINotifyPointer: detail = "NotifyPointer"; break; 17653719b08Smrg case XINotifyPointerRoot: detail = "NotifyPointerRoot"; break; 17753719b08Smrg case XINotifyDetailNone: detail = "NotifyDetailNone"; break; 17853719b08Smrg } 17953719b08Smrg printf(" mode: %s (detail %s)\n", mode, detail); 18053719b08Smrg printf(" flags: %s %s\n", event->focus ? "[focus]" : "", 18153719b08Smrg event->same_screen ? "[same screen]" : ""); 18253719b08Smrg printf(" buttons:"); 18353719b08Smrg for (i = 0; i < event->buttons.mask_len * 8; i++) 18453719b08Smrg if (XIMaskIsSet(event->buttons.mask, i)) 18553719b08Smrg printf(" %d", i); 18653719b08Smrg printf("\n"); 18753719b08Smrg 18853719b08Smrg printf(" modifiers: locked %#x latched %#x base %#x effective: %#x\n", 18953719b08Smrg event->mods.locked, event->mods.latched, 19053719b08Smrg event->mods.base, event->mods.effective); 19153719b08Smrg printf(" group: locked %#x latched %#x base %#x effective: %#x\n", 19253719b08Smrg event->group.locked, event->group.latched, 19353719b08Smrg event->group.base, event->group.effective); 19453719b08Smrg 19553719b08Smrg printf(" root x/y: %.2f / %.2f\n", event->root_x, event->root_y); 19653719b08Smrg printf(" event x/y: %.2f / %.2f\n", event->event_x, event->event_y); 19753719b08Smrg 19853719b08Smrg} 19953719b08Smrg 20053719b08Smrgstatic void print_propertyevent(Display *display, XIPropertyEvent* event) 20153719b08Smrg{ 20253719b08Smrg char *changed; 20353719b08Smrg char *name; 20453719b08Smrg 20553719b08Smrg if (event->what == XIPropertyDeleted) 20653719b08Smrg changed = "deleted"; 20753719b08Smrg else if (event->what == XIPropertyCreated) 20853719b08Smrg changed = "created"; 20953719b08Smrg else 21053719b08Smrg changed = "modified"; 21153719b08Smrg name = XGetAtomName(display, event->property); 21253719b08Smrg printf(" property: %ld '%s'\n", event->property, name); 21353719b08Smrg printf(" changed: %s\n", changed); 21453719b08Smrg 21553719b08Smrg XFree(name); 21653719b08Smrg} 21753719b08Smrgvoid 21853719b08Smrgtest_sync_grab(Display *display, Window win) 21953719b08Smrg{ 22053719b08Smrg int loop = 3; 22153719b08Smrg int rc; 22253719b08Smrg XIEventMask mask; 22353719b08Smrg 22453719b08Smrg /* Select for motion events */ 22553719b08Smrg mask.deviceid = XIAllDevices; 22653719b08Smrg mask.mask_len = 2; 22753719b08Smrg mask.mask = calloc(2, sizeof(char)); 22853719b08Smrg XISetMask(mask.mask, XI_ButtonPress); 22953719b08Smrg 23053719b08Smrg if ((rc = XIGrabDevice(display, 2, win, CurrentTime, None, GrabModeSync, 23153719b08Smrg GrabModeAsync, False, &mask)) != GrabSuccess) 23253719b08Smrg { 23353719b08Smrg fprintf(stderr, "Grab failed with %d\n", rc); 23453719b08Smrg return; 23553719b08Smrg } 23653719b08Smrg free(mask.mask); 23753719b08Smrg 23853719b08Smrg XSync(display, True); 23953719b08Smrg XIAllowEvents(display, 2, SyncPointer, CurrentTime); 24053719b08Smrg XFlush(display); 24153719b08Smrg 24253719b08Smrg printf("Holding sync grab for %d button presses.\n", loop); 24353719b08Smrg 24453719b08Smrg while(loop--) 24553719b08Smrg { 24653719b08Smrg XIEvent ev; 24753719b08Smrg 24853719b08Smrg XNextEvent(display, (XEvent*)&ev); 24953719b08Smrg if (ev.type == GenericEvent && ev.extension == xi_opcode ) 25053719b08Smrg { 25153719b08Smrg XIDeviceEvent *event = (XIDeviceEvent*)&ev; 25253719b08Smrg print_deviceevent(event); 25353719b08Smrg XIAllowEvents(display, 2, SyncPointer, CurrentTime); 25453719b08Smrg } 25553719b08Smrg } 25653719b08Smrg 25753719b08Smrg XIUngrabDevice(display, 2, CurrentTime); 25853719b08Smrg printf("Done\n"); 25953719b08Smrg} 26053719b08Smrg 26153719b08Smrgstatic const char* type_to_name(int evtype) 26253719b08Smrg{ 26353719b08Smrg const char *name; 26453719b08Smrg 26553719b08Smrg switch(evtype) { 26653719b08Smrg case XI_DeviceChanged: name = "DeviceChanged"; break; 26753719b08Smrg case XI_KeyPress: name = "KeyPress"; break; 26853719b08Smrg case XI_KeyRelease: name = "KeyRelease"; break; 26953719b08Smrg case XI_ButtonPress: name = "ButtonPress"; break; 27053719b08Smrg case XI_ButtonRelease: name = "ButtonRelease"; break; 27153719b08Smrg case XI_Motion: name = "Motion"; break; 27253719b08Smrg case XI_Enter: name = "Enter"; break; 27353719b08Smrg case XI_Leave: name = "Leave"; break; 27453719b08Smrg case XI_FocusIn: name = "FocusIn"; break; 27553719b08Smrg case XI_FocusOut: name = "FocusOut"; break; 27653719b08Smrg case XI_HierarchyChanged: name = "HierarchyChanged"; break; 27753719b08Smrg case XI_PropertyEvent: name = "PropertyEvent"; break; 27853719b08Smrg case XI_RawKeyPress: name = "RawKeyPress"; break; 27953719b08Smrg case XI_RawKeyRelease: name = "RawKeyRelease"; break; 28053719b08Smrg case XI_RawButtonPress: name = "RawButtonPress"; break; 28153719b08Smrg case XI_RawButtonRelease: name = "RawButtonRelease"; break; 28253719b08Smrg case XI_RawMotion: name = "RawMotion"; break; 28353719b08Smrg default: 28453719b08Smrg name = "unknown event type"; break; 28553719b08Smrg } 28653719b08Smrg return name; 28753719b08Smrg} 28853719b08Smrg 28953719b08Smrg 29053719b08Smrgint 29153719b08Smrgtest_xi2(Display *display, 29253719b08Smrg int argc, 29353719b08Smrg char *argv[], 29453719b08Smrg char *name, 29553719b08Smrg char *desc) 29653719b08Smrg{ 29753719b08Smrg XIEventMask mask; 29853719b08Smrg Window win; 29953719b08Smrg 30053719b08Smrg list(display, argc, argv, name, desc); 30153719b08Smrg win = create_win(display); 30253719b08Smrg 30353719b08Smrg /* Select for motion events */ 30453719b08Smrg mask.deviceid = XIAllDevices; 30553719b08Smrg mask.mask_len = XIMaskLen(XI_RawMotion); 30653719b08Smrg mask.mask = calloc(mask.mask_len, sizeof(char)); 30753719b08Smrg XISetMask(mask.mask, XI_ButtonPress); 30853719b08Smrg XISetMask(mask.mask, XI_ButtonRelease); 30953719b08Smrg XISetMask(mask.mask, XI_KeyPress); 31053719b08Smrg XISetMask(mask.mask, XI_KeyRelease); 31153719b08Smrg XISetMask(mask.mask, XI_Motion); 31253719b08Smrg XISetMask(mask.mask, XI_DeviceChanged); 31353719b08Smrg XISetMask(mask.mask, XI_Enter); 31453719b08Smrg XISetMask(mask.mask, XI_Leave); 31553719b08Smrg XISetMask(mask.mask, XI_FocusIn); 31653719b08Smrg XISetMask(mask.mask, XI_FocusOut); 31753719b08Smrg XISetMask(mask.mask, XI_HierarchyChanged); 31853719b08Smrg XISetMask(mask.mask, XI_PropertyEvent); 31953719b08Smrg XISelectEvents(display, win, &mask, 1); 32053719b08Smrg XMapWindow(display, win); 32153719b08Smrg XSync(display, False); 32253719b08Smrg 32353719b08Smrg { 32453719b08Smrg XIGrabModifiers modifiers[] = {{0, 0}, {0, 0x10}, {0, 0x1}, {0, 0x11}}; 32553719b08Smrg int nmods = sizeof(modifiers)/sizeof(modifiers[0]); 32653719b08Smrg 32753719b08Smrg mask.deviceid = 2; 32853719b08Smrg memset(mask.mask, 0, 2); 32953719b08Smrg XISetMask(mask.mask, XI_KeyPress); 33053719b08Smrg XISetMask(mask.mask, XI_KeyRelease); 33153719b08Smrg XISetMask(mask.mask, XI_ButtonPress); 33253719b08Smrg XISetMask(mask.mask, XI_ButtonRelease); 33353719b08Smrg XISetMask(mask.mask, XI_Motion); 33453719b08Smrg XIGrabButton(display, 2, 1, win, None, GrabModeAsync, GrabModeAsync, 33553719b08Smrg False, &mask, nmods, modifiers); 33653719b08Smrg XIGrabKeycode(display, 3, 24 /* q */, win, GrabModeAsync, GrabModeAsync, 33753719b08Smrg False, &mask, nmods, modifiers); 33853719b08Smrg XIUngrabButton(display, 3, 1, win, nmods - 2, &modifiers[2]); 33953719b08Smrg XIUngrabKeycode(display, 3, 24 /* q */, win, nmods - 2, &modifiers[2]); 34053719b08Smrg } 34153719b08Smrg 34253719b08Smrg mask.deviceid = XIAllMasterDevices; 34353719b08Smrg memset(mask.mask, 0, 2); 34453719b08Smrg XISetMask(mask.mask, XI_RawKeyPress); 34553719b08Smrg XISetMask(mask.mask, XI_RawKeyRelease); 34653719b08Smrg XISetMask(mask.mask, XI_RawButtonPress); 34753719b08Smrg XISetMask(mask.mask, XI_RawButtonRelease); 34853719b08Smrg XISetMask(mask.mask, XI_RawMotion); 34953719b08Smrg XISelectEvents(display, DefaultRootWindow(display), &mask, 1); 35053719b08Smrg 35153719b08Smrg free(mask.mask); 35253719b08Smrg 35353719b08Smrg { 35453719b08Smrg XEvent event; 35553719b08Smrg XMaskEvent(display, ExposureMask, &event); 35653719b08Smrg XSelectInput(display, win, 0); 35753719b08Smrg } 35853719b08Smrg 35953719b08Smrg /* 36053719b08Smrg test_sync_grab(display, win); 36153719b08Smrg */ 36253719b08Smrg 36353719b08Smrg while(1) 36453719b08Smrg { 36553719b08Smrg XEvent ev; 36653719b08Smrg XGenericEventCookie *cookie = (XGenericEventCookie*)&ev.xcookie; 36753719b08Smrg XNextEvent(display, (XEvent*)&ev); 36853719b08Smrg 36953719b08Smrg if (XGetEventData(display, cookie) && 37053719b08Smrg cookie->type == GenericEvent && 37153719b08Smrg cookie->extension == xi_opcode) 37253719b08Smrg { 37353719b08Smrg printf("EVENT type %d (%s)\n", cookie->evtype, type_to_name(cookie->evtype)); 37453719b08Smrg switch (cookie->evtype) 37553719b08Smrg { 37653719b08Smrg case XI_DeviceChanged: 37753719b08Smrg print_devicechangedevent(display, cookie->data); 37853719b08Smrg break; 37953719b08Smrg case XI_HierarchyChanged: 38053719b08Smrg print_hierarchychangedevent(cookie->data); 38153719b08Smrg break; 38253719b08Smrg case XI_RawKeyPress: 38353719b08Smrg case XI_RawKeyRelease: 38453719b08Smrg case XI_RawButtonPress: 38553719b08Smrg case XI_RawButtonRelease: 38653719b08Smrg case XI_RawMotion: 38753719b08Smrg print_rawevent(cookie->data); 38853719b08Smrg break; 38953719b08Smrg case XI_Enter: 39053719b08Smrg case XI_Leave: 39153719b08Smrg case XI_FocusIn: 39253719b08Smrg case XI_FocusOut: 39353719b08Smrg print_enterleave(cookie->data); 39453719b08Smrg break; 39553719b08Smrg case XI_PropertyEvent: 39653719b08Smrg print_propertyevent(display, cookie->data); 39753719b08Smrg break; 39853719b08Smrg default: 39953719b08Smrg print_deviceevent(cookie->data); 40053719b08Smrg break; 40153719b08Smrg } 40253719b08Smrg } 40353719b08Smrg 40453719b08Smrg XFreeEventData(display, cookie); 40553719b08Smrg } 40653719b08Smrg 40753719b08Smrg XDestroyWindow(display, win); 40853719b08Smrg 40953719b08Smrg return EXIT_SUCCESS; 41053719b08Smrg} 411