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