356 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			356 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * Copyright 2001,2002 Red Hat Inc., Durham, North Carolina.
 | |
|  *
 | |
|  * All Rights Reserved.
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining
 | |
|  * a copy of this software and associated documentation files (the
 | |
|  * "Software"), to deal in the Software without restriction, including
 | |
|  * without limitation on the rights to use, copy, modify, merge,
 | |
|  * publish, distribute, sublicense, and/or sell copies of the Software,
 | |
|  * and to permit persons to whom the Software is furnished to do so,
 | |
|  * subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice (including the
 | |
|  * next paragraph) shall be included in all copies or substantial
 | |
|  * portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 | |
|  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | |
|  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 | |
|  * NON-INFRINGEMENT.  IN NO EVENT SHALL RED HAT AND/OR THEIR SUPPLIERS
 | |
|  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
 | |
|  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 | |
|  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | |
|  * SOFTWARE.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Authors:
 | |
|  *   Rickard E. (Rik) Faith <faith@redhat.com>
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <X11/Xlib.h>
 | |
| #include <X11/XKBlib.h>
 | |
| #include <X11/extensions/XInput.h>
 | |
| #include <X11/extensions/XKB.h>
 | |
| #include <X11/extensions/XKBstr.h>
 | |
| #include <X11/extensions/dmxext.h>
 | |
| #include <sys/time.h>
 | |
| 
 | |
| static const char *
 | |
| core(DMXInputAttributes * iinf)
 | |
| {
 | |
|     if (iinf->isCore)
 | |
|         return "core";
 | |
|     else if (iinf->sendsCore)
 | |
|         return "extension (sends core events)";
 | |
|     else
 | |
|         return "extension";
 | |
| }
 | |
| 
 | |
| static void
 | |
| printdmxinfo(Display * display, int id)
 | |
| {
 | |
|     int event_base;
 | |
|     int error_base;
 | |
|     int major_version, minor_version, patch_version;
 | |
|     DMXInputAttributes iinf;
 | |
|     Display *backend;
 | |
|     char *backendname = NULL;
 | |
| 
 | |
|     if (!DMXQueryExtension(display, &event_base, &error_base))
 | |
|         return;
 | |
|     if (!DMXQueryVersion(display, &major_version, &minor_version,
 | |
|                          &patch_version))
 | |
|         return;
 | |
|     if (major_version == 1 && minor_version == 0)
 | |
|         return;                 /* too old */
 | |
|     if (!DMXGetInputAttributes(display, id, &iinf))
 | |
|         return;
 | |
| 
 | |
|     printf("   DMX Information: ");
 | |
|     if (iinf.detached)
 | |
|         printf("detached ");
 | |
|     else
 | |
|         printf("active   ");
 | |
|     switch (iinf.inputType) {
 | |
|     case DMXLocalInputType:
 | |
|         printf("local, %s", core(&iinf));
 | |
|         break;
 | |
|     case DMXConsoleInputType:
 | |
|         printf("console %s, %s", iinf.name, core(&iinf));
 | |
|         break;
 | |
|     case DMXBackendInputType:
 | |
|         if (iinf.physicalId >= 0) {
 | |
|             if ((backend = XOpenDisplay(iinf.name))) {
 | |
|                 XExtensionVersion *ext = XGetExtensionVersion(backend, INAME);
 | |
| 
 | |
|                 if (ext && ext != (XExtensionVersion *) NoSuchExtension) {
 | |
|                     int count, i;
 | |
|                     XDeviceInfo *devInfo = XListInputDevices(backend, &count);
 | |
| 
 | |
|                     if (devInfo) {
 | |
|                         for (i = 0; i < count; i++) {
 | |
|                             if ((unsigned) iinf.physicalId == devInfo[i].id
 | |
|                                 && devInfo[i].name) {
 | |
|                                 backendname = strdup(devInfo[i].name);
 | |
|                                 break;
 | |
|                             }
 | |
|                         }
 | |
|                         XFreeDeviceList(devInfo);
 | |
|                     }
 | |
|                 }
 | |
|                 XCloseDisplay(backend);
 | |
|             }
 | |
|         }
 | |
|         printf("backend o%d/%s", iinf.physicalScreen, iinf.name);
 | |
|         if (iinf.physicalId >= 0)
 | |
|             printf("/id%d", iinf.physicalId);
 | |
|         if (backendname) {
 | |
|             printf("=%s", backendname);
 | |
|             free(backendname);
 | |
|         }
 | |
|         printf(" %s", core(&iinf));
 | |
|         break;
 | |
|     }
 | |
|     printf("\n");
 | |
| }
 | |
| 
 | |
| int
 | |
| main(int argc, char **argv)
 | |
| {
 | |
|     Display *display = NULL;
 | |
|     int device = -1;
 | |
|     int newmouse = -1;
 | |
|     int newkbd = -1;
 | |
|     int count;
 | |
|     int i, j;
 | |
|     XDeviceInfo *devInfo;
 | |
|     XExtensionVersion *ext;
 | |
| 
 | |
|     if (argc == 2 || argc == 3 || argc == 4 || argc == 5) {
 | |
|         if (!(display = XOpenDisplay(argv[1]))) {
 | |
|             printf("Cannot open display %s\n", argv[1]);
 | |
|             return -1;
 | |
|         }
 | |
|         if (argc >= 3)
 | |
|             device = strtol(argv[2], NULL, 0);
 | |
|         if (argc >= 4)
 | |
|             newmouse = strtol(argv[3], NULL, 0);
 | |
|         if (argc >= 5)
 | |
|             newkbd = strtol(argv[4], NULL, 0);
 | |
|     }
 | |
|     else {
 | |
|         printf("Usage: %s display [device] [newmouse] [newkbd]\n", argv[0]);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     if (!display && !(display = XOpenDisplay(NULL))) {
 | |
|         printf("Cannot open default display\n");
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     ext = XGetExtensionVersion(display, INAME);
 | |
|     if (!ext || ext == (XExtensionVersion *) NoSuchExtension) {
 | |
|         printf("No XInputExtension\n");
 | |
|         return -1;
 | |
|     }
 | |
|     printf("%s version %d.%d\n", INAME, ext->major_version, ext->minor_version);
 | |
| 
 | |
|     if (!(devInfo = XListInputDevices(display, &count)) || !count) {
 | |
|         printf("Cannot list devices\n");
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     for (i = 0; i < count; i++) {
 | |
|         XAnyClassPtr any;
 | |
|         const char *kind = "Unknown";
 | |
|         int has_key = 0;
 | |
| 
 | |
|         switch (devInfo[i].use) {
 | |
|         case IsXPointer:
 | |
|             kind = "XPointer";
 | |
|             break;
 | |
|         case IsXKeyboard:
 | |
|             kind = "XKeyboard";
 | |
|             break;
 | |
|         case IsXExtensionDevice:
 | |
|             kind = "XExtensionDevice";
 | |
|             break;
 | |
|         }
 | |
|         printf("%2lu %-20.20s %-16.16s",
 | |
|                (long unsigned) devInfo[i].id,
 | |
|                devInfo[i].name ? devInfo[i].name : "", kind);
 | |
| 
 | |
|         for (j = 0, any = devInfo[i].inputclassinfo;
 | |
|              j < devInfo[i].num_classes;
 | |
|              any = (XAnyClassPtr) ((char *) any + any->length), j++) {
 | |
|             const char *class = "unk";
 | |
| 
 | |
|             switch (any->class) {
 | |
|             case KeyClass:
 | |
|                 class = "key";
 | |
|                 ++has_key;
 | |
|                 break;
 | |
|             case ButtonClass:
 | |
|                 class = "btn";
 | |
|                 break;
 | |
|             case ValuatorClass:
 | |
|                 class = "val";
 | |
|                 break;
 | |
|             case FeedbackClass:
 | |
|                 class = "fdb";
 | |
|                 break;
 | |
|             case ProximityClass:
 | |
|                 class = "prx";
 | |
|                 break;
 | |
|             case FocusClass:
 | |
|                 class = "foc";
 | |
|                 break;
 | |
|             case OtherClass:
 | |
|                 class = "oth";
 | |
|                 break;
 | |
|             }
 | |
|             printf(" %s", class);
 | |
|         }
 | |
|         printf("\n");
 | |
|         printdmxinfo(display, i);
 | |
| 
 | |
|         if (has_key) {
 | |
|             XkbDescPtr xkb;
 | |
| 
 | |
|             if ((xkb = XkbGetKeyboard(display,
 | |
|                                       XkbAllComponentsMask, devInfo[i].id))) {
 | |
|                 printf("   Xkb Information:\n");
 | |
|                 printf("      Device id = %d\n", xkb->device_spec);
 | |
|                 printf("      Min keycode = 0x%02x\n", xkb->min_key_code);
 | |
|                 printf("      Max keycode = 0x%02x\n", xkb->max_key_code);
 | |
| #define PRINTNAME(x)                                                     \
 | |
|     printf("      %s = %s\n",                                            \
 | |
|            #x, xkb->names->x ? XGetAtomName(display, xkb->names->x) : "")
 | |
|                 PRINTNAME(keycodes);
 | |
|                 PRINTNAME(geometry);
 | |
|                 PRINTNAME(symbols);
 | |
|                 PRINTNAME(types);
 | |
|                 PRINTNAME(compat);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (newmouse >= 0) {
 | |
|         XDevice *dev;
 | |
| 
 | |
|         printf("Trying to make device %d core mouse\n", newmouse);
 | |
|         dev = XOpenDevice(display, devInfo[newmouse].id);
 | |
|         printf("Status = %d\n", XChangePointerDevice(display, dev, 0, 1));
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     if (newkbd >= 0) {
 | |
|         XDevice *dev;
 | |
| 
 | |
|         printf("Trying to make device %d core keyboard\n", newkbd);
 | |
|         dev = XOpenDevice(display, devInfo[newkbd].id);
 | |
|         printf("Status = %d\n", XChangeKeyboardDevice(display, dev));
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     if (device >= 0) {
 | |
| #define MAX_EVENTS 100
 | |
|         int cnt = 0;
 | |
|         XDevice *dev;
 | |
|         XEventClass event_list[MAX_EVENTS];
 | |
|         int event_type[MAX_EVENTS];
 | |
|         const char *names[MAX_EVENTS];
 | |
|         int total = 0;
 | |
| 
 | |
| #define ADD(type)                                     \
 | |
|         if (cnt >= MAX_EVENTS) abort();             \
 | |
|         names[cnt] = #type;                           \
 | |
|         type(dev, event_type[cnt], event_list[cnt]);  \
 | |
|         if (event_type[cnt]) ++cnt
 | |
| 
 | |
|         dev = XOpenDevice(display, devInfo[device].id);
 | |
|         ADD(DeviceKeyPress);
 | |
|         ADD(DeviceKeyRelease);
 | |
|         ADD(DeviceButtonPress);
 | |
|         ADD(DeviceButtonRelease);
 | |
|         ADD(DeviceMotionNotify);
 | |
|         ADD(DeviceFocusIn);
 | |
|         ADD(DeviceFocusOut);
 | |
|         ADD(ProximityIn);
 | |
|         ADD(ProximityOut);
 | |
|         ADD(DeviceStateNotify);
 | |
|         ADD(DeviceMappingNotify);
 | |
|         ADD(ChangeDeviceNotify);
 | |
| 
 | |
|         for (i = 0; i < cnt; i++) {
 | |
|             printf("Waiting for %s events of type %d (%lu) on 0x%08lx\n",
 | |
|                    names[i],
 | |
|                    event_type[i], (unsigned long) event_list[i],
 | |
|                    (long unsigned) DefaultRootWindow(display));
 | |
|         }
 | |
|         XSelectExtensionEvent(display, DefaultRootWindow(display),
 | |
|                               event_list, cnt);
 | |
| 
 | |
|         for (;;) {
 | |
|             XEvent event;
 | |
| 
 | |
|             XNextEvent(display, &event);
 | |
|             for (i = 0; i < cnt; i++) {
 | |
|                 XDeviceMotionEvent *e = (XDeviceMotionEvent *) & event;
 | |
|                 XDeviceButtonEvent *b = (XDeviceButtonEvent *) & event;
 | |
| 
 | |
|                 if (event.type == event_type[i]) {
 | |
|                     printf("%s id=%lu (%d @ %d,%d; s=0x%04x, d=%d, t=%lu)"
 | |
|                            " axes_count=%d first=%d %d %d %d %d %d %d\n",
 | |
|                            names[i],
 | |
|                            (long unsigned) e->deviceid,
 | |
|                            e->type,
 | |
|                            e->x, e->y,
 | |
|                            e->device_state,
 | |
|                            b->button,
 | |
|                            (long unsigned) b->time,
 | |
|                            e->axes_count,
 | |
|                            e->first_axis,
 | |
|                            e->axis_data[0],
 | |
|                            e->axis_data[1],
 | |
|                            e->axis_data[2],
 | |
|                            e->axis_data[3], e->axis_data[4], e->axis_data[5]);
 | |
|                 }
 | |
|             }
 | |
|             ++total;
 | |
| #if 0
 | |
|             /* Used to check motion history for
 | |
|              * extension devices. */
 | |
|             if (!(total % 10)) {
 | |
|                 XDeviceTimeCoord *tc;
 | |
|                 int n, m, a;
 | |
|                 struct timeval tv;
 | |
|                 unsigned long ms;
 | |
| 
 | |
|                 gettimeofday(&tv, NULL);
 | |
|                 ms = tv.tv_sec * 1000 + tv.tv_usec / 1000;
 | |
|                 tc = XGetDeviceMotionEvents(display, dev, ms - 1000, ms,
 | |
|                                             &n, &m, &a);
 | |
|                 printf("Got %d events of mode %s with %d axes\n",
 | |
|                        n, m == Absolute ? "Absolute" : "Relative", a);
 | |
|                 for (i = 0; i < n && i < 10; i++) {
 | |
|                     printf("  %d: %lu %d %d\n",
 | |
|                            i, tc[i].time, tc[i].data[0], tc[i].data[1]);
 | |
|                 }
 | |
|                 XFreeDeviceMotionEvents(tc);
 | |
|             }
 | |
| #endif
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     XCloseDisplay(display);
 | |
|     return 0;
 | |
| }
 |