16747b715Smrg/** 26747b715Smrg * Copyright © 2009 Red Hat, Inc. 36747b715Smrg * 46747b715Smrg * Permission is hereby granted, free of charge, to any person obtaining a 56747b715Smrg * copy of this software and associated documentation files (the "Software"), 66747b715Smrg * to deal in the Software without restriction, including without limitation 76747b715Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 86747b715Smrg * and/or sell copies of the Software, and to permit persons to whom the 96747b715Smrg * Software is furnished to do so, subject to the following conditions: 106747b715Smrg * 116747b715Smrg * The above copyright notice and this permission notice (including the next 126747b715Smrg * paragraph) shall be included in all copies or substantial portions of the 136747b715Smrg * Software. 146747b715Smrg * 156747b715Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 166747b715Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 176747b715Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 186747b715Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 196747b715Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 206747b715Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 216747b715Smrg * DEALINGS IN THE SOFTWARE. 226747b715Smrg */ 236747b715Smrg 24ed6184dfSmrg/* Test relies on assert() */ 25ed6184dfSmrg#undef NDEBUG 26ed6184dfSmrg 276747b715Smrg#ifdef HAVE_DIX_CONFIG_H 286747b715Smrg#include <dix-config.h> 296747b715Smrg#endif 306747b715Smrg 316747b715Smrg#include <stdint.h> 326747b715Smrg#include <X11/X.h> 336747b715Smrg#include <X11/Xproto.h> 346747b715Smrg#include <X11/extensions/XI2proto.h> 356747b715Smrg#include <X11/Xatom.h> 366747b715Smrg#include "inputstr.h" 376747b715Smrg#include "extinit.h" 3835c4bbdfSmrg#include "exglobals.h" 396747b715Smrg#include "scrnintstr.h" 406747b715Smrg#include "xkbsrv.h" 416747b715Smrg 426747b715Smrg#include "xiquerydevice.h" 436747b715Smrg 446747b715Smrg#include "protocol-common.h" 456747b715Smrg/* 466747b715Smrg * Protocol testing for XIQueryDevice request and reply. 476747b715Smrg * 486747b715Smrg * Test approach: 496747b715Smrg * Wrap WriteToClient to intercept server's reply. ProcXIQueryDevice returns 506747b715Smrg * data in two batches, once for the request, once for the trailing data 516747b715Smrg * with the device information. 526747b715Smrg * Repeatedly test with varying deviceids and check against data in reply. 536747b715Smrg */ 546747b715Smrg 556747b715Smrgstruct test_data { 566747b715Smrg int which_device; 576747b715Smrg int num_devices_in_reply; 586747b715Smrg}; 596747b715Smrg 605a7dfde8Smrgextern ClientRec client_window; 611b5d61b8Smrg 6235c4bbdfSmrgstatic void reply_XIQueryDevice_data(ClientPtr client, int len, char *data, 6335c4bbdfSmrg void *closure); 6435c4bbdfSmrgstatic void reply_XIQueryDevice(ClientPtr client, int len, char *data, 6535c4bbdfSmrg void *closure); 666747b715Smrg 676747b715Smrg/* reply handling for the first bytes that constitute the reply */ 6835c4bbdfSmrgstatic void 6935c4bbdfSmrgreply_XIQueryDevice(ClientPtr client, int len, char *data, void *userdata) 706747b715Smrg{ 7135c4bbdfSmrg xXIQueryDeviceReply *rep = (xXIQueryDeviceReply *) data; 7235c4bbdfSmrg struct test_data *querydata = (struct test_data *) userdata; 7335c4bbdfSmrg 7435c4bbdfSmrg if (client->swapped) { 7535c4bbdfSmrg swapl(&rep->length); 7635c4bbdfSmrg swaps(&rep->sequenceNumber); 7735c4bbdfSmrg swaps(&rep->num_devices); 786747b715Smrg } 796747b715Smrg 806747b715Smrg reply_check_defaults(rep, len, XIQueryDevice); 816747b715Smrg 826747b715Smrg if (querydata->which_device == XIAllDevices) 8335c4bbdfSmrg assert(rep->num_devices == devices.num_devices); 846747b715Smrg else if (querydata->which_device == XIAllMasterDevices) 8535c4bbdfSmrg assert(rep->num_devices == devices.num_master_devices); 866747b715Smrg else 8735c4bbdfSmrg assert(rep->num_devices == 1); 886747b715Smrg 896747b715Smrg querydata->num_devices_in_reply = rep->num_devices; 906747b715Smrg reply_handler = reply_XIQueryDevice_data; 916747b715Smrg} 926747b715Smrg 936747b715Smrg/* reply handling for the trailing bytes that constitute the device info */ 9435c4bbdfSmrgstatic void 9535c4bbdfSmrgreply_XIQueryDevice_data(ClientPtr client, int len, char *data, void *closure) 966747b715Smrg{ 976747b715Smrg int i, j; 9835c4bbdfSmrg struct test_data *querydata = (struct test_data *) closure; 996747b715Smrg 1006747b715Smrg DeviceIntPtr dev; 10135c4bbdfSmrg xXIDeviceInfo *info = (xXIDeviceInfo *) data; 1026747b715Smrg xXIAnyInfo *any; 1036747b715Smrg 10435c4bbdfSmrg for (i = 0; i < querydata->num_devices_in_reply; i++) { 10535c4bbdfSmrg if (client->swapped) { 10635c4bbdfSmrg swaps(&info->deviceid); 10735c4bbdfSmrg swaps(&info->attachment); 10835c4bbdfSmrg swaps(&info->use); 10935c4bbdfSmrg swaps(&info->num_classes); 11035c4bbdfSmrg swaps(&info->name_len); 1116747b715Smrg } 1126747b715Smrg 1136747b715Smrg if (querydata->which_device > XIAllMasterDevices) 11435c4bbdfSmrg assert(info->deviceid == querydata->which_device); 11535c4bbdfSmrg 11635c4bbdfSmrg assert(info->deviceid >= 2); /* 0 and 1 is reserved */ 11735c4bbdfSmrg 11835c4bbdfSmrg switch (info->deviceid) { 11935c4bbdfSmrg case 2: /* VCP */ 12035c4bbdfSmrg dev = devices.vcp; 12135c4bbdfSmrg assert(info->use == XIMasterPointer); 12235c4bbdfSmrg assert(info->attachment == devices.vck->id); 12335c4bbdfSmrg assert(info->num_classes == 3); /* 2 axes + button */ 12435c4bbdfSmrg break; 12535c4bbdfSmrg case 3: /* VCK */ 12635c4bbdfSmrg dev = devices.vck; 12735c4bbdfSmrg assert(info->use == XIMasterKeyboard); 12835c4bbdfSmrg assert(info->attachment == devices.vcp->id); 12935c4bbdfSmrg assert(info->num_classes == 1); 13035c4bbdfSmrg break; 13135c4bbdfSmrg case 4: /* mouse */ 13235c4bbdfSmrg dev = devices.mouse; 13335c4bbdfSmrg assert(info->use == XISlavePointer); 13435c4bbdfSmrg assert(info->attachment == devices.vcp->id); 13535c4bbdfSmrg assert(info->num_classes == 7); /* 4 axes + button + 2 scroll */ 13635c4bbdfSmrg break; 13735c4bbdfSmrg case 5: /* keyboard */ 13835c4bbdfSmrg dev = devices.kbd; 13935c4bbdfSmrg assert(info->use == XISlaveKeyboard); 14035c4bbdfSmrg assert(info->attachment == devices.vck->id); 14135c4bbdfSmrg assert(info->num_classes == 1); 14235c4bbdfSmrg break; 14335c4bbdfSmrg 14435c4bbdfSmrg default: 14535c4bbdfSmrg /* We shouldn't get here */ 14635c4bbdfSmrg assert(0); 14735c4bbdfSmrg break; 1486747b715Smrg } 14935c4bbdfSmrg assert(info->enabled == dev->enabled); 15035c4bbdfSmrg assert(info->name_len == strlen(dev->name)); 15135c4bbdfSmrg assert(strncmp((char *) &info[1], dev->name, info->name_len) == 0); 15235c4bbdfSmrg 15335c4bbdfSmrg any = 15435c4bbdfSmrg (xXIAnyInfo *) ((char *) &info[1] + ((info->name_len + 3) / 4) * 4); 15535c4bbdfSmrg for (j = 0; j < info->num_classes; j++) { 15635c4bbdfSmrg if (client->swapped) { 15735c4bbdfSmrg swaps(&any->type); 15835c4bbdfSmrg swaps(&any->length); 15935c4bbdfSmrg swaps(&any->sourceid); 1606747b715Smrg } 1616747b715Smrg 16235c4bbdfSmrg switch (info->deviceid) { 16335c4bbdfSmrg case 3: /* VCK and kbd have the same properties */ 16435c4bbdfSmrg case 5: 16535c4bbdfSmrg { 16635c4bbdfSmrg int k; 16735c4bbdfSmrg xXIKeyInfo *ki = (xXIKeyInfo *) any; 16835c4bbdfSmrg XkbDescPtr xkb = devices.vck->key->xkbInfo->desc; 16935c4bbdfSmrg uint32_t *kc; 17035c4bbdfSmrg 17135c4bbdfSmrg if (client->swapped) 17235c4bbdfSmrg swaps(&ki->num_keycodes); 17335c4bbdfSmrg 17435c4bbdfSmrg assert(any->type == XIKeyClass); 17535c4bbdfSmrg assert(ki->num_keycodes == 17635c4bbdfSmrg (xkb->max_key_code - xkb->min_key_code + 1)); 17735c4bbdfSmrg assert(any->length == (2 + ki->num_keycodes)); 17835c4bbdfSmrg 17935c4bbdfSmrg kc = (uint32_t *) &ki[1]; 18035c4bbdfSmrg for (k = 0; k < ki->num_keycodes; k++, kc++) { 18135c4bbdfSmrg if (client->swapped) 18235c4bbdfSmrg swapl(kc); 18335c4bbdfSmrg 18435c4bbdfSmrg assert(*kc >= xkb->min_key_code); 18535c4bbdfSmrg assert(*kc <= xkb->max_key_code); 18635c4bbdfSmrg } 18735c4bbdfSmrg break; 18835c4bbdfSmrg } 18935c4bbdfSmrg case 4: 1906747b715Smrg { 19135c4bbdfSmrg assert(any->type == XIButtonClass || 19235c4bbdfSmrg any->type == XIValuatorClass || 19335c4bbdfSmrg any->type == XIScrollClass); 19435c4bbdfSmrg 19535c4bbdfSmrg if (any->type == XIScrollClass) { 19635c4bbdfSmrg xXIScrollInfo *si = (xXIScrollInfo *) any; 19735c4bbdfSmrg 19835c4bbdfSmrg if (client->swapped) { 19935c4bbdfSmrg swaps(&si->number); 20035c4bbdfSmrg swaps(&si->scroll_type); 20135c4bbdfSmrg swapl(&si->increment.integral); 20235c4bbdfSmrg swapl(&si->increment.frac); 20335c4bbdfSmrg } 20435c4bbdfSmrg assert(si->length == 6); 20535c4bbdfSmrg assert(si->number == 2 || si->number == 3); 20635c4bbdfSmrg if (si->number == 2) { 20735c4bbdfSmrg assert(si->scroll_type == XIScrollTypeVertical); 20835c4bbdfSmrg assert(!si->flags); 2096747b715Smrg } 21035c4bbdfSmrg if (si->number == 3) { 21135c4bbdfSmrg assert(si->scroll_type == XIScrollTypeHorizontal); 21235c4bbdfSmrg assert(si->flags & XIScrollFlagPreferred); 21335c4bbdfSmrg assert(!(si->flags & ~XIScrollFlagPreferred)); 2146747b715Smrg } 21535c4bbdfSmrg 21635c4bbdfSmrg assert(si->increment.integral == si->number); 21735c4bbdfSmrg /* protocol-common.c sets up increments of 2.4 and 3.5 */ 21835c4bbdfSmrg assert(si->increment.frac > 0.3 * (1ULL << 32)); 21935c4bbdfSmrg assert(si->increment.frac < 0.6 * (1ULL << 32)); 22035c4bbdfSmrg } 22135c4bbdfSmrg 2226747b715Smrg } 22335c4bbdfSmrg /* fall through */ 22435c4bbdfSmrg case 2: /* VCP and mouse have the same properties except for scroll */ 22535c4bbdfSmrg { 22635c4bbdfSmrg if (info->deviceid == 2) /* VCP */ 22735c4bbdfSmrg assert(any->type == XIButtonClass || 22835c4bbdfSmrg any->type == XIValuatorClass); 22935c4bbdfSmrg 23035c4bbdfSmrg if (any->type == XIButtonClass) { 23135c4bbdfSmrg int l; 23235c4bbdfSmrg xXIButtonInfo *bi = (xXIButtonInfo *) any; 23335c4bbdfSmrg 23435c4bbdfSmrg if (client->swapped) 23535c4bbdfSmrg swaps(&bi->num_buttons); 23635c4bbdfSmrg 23735c4bbdfSmrg assert(bi->num_buttons == devices.vcp->button->numButtons); 23835c4bbdfSmrg 23935c4bbdfSmrg l = 2 + bi->num_buttons + 24035c4bbdfSmrg bytes_to_int32(bits_to_bytes(bi->num_buttons)); 24135c4bbdfSmrg assert(bi->length == l); 24235c4bbdfSmrg } 24335c4bbdfSmrg else if (any->type == XIValuatorClass) { 24435c4bbdfSmrg xXIValuatorInfo *vi = (xXIValuatorInfo *) any; 24535c4bbdfSmrg 24635c4bbdfSmrg if (client->swapped) { 24735c4bbdfSmrg swaps(&vi->number); 24835c4bbdfSmrg swapl(&vi->label); 24935c4bbdfSmrg swapl(&vi->min.integral); 25035c4bbdfSmrg swapl(&vi->min.frac); 25135c4bbdfSmrg swapl(&vi->max.integral); 25235c4bbdfSmrg swapl(&vi->max.frac); 25335c4bbdfSmrg swapl(&vi->resolution); 25435c4bbdfSmrg } 25535c4bbdfSmrg 25635c4bbdfSmrg assert(vi->length == 11); 257ed6184dfSmrg assert(vi->number >= 0); 258ed6184dfSmrg assert(vi->number < 4); 25935c4bbdfSmrg if (info->deviceid == 2) /* VCP */ 26035c4bbdfSmrg assert(vi->number < 2); 26135c4bbdfSmrg 26235c4bbdfSmrg assert(vi->mode == XIModeRelative); 26335c4bbdfSmrg /* device was set up as relative, so standard 26435c4bbdfSmrg * values here. */ 26535c4bbdfSmrg assert(vi->min.integral == -1); 26635c4bbdfSmrg assert(vi->min.frac == 0); 26735c4bbdfSmrg assert(vi->max.integral == -1); 26835c4bbdfSmrg assert(vi->max.frac == 0); 26935c4bbdfSmrg assert(vi->resolution == 0); 27035c4bbdfSmrg } 27135c4bbdfSmrg } 27235c4bbdfSmrg break; 27335c4bbdfSmrg } 27435c4bbdfSmrg any = (xXIAnyInfo *) (((char *) any) + any->length * 4); 2756747b715Smrg } 2766747b715Smrg 27735c4bbdfSmrg info = (xXIDeviceInfo *) any; 2786747b715Smrg } 2796747b715Smrg} 2806747b715Smrg 28135c4bbdfSmrgstatic void 28235c4bbdfSmrgrequest_XIQueryDevice(struct test_data *querydata, int deviceid, int error) 2836747b715Smrg{ 2846747b715Smrg int rc; 2856747b715Smrg ClientRec client; 2866747b715Smrg xXIQueryDeviceReq request; 2876747b715Smrg 2886747b715Smrg request_init(&request, XIQueryDevice); 2896747b715Smrg client = init_client(request.length, &request); 2906747b715Smrg reply_handler = reply_XIQueryDevice; 2916747b715Smrg 2926747b715Smrg querydata->which_device = deviceid; 2936747b715Smrg 2946747b715Smrg request.deviceid = deviceid; 2956747b715Smrg rc = ProcXIQueryDevice(&client); 29635c4bbdfSmrg assert(rc == error); 2976747b715Smrg 2986747b715Smrg if (rc != Success) 29935c4bbdfSmrg assert(client.errorValue == deviceid); 3006747b715Smrg 3016747b715Smrg reply_handler = reply_XIQueryDevice; 3026747b715Smrg 3036747b715Smrg client.swapped = TRUE; 30435c4bbdfSmrg swaps(&request.length); 30535c4bbdfSmrg swaps(&request.deviceid); 3066747b715Smrg rc = SProcXIQueryDevice(&client); 30735c4bbdfSmrg assert(rc == error); 3086747b715Smrg 3096747b715Smrg if (rc != Success) 31035c4bbdfSmrg assert(client.errorValue == deviceid); 3116747b715Smrg} 3126747b715Smrg 31335c4bbdfSmrgstatic void 31435c4bbdfSmrgtest_XIQueryDevice(void) 3156747b715Smrg{ 3166747b715Smrg int i; 3176747b715Smrg xXIQueryDeviceReq request; 3186747b715Smrg struct test_data data; 3196747b715Smrg 3206747b715Smrg reply_handler = reply_XIQueryDevice; 32135c4bbdfSmrg global_userdata = &data; 3226747b715Smrg request_init(&request, XIQueryDevice); 3236747b715Smrg 32435c4bbdfSmrg printf("Testing XIAllDevices.\n"); 3256747b715Smrg request_XIQueryDevice(&data, XIAllDevices, Success); 32635c4bbdfSmrg printf("Testing XIAllMasterDevices.\n"); 3276747b715Smrg request_XIQueryDevice(&data, XIAllMasterDevices, Success); 3286747b715Smrg 32935c4bbdfSmrg printf("Testing existing device ids.\n"); 3306747b715Smrg for (i = 2; i < 6; i++) 3316747b715Smrg request_XIQueryDevice(&data, i, Success); 3326747b715Smrg 33335c4bbdfSmrg printf("Testing non-existing device ids.\n"); 3346747b715Smrg for (i = 6; i <= 0xFFFF; i++) 3356747b715Smrg request_XIQueryDevice(&data, i, BadDevice); 3366747b715Smrg 3376747b715Smrg reply_handler = NULL; 3386747b715Smrg 3396747b715Smrg} 3406747b715Smrg 34135c4bbdfSmrgint 3421b5d61b8Smrgprotocol_xiquerydevice_test(void) 3436747b715Smrg{ 3446747b715Smrg init_simple(); 3456747b715Smrg 34635c4bbdfSmrg test_XIQueryDevice(); 3476747b715Smrg 34835c4bbdfSmrg return 0; 3496747b715Smrg} 350