2 * Copyright 2007 Peter Hutterer
4 * Permission to use, copy, modify, distribute, and sell this software and its
5 * documentation for any purpose is hereby granted without fee, provided that
6 * the above copyright notice appear in all copies and that both that
7 * copyright notice and this permission notice appear in supporting
10 * The above copyright notice and this permission notice shall be included
11 * in all copies or substantial portions of the Software.
13 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
14 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
15 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
16 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY CLAIM, DAMAGES OR
17 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
18 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
19 * OTHER DEALINGS IN THE SOFTWARE.
21 * Except as contained in this notice, the name of the author shall
22 * not be used in advertising or otherwise to promote the sale, use or
23 * other dealings in this Software without prior written authorization
32 #include <X11/Xatom.h>
33 #include <X11/extensions/XIproto.h>
37 static Atom
parse_atom(Display
*dpy
, char *name
) {
41 for (i
= 0; name
[i
] != '\0'; i
++) {
42 if (!isdigit(name
[i
])) {
51 return XInternAtom(dpy
, name
, False
);
55 print_property(Display
*dpy
, XDevice
* dev
, Atom property
)
60 unsigned long nitems
, bytes_after
;
61 unsigned char *data
, *ptr
;
62 int j
, done
= False
, size
;
64 name
= XGetAtomName(dpy
, property
);
65 printf("\t%s (%ld):\t", name
, property
);
67 if (XGetDeviceProperty(dpy
, dev
, property
, 0, 1000, False
,
68 AnyPropertyType
, &act_type
, &act_format
,
69 &nitems
, &bytes_after
, &data
) == Success
)
71 Atom float_atom
= XInternAtom(dpy
, "FLOAT", True
);
80 case 8: size
= sizeof(char); break;
81 case 16: size
= sizeof(short); break;
82 case 32: size
= sizeof(long); break;
85 for (j
= 0; j
< nitems
; j
++)
93 printf("%d", *((char*)ptr
));
96 printf("%d", *((short*)ptr
));
99 printf("%ld", *((long*)ptr
));
106 printf("Unknown string format.\n");
110 printf("\"%s\"", ptr
);
111 j
+= strlen((char*)ptr
); /* The loop's j++ jumps over the
113 ptr
+= strlen((char*)ptr
); /* ptr += size below jumps over
118 Atom a
= *(Atom
*)ptr
;
119 printf("\"%s\" (%d)",
120 (a
) ? XGetAtomName(dpy
, a
) : "None",
125 if (float_atom
!= None
&& act_type
== float_atom
)
127 printf("%f", *((float*)ptr
));
131 printf("\t... of unknown type %s\n",
132 XGetAtomName(dpy
, act_type
));
147 printf("\tFetch failure\n");
152 list_props_xi1(Display
*dpy
, int argc
, char** argv
, char* name
, char *desc
)
162 fprintf(stderr
, "Usage: xinput %s %s\n", name
, desc
);
166 for (i
= 0; i
< argc
; i
++)
168 info
= find_device_info(dpy
, argv
[i
], False
);
171 fprintf(stderr
, "unable to find device %s\n", argv
[i
]);
175 dev
= XOpenDevice(dpy
, info
->id
);
178 fprintf(stderr
, "unable to open device '%s'\n", info
->name
);
182 props
= XListDeviceProperties(dpy
, dev
, &nprops
);
185 printf("Device '%s' does not report any properties.\n", info
->name
);
189 printf("Device '%s':\n", info
->name
);
192 print_property(dpy
, dev
, props
[nprops
]);
196 XCloseDevice(dpy
, dev
);
202 set_int_prop(Display
*dpy
, int argc
, char** argv
, char* n
, char *desc
)
210 int format
, nelements
= 0;
214 fprintf(stderr
, "Usage: xinput %s %s\n", n
, desc
);
218 info
= find_device_info(dpy
, argv
[0], False
);
221 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
225 dev
= XOpenDevice(dpy
, info
->id
);
228 fprintf(stderr
, "unable to open device %s\n", argv
[0]);
234 prop
= parse_atom(dpy
, name
);
236 nelements
= argc
- 3;
237 format
= atoi(argv
[2]);
238 if (format
!= 8 && format
!= 16 && format
!= 32)
240 fprintf(stderr
, "Invalid format %d\n", format
);
244 data
= calloc(nelements
, sizeof(long));
245 for (i
= 0; i
< nelements
; i
++)
250 *(((char*)data
) + i
) = atoi(argv
[3 + i
]);
253 *(((short*)data
) + i
) = atoi(argv
[3 + i
]);
256 *(((long*)data
) + i
) = atoi(argv
[3 + i
]);
261 XChangeDeviceProperty(dpy
, dev
, prop
, XA_INTEGER
, format
, PropModeReplace
,
262 (unsigned char*)data
, nelements
);
265 XCloseDevice(dpy
, dev
);
270 set_float_prop(Display
*dpy
, int argc
, char** argv
, char* n
, char *desc
)
274 Atom prop
, float_atom
;
283 fprintf(stderr
, "Usage: xinput %s %s\n", n
, desc
);
287 info
= find_device_info(dpy
, argv
[0], False
);
290 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
294 dev
= XOpenDevice(dpy
, info
->id
);
297 fprintf(stderr
, "unable to open device %s\n", argv
[0]);
303 prop
= parse_atom(dpy
, name
);
305 nelements
= argc
- 2;
307 float_atom
= XInternAtom(dpy
, "FLOAT", False
);
309 if (float_atom
== (Atom
)0)
311 fprintf(stderr
, "no FLOAT atom present in server\n");
315 if (sizeof(float) != 4)
317 fprintf(stderr
, "sane FP required\n");
321 data
= calloc(nelements
, sizeof(long));
322 for (i
= 0; i
< nelements
; i
++)
324 *((float*)(data
+ i
)) = strtod(argv
[2 + i
], &endptr
);
325 if(endptr
== argv
[2 + i
]){
326 fprintf(stderr
, "argument %s could not be parsed\n", argv
[2 + i
]);
331 XChangeDeviceProperty(dpy
, dev
, prop
, float_atom
, 32, PropModeReplace
,
332 (unsigned char*)data
, nelements
);
335 XCloseDevice(dpy
, dev
);
340 int watch_props(Display
*dpy
, int argc
, char** argv
, char* n
, char *desc
)
345 XDevicePropertyNotifyEvent
*dpev
;
348 XEventClass cls_prop
;
350 if (list_props(dpy
, argc
, argv
, n
, desc
) != EXIT_SUCCESS
)
353 info
= find_device_info(dpy
, argv
[0], False
);
356 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
360 dev
= XOpenDevice(dpy
, info
->id
);
363 fprintf(stderr
, "unable to open device '%s'\n", info
->name
);
367 DevicePropertyNotify(dev
, type_prop
, cls_prop
);
368 XSelectExtensionEvent(dpy
, DefaultRootWindow(dpy
), &cls_prop
, 1);
372 XNextEvent(dpy
, &ev
);
374 dpev
= (XDevicePropertyNotifyEvent
*)&ev
;
375 if (dpev
->type
!= type_prop
)
378 name
= XGetAtomName(dpy
, dpev
->atom
);
379 printf("Property '%s' changed.\n", name
);
380 print_property(dpy
, dev
, dpev
->atom
);
383 XCloseDevice(dpy
, dev
);
387 delete_prop_xi1(Display
*dpy
, int argc
, char** argv
, char* n
, char *desc
)
394 info
= find_device_info(dpy
, argv
[0], False
);
397 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
401 dev
= XOpenDevice(dpy
, info
->id
);
404 fprintf(stderr
, "unable to open device '%s'\n", info
->name
);
410 prop
= parse_atom(dpy
, name
);
412 XDeleteDeviceProperty(dpy
, dev
, prop
);
414 XCloseDevice(dpy
, dev
);
419 set_atom_prop(Display
*dpy
, int argc
, char** argv
, char* n
, char *desc
)
432 fprintf(stderr
, "Usage: xinput %s %s\n", n
, desc
);
436 info
= find_device_info(dpy
, argv
[0], False
);
439 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
443 dev
= XOpenDevice(dpy
, info
->id
);
446 fprintf(stderr
, "unable to open device %s\n", argv
[0]);
452 prop
= parse_atom(dpy
, name
);
454 nelements
= argc
- 2;
455 data
= calloc(nelements
, sizeof(Atom
));
456 for (i
= 0; i
< nelements
; i
++)
460 for(j
= 0; j
< strlen(name
); j
++) {
461 if (!isdigit(name
[j
])) {
468 data
[i
] = XInternAtom(dpy
, name
, False
);
471 data
[i
] = atoi(name
);
472 XFree(XGetAtomName(dpy
, data
[i
]));
476 XChangeDeviceProperty(dpy
, dev
, prop
, XA_ATOM
, 32, PropModeReplace
,
477 (unsigned char*)data
, nelements
);
480 XCloseDevice(dpy
, dev
);
485 set_prop_xi1(Display
*dpy
, int argc
, char **argv
, char *n
, char *desc
)
494 int format
, nelements
= 0;
495 unsigned long act_nitems
, bytes_after
;
506 fprintf(stderr
, "Usage: xinput %s %s\n", n
, desc
);
510 info
= find_device_info(dpy
, argv
[0], False
);
513 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
517 dev
= XOpenDevice(dpy
, info
->id
);
520 fprintf(stderr
, "unable to open device %s\n", argv
[0]);
526 prop
= parse_atom(dpy
, name
);
529 fprintf(stderr
, "invalid property %s\n", name
);
533 float_atom
= XInternAtom(dpy
, "FLOAT", False
);
535 nelements
= argc
- 2;
536 if (XGetDeviceProperty(dpy
, dev
, prop
, 0, 0, False
, AnyPropertyType
,
537 &type
, &format
, &act_nitems
, &bytes_after
, &data
.c
)
539 fprintf(stderr
, "failed to get property type and format for %s\n", name
);
546 fprintf(stderr
, "property %s doesn't exist\n", name
);
550 data
.c
= calloc(nelements
, sizeof(long));
552 for (i
= 0; i
< nelements
; i
++)
554 if (type
== XA_INTEGER
) {
558 data
.c
[i
] = atoi(argv
[2 + i
]);
561 data
.s
[i
] = atoi(argv
[2 + i
]);
564 data
.l
[i
] = atoi(argv
[2 + i
]);
567 fprintf(stderr
, "unexpected size for property %s", name
);
570 } else if (type
== float_atom
) {
572 fprintf(stderr
, "unexpected format %d for property %s\n",
576 *(float *)(data
.l
+ i
) = strtod(argv
[2 + i
], &endptr
);
577 if (endptr
== argv
[2 + i
]) {
578 fprintf(stderr
, "argument %s could not be parsed\n", argv
[2 + i
]);
581 } else if (type
== XA_ATOM
) {
583 fprintf(stderr
, "unexpected format %d for property %s\n",
587 data
.a
[i
] = parse_atom(dpy
, argv
[2 + i
]);
589 fprintf(stderr
, "unexpected type for property %s\n", name
);
594 XChangeDeviceProperty(dpy
, dev
, prop
, type
, format
, PropModeReplace
,
597 XCloseDevice(dpy
, dev
);
603 print_property_xi2(Display
*dpy
, int deviceid
, Atom property
)
608 unsigned long nitems
, bytes_after
;
609 unsigned char *data
, *ptr
;
612 name
= XGetAtomName(dpy
, property
);
613 printf("\t%s (%ld):\t", name
, property
);
615 if (XIGetProperty(dpy
, deviceid
, property
, 0, 1000, False
,
616 AnyPropertyType
, &act_type
, &act_format
,
617 &nitems
, &bytes_after
, &data
) == Success
)
619 Atom float_atom
= XInternAtom(dpy
, "FLOAT", True
);
624 printf("<no items>");
626 for (j
= 0; j
< nitems
; j
++)
634 printf("%d", *((int8_t*)ptr
));
637 printf("%d", *((int16_t*)ptr
));
640 printf("%d", *((int32_t*)ptr
));
647 printf("Unknown string format.\n");
651 printf("\"%s\"", ptr
);
652 j
+= strlen((char*)ptr
); /* The loop's j++ jumps over the
654 ptr
+= strlen((char*)ptr
); /* ptr += size below jumps over
659 Atom a
= *(Atom
*)ptr
;
660 printf("\"%s\" (%d)",
661 (a
) ? XGetAtomName(dpy
, a
) : "None",
667 if (float_atom
!= None
&& act_type
== float_atom
)
669 printf("%f", *((float*)ptr
));
673 printf("\t... of unknown type %s\n",
674 XGetAtomName(dpy
, act_type
));
689 printf("\tFetch failure\n");
694 list_props_xi2(Display
*dpy
, int argc
, char** argv
, char* name
, char *desc
)
703 fprintf(stderr
, "Usage: xinput %s %s\n", name
, desc
);
707 for (i
= 0; i
< argc
; i
++)
709 info
= xi2_find_device_info(dpy
, argv
[i
]);
712 fprintf(stderr
, "unable to find device %s\n", argv
[i
]);
716 props
= XIListProperties(dpy
, info
->deviceid
, &nprops
);
719 printf("Device '%s' does not report any properties.\n", info
->name
);
723 printf("Device '%s':\n", info
->name
);
726 print_property_xi2(dpy
, info
->deviceid
, props
[nprops
]);
735 delete_prop_xi2(Display
*dpy
, int argc
, char** argv
, char* n
, char *desc
)
741 info
= xi2_find_device_info(dpy
, argv
[0]);
744 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
750 prop
= parse_atom(dpy
, name
);
752 XIDeleteProperty(dpy
, info
->deviceid
, prop
);
758 set_prop_xi2(Display
*dpy
, int argc
, char **argv
, char *n
, char *desc
)
766 int format
, nelements
= 0;
767 unsigned long act_nitems
, bytes_after
;
777 fprintf(stderr
, "Usage: xinput %s %s\n", n
, desc
);
781 info
= xi2_find_device_info(dpy
, argv
[0]);
784 fprintf(stderr
, "unable to find device %s\n", argv
[0]);
790 prop
= parse_atom(dpy
, name
);
793 fprintf(stderr
, "invalid property %s\n", name
);
797 float_atom
= XInternAtom(dpy
, "FLOAT", False
);
799 nelements
= argc
- 2;
800 if (XIGetProperty(dpy
, info
->deviceid
, prop
, 0, 0, False
, AnyPropertyType
,
801 &type
, &format
, &act_nitems
, &bytes_after
, &data
.c
)
803 fprintf(stderr
, "failed to get property type and format for %s\n", name
);
810 fprintf(stderr
, "property %s doesn't exist\n", name
);
814 data
.c
= calloc(nelements
, sizeof(int32_t));
816 for (i
= 0; i
< nelements
; i
++)
818 if (type
== XA_INTEGER
) {
822 data
.c
[i
] = atoi(argv
[2 + i
]);
825 data
.s
[i
] = atoi(argv
[2 + i
]);
828 data
.l
[i
] = atoi(argv
[2 + i
]);
831 fprintf(stderr
, "unexpected size for property %s", name
);
834 } else if (type
== float_atom
) {
836 fprintf(stderr
, "unexpected format %d for property %s\n",
840 *(float *)(data
.l
+ i
) = strtod(argv
[2 + i
], &endptr
);
841 if (endptr
== argv
[2 + i
]) {
842 fprintf(stderr
, "argument %s could not be parsed\n", argv
[2 + i
]);
845 } else if (type
== XA_ATOM
) {
847 fprintf(stderr
, "unexpected format %d for property %s\n",
851 data
.l
[i
] = parse_atom(dpy
, argv
[2 + i
]);
853 fprintf(stderr
, "unexpected type for property %s\n", name
);
858 XIChangeProperty(dpy
, info
->deviceid
, prop
, type
, format
, PropModeReplace
,
865 int list_props(Display
*display
, int argc
, char *argv
[], char *name
,
869 if (xinput_version(display
) == XI_2_Major
)
870 return list_props_xi2(display
, argc
, argv
, name
, desc
);
872 return list_props_xi1(display
, argc
, argv
, name
, desc
);
876 int delete_prop(Display
*display
, int argc
, char *argv
[], char *name
,
880 if (xinput_version(display
) == XI_2_Major
)
881 return delete_prop_xi2(display
, argc
, argv
, name
, desc
);
883 return delete_prop_xi1(display
, argc
, argv
, name
, desc
);
887 int set_prop(Display
*display
, int argc
, char *argv
[], char *name
,
891 if (xinput_version(display
) == XI_2_Major
)
892 return set_prop_xi2(display
, argc
, argv
, name
, desc
);
894 return set_prop_xi1(display
, argc
, argv
, name
, desc
);