1*4882a593Smuzhiyun /**
2*4882a593Smuzhiyun * Copyright © 2009 Red Hat, Inc.
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Permission is hereby granted, free of charge, to any person obtaining a
5*4882a593Smuzhiyun * copy of this software and associated documentation files (the "Software"),
6*4882a593Smuzhiyun * to deal in the Software without restriction, including without limitation
7*4882a593Smuzhiyun * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*4882a593Smuzhiyun * and/or sell copies of the Software, and to permit persons to whom the
9*4882a593Smuzhiyun * Software is furnished to do so, subject to the following conditions:
10*4882a593Smuzhiyun *
11*4882a593Smuzhiyun * The above copyright notice and this permission notice (including the next
12*4882a593Smuzhiyun * paragraph) shall be included in all copies or substantial portions of the
13*4882a593Smuzhiyun * Software.
14*4882a593Smuzhiyun *
15*4882a593Smuzhiyun * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*4882a593Smuzhiyun * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*4882a593Smuzhiyun * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*4882a593Smuzhiyun * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*4882a593Smuzhiyun * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*4882a593Smuzhiyun * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21*4882a593Smuzhiyun * DEALINGS IN THE SOFTWARE.
22*4882a593Smuzhiyun */
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #ifdef HAVE_DIX_CONFIG_H
25*4882a593Smuzhiyun #include <dix-config.h>
26*4882a593Smuzhiyun #endif
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun /*
29*4882a593Smuzhiyun * Protocol testing for XIGetSelectedEvents request.
30*4882a593Smuzhiyun *
31*4882a593Smuzhiyun * Tests include:
32*4882a593Smuzhiyun * BadWindow on wrong window.
33*4882a593Smuzhiyun * Zero-length masks if no masks are set.
34*4882a593Smuzhiyun * Valid masks for valid devices.
35*4882a593Smuzhiyun * Masks set on non-existent devices are not returned.
36*4882a593Smuzhiyun *
37*4882a593Smuzhiyun * Note that this test is not connected to the XISelectEvents request.
38*4882a593Smuzhiyun */
39*4882a593Smuzhiyun #include <stdint.h>
40*4882a593Smuzhiyun #include <X11/X.h>
41*4882a593Smuzhiyun #include <X11/Xproto.h>
42*4882a593Smuzhiyun #include <X11/extensions/XI2proto.h>
43*4882a593Smuzhiyun #include "inputstr.h"
44*4882a593Smuzhiyun #include "windowstr.h"
45*4882a593Smuzhiyun #include "extinit.h" /* for XInputExtensionInit */
46*4882a593Smuzhiyun #include "scrnintstr.h"
47*4882a593Smuzhiyun #include "xiselectev.h"
48*4882a593Smuzhiyun #include "exevents.h"
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun #include "protocol-common.h"
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun static void reply_XIGetSelectedEvents(ClientPtr client, int len, char *data,
53*4882a593Smuzhiyun void *userdata);
54*4882a593Smuzhiyun static void reply_XIGetSelectedEvents_data(ClientPtr client, int len,
55*4882a593Smuzhiyun char *data, void *userdata);
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun static struct {
58*4882a593Smuzhiyun int num_masks_expected;
59*4882a593Smuzhiyun unsigned char mask[MAXDEVICES][XI2LASTEVENT]; /* intentionally bigger */
60*4882a593Smuzhiyun int mask_len;
61*4882a593Smuzhiyun } test_data;
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun extern ClientRec client_window;
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun /* AddResource is called from XISetSEventMask, we don't need this */
66*4882a593Smuzhiyun Bool
__wrap_AddResource(XID id,RESTYPE type,void * value)67*4882a593Smuzhiyun __wrap_AddResource(XID id, RESTYPE type, void *value)
68*4882a593Smuzhiyun {
69*4882a593Smuzhiyun return TRUE;
70*4882a593Smuzhiyun }
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun static void
reply_XIGetSelectedEvents(ClientPtr client,int len,char * data,void * userdata)73*4882a593Smuzhiyun reply_XIGetSelectedEvents(ClientPtr client, int len, char *data, void *userdata)
74*4882a593Smuzhiyun {
75*4882a593Smuzhiyun xXIGetSelectedEventsReply *rep = (xXIGetSelectedEventsReply *) data;
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun if (client->swapped) {
78*4882a593Smuzhiyun swapl(&rep->length);
79*4882a593Smuzhiyun swaps(&rep->sequenceNumber);
80*4882a593Smuzhiyun swaps(&rep->num_masks);
81*4882a593Smuzhiyun }
82*4882a593Smuzhiyun
83*4882a593Smuzhiyun reply_check_defaults(rep, len, XIGetSelectedEvents);
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun assert(rep->num_masks == test_data.num_masks_expected);
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun reply_handler = reply_XIGetSelectedEvents_data;
88*4882a593Smuzhiyun }
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun static void
reply_XIGetSelectedEvents_data(ClientPtr client,int len,char * data,void * userdata)91*4882a593Smuzhiyun reply_XIGetSelectedEvents_data(ClientPtr client, int len, char *data,
92*4882a593Smuzhiyun void *userdata)
93*4882a593Smuzhiyun {
94*4882a593Smuzhiyun int i;
95*4882a593Smuzhiyun xXIEventMask *mask;
96*4882a593Smuzhiyun unsigned char *bitmask;
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun mask = (xXIEventMask *) data;
99*4882a593Smuzhiyun for (i = 0; i < test_data.num_masks_expected; i++) {
100*4882a593Smuzhiyun if (client->swapped) {
101*4882a593Smuzhiyun swaps(&mask->deviceid);
102*4882a593Smuzhiyun swaps(&mask->mask_len);
103*4882a593Smuzhiyun }
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun assert(mask->deviceid < 6);
106*4882a593Smuzhiyun assert(mask->mask_len <= (((XI2LASTEVENT + 8) / 8) + 3) / 4);
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun bitmask = (unsigned char *) &mask[1];
109*4882a593Smuzhiyun assert(memcmp(bitmask,
110*4882a593Smuzhiyun test_data.mask[mask->deviceid], mask->mask_len * 4) == 0);
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun mask =
113*4882a593Smuzhiyun (xXIEventMask *) ((char *) mask + mask->mask_len * 4 +
114*4882a593Smuzhiyun sizeof(xXIEventMask));
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun
117*4882a593Smuzhiyun }
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun static void
request_XIGetSelectedEvents(xXIGetSelectedEventsReq * req,int error)120*4882a593Smuzhiyun request_XIGetSelectedEvents(xXIGetSelectedEventsReq * req, int error)
121*4882a593Smuzhiyun {
122*4882a593Smuzhiyun int rc;
123*4882a593Smuzhiyun ClientRec client;
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun client = init_client(req->length, req);
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun reply_handler = reply_XIGetSelectedEvents;
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun rc = ProcXIGetSelectedEvents(&client);
130*4882a593Smuzhiyun assert(rc == error);
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun reply_handler = reply_XIGetSelectedEvents;
133*4882a593Smuzhiyun client.swapped = TRUE;
134*4882a593Smuzhiyun swapl(&req->win);
135*4882a593Smuzhiyun swaps(&req->length);
136*4882a593Smuzhiyun rc = SProcXIGetSelectedEvents(&client);
137*4882a593Smuzhiyun assert(rc == error);
138*4882a593Smuzhiyun }
139*4882a593Smuzhiyun
140*4882a593Smuzhiyun static void
test_XIGetSelectedEvents(void)141*4882a593Smuzhiyun test_XIGetSelectedEvents(void)
142*4882a593Smuzhiyun {
143*4882a593Smuzhiyun int i, j;
144*4882a593Smuzhiyun xXIGetSelectedEventsReq request;
145*4882a593Smuzhiyun ClientRec client = init_client(0, NULL);
146*4882a593Smuzhiyun unsigned char *mask;
147*4882a593Smuzhiyun DeviceIntRec dev;
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun request_init(&request, XIGetSelectedEvents);
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun printf("Testing for BadWindow on invalid window.\n");
152*4882a593Smuzhiyun request.win = None;
153*4882a593Smuzhiyun request_XIGetSelectedEvents(&request, BadWindow);
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun printf("Testing for zero-length (unset) masks.\n");
156*4882a593Smuzhiyun /* No masks set yet */
157*4882a593Smuzhiyun test_data.num_masks_expected = 0;
158*4882a593Smuzhiyun request.win = ROOT_WINDOW_ID;
159*4882a593Smuzhiyun request_XIGetSelectedEvents(&request, Success);
160*4882a593Smuzhiyun
161*4882a593Smuzhiyun request.win = CLIENT_WINDOW_ID;
162*4882a593Smuzhiyun request_XIGetSelectedEvents(&request, Success);
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun memset(test_data.mask, 0, sizeof(test_data.mask));
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun printf("Testing for valid masks\n");
167*4882a593Smuzhiyun memset(&dev, 0, sizeof(dev)); /* dev->id is enough for XISetEventMask */
168*4882a593Smuzhiyun request.win = ROOT_WINDOW_ID;
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun /* devices 6 - MAXDEVICES don't exist, they mustn't be included in the
171*4882a593Smuzhiyun * reply even if a mask is set */
172*4882a593Smuzhiyun for (j = 0; j < MAXDEVICES; j++) {
173*4882a593Smuzhiyun test_data.num_masks_expected = min(j + 1, devices.num_devices + 2);
174*4882a593Smuzhiyun dev.id = j;
175*4882a593Smuzhiyun mask = test_data.mask[j];
176*4882a593Smuzhiyun /* bits one-by-one */
177*4882a593Smuzhiyun for (i = 0; i < XI2LASTEVENT; i++) {
178*4882a593Smuzhiyun SetBit(mask, i);
179*4882a593Smuzhiyun XISetEventMask(&dev, &root, &client, (i + 8) / 8, mask);
180*4882a593Smuzhiyun request_XIGetSelectedEvents(&request, Success);
181*4882a593Smuzhiyun ClearBit(mask, i);
182*4882a593Smuzhiyun }
183*4882a593Smuzhiyun
184*4882a593Smuzhiyun /* all valid mask bits */
185*4882a593Smuzhiyun for (i = 0; i < XI2LASTEVENT; i++) {
186*4882a593Smuzhiyun SetBit(mask, i);
187*4882a593Smuzhiyun XISetEventMask(&dev, &root, &client, (i + 8) / 8, mask);
188*4882a593Smuzhiyun request_XIGetSelectedEvents(&request, Success);
189*4882a593Smuzhiyun }
190*4882a593Smuzhiyun }
191*4882a593Smuzhiyun
192*4882a593Smuzhiyun printf("Testing removing all masks\n");
193*4882a593Smuzhiyun /* Unset all masks one-by-one */
194*4882a593Smuzhiyun for (j = MAXDEVICES - 1; j >= 0; j--) {
195*4882a593Smuzhiyun if (j < devices.num_devices + 2)
196*4882a593Smuzhiyun test_data.num_masks_expected--;
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun mask = test_data.mask[j];
199*4882a593Smuzhiyun memset(mask, 0, XI2LASTEVENT);
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun dev.id = j;
202*4882a593Smuzhiyun XISetEventMask(&dev, &root, &client, 0, NULL);
203*4882a593Smuzhiyun
204*4882a593Smuzhiyun request_XIGetSelectedEvents(&request, Success);
205*4882a593Smuzhiyun }
206*4882a593Smuzhiyun }
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun int
protocol_xigetselectedevents_test(void)209*4882a593Smuzhiyun protocol_xigetselectedevents_test(void)
210*4882a593Smuzhiyun {
211*4882a593Smuzhiyun init_simple();
212*4882a593Smuzhiyun enable_GrabButton_wrap = 0;
213*4882a593Smuzhiyun enable_XISetEventMask_wrap = 0;
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun test_XIGetSelectedEvents();
216*4882a593Smuzhiyun
217*4882a593Smuzhiyun return 0;
218*4882a593Smuzhiyun }
219