Imported Upstream version 1.15.1
[deb_xorg-server.git] / Xi / queryst.c
CommitLineData
a09e091a
JB
1/*
2
3Copyright 1998, 1998 The Open Group
4
5Permission to use, copy, modify, distribute, and sell this software and its
6documentation for any purpose is hereby granted without fee, provided that
7the above copyright notice appear in all copies and that both that
8copyright notice and this permission notice appear in supporting
9documentation.
10
11The above copyright notice and this permission notice shall be included
12in all copies or substantial portions of the Software.
13
14THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
17IN NO EVENT SHALL THE OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR
18OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20OTHER DEALINGS IN THE SOFTWARE.
21
22Except as contained in this notice, the name of The Open Group shall
23not be used in advertising or otherwise to promote the sale, use or
24other dealings in this Software without prior written authorization
25from The Open Group.
26
27*/
28
29/***********************************************************************
30 *
31 * Request to query the state of an extension input device.
32 *
33 */
34
35#ifdef HAVE_DIX_CONFIG_H
36#include <dix-config.h>
37#endif
38
39#include "inputstr.h" /* DeviceIntPtr */
40#include "windowstr.h" /* window structure */
41#include <X11/extensions/XI.h>
42#include <X11/extensions/XIproto.h>
43#include "exevents.h"
44#include "exglobals.h"
45#include "xkbsrv.h"
46#include "xkbstr.h"
47
48#include "queryst.h"
49
50/***********************************************************************
51 *
52 * This procedure allows a client to query the state of a device.
53 *
54 */
55
56int
57SProcXQueryDeviceState(ClientPtr client)
58{
59 REQUEST(xQueryDeviceStateReq);
60 swaps(&stuff->length);
61 return (ProcXQueryDeviceState(client));
62}
63
64/***********************************************************************
65 *
66 * This procedure allows frozen events to be routed.
67 *
68 */
69
70int
71ProcXQueryDeviceState(ClientPtr client)
72{
73 int rc, i;
74 int num_classes = 0;
75 int total_length = 0;
76 char *buf, *savbuf;
77 KeyClassPtr k;
78 xKeyState *tk;
79 ButtonClassPtr b;
80 xButtonState *tb;
81 ValuatorClassPtr v;
82 xValuatorState *tv;
83 xQueryDeviceStateReply rep;
84 DeviceIntPtr dev;
85 double *values;
86
87 REQUEST(xQueryDeviceStateReq);
88 REQUEST_SIZE_MATCH(xQueryDeviceStateReq);
89
90 rc = dixLookupDevice(&dev, stuff->deviceid, client, DixReadAccess);
91 if (rc != Success && rc != BadAccess)
92 return rc;
93
94 v = dev->valuator;
95 if (v != NULL && v->motionHintWindow != NULL)
96 MaybeStopDeviceHint(dev, client);
97
98 k = dev->key;
99 if (k != NULL) {
100 total_length += sizeof(xKeyState);
101 num_classes++;
102 }
103
104 b = dev->button;
105 if (b != NULL) {
106 total_length += sizeof(xButtonState);
107 num_classes++;
108 }
109
110 if (v != NULL) {
111 total_length += (sizeof(xValuatorState) + (v->numAxes * sizeof(int)));
112 num_classes++;
113 }
114 buf = (char *) calloc(total_length, 1);
115 if (!buf)
116 return BadAlloc;
117 savbuf = buf;
118
119 if (k != NULL) {
120 tk = (xKeyState *) buf;
121 tk->class = KeyClass;
122 tk->length = sizeof(xKeyState);
123 tk->num_keys = k->xkbInfo->desc->max_key_code -
124 k->xkbInfo->desc->min_key_code + 1;
125 if (rc != BadAccess)
126 for (i = 0; i < 32; i++)
127 tk->keys[i] = k->down[i];
128 buf += sizeof(xKeyState);
129 }
130
131 if (b != NULL) {
132 tb = (xButtonState *) buf;
133 tb->class = ButtonClass;
134 tb->length = sizeof(xButtonState);
135 tb->num_buttons = b->numButtons;
136 if (rc != BadAccess)
137 memcpy(tb->buttons, b->down, sizeof(b->down));
138 buf += sizeof(xButtonState);
139 }
140
141 if (v != NULL) {
142 tv = (xValuatorState *) buf;
143 tv->class = ValuatorClass;
144 tv->length = sizeof(xValuatorState) + v->numAxes * 4;
145 tv->num_valuators = v->numAxes;
146 tv->mode = valuator_get_mode(dev, 0);
147 tv->mode |= (dev->proximity &&
148 !dev->proximity->in_proximity) ? OutOfProximity : 0;
149 buf += sizeof(xValuatorState);
150 for (i = 0, values = v->axisVal; i < v->numAxes; i++) {
151 if (rc != BadAccess)
152 *((int *) buf) = *values;
153 values++;
154 if (client->swapped) {
155 swapl((int *) buf);
156 }
157 buf += sizeof(int);
158 }
159 }
160
161 rep = (xQueryDeviceStateReply) {
162 .repType = X_Reply,
163 .RepType = X_QueryDeviceState,
164 .sequenceNumber = client->sequence,
165 .length = bytes_to_int32(total_length),
166 .num_classes = num_classes
167 };
168 WriteReplyToClient(client, sizeof(xQueryDeviceStateReply), &rep);
169 if (total_length > 0)
170 WriteToClient(client, total_length, savbuf);
171 free(savbuf);
172 return Success;
173}
174
175/***********************************************************************
176 *
177 * This procedure writes the reply for the XQueryDeviceState function,
178 * if the client and server have a different byte ordering.
179 *
180 */
181
182void
183SRepXQueryDeviceState(ClientPtr client, int size, xQueryDeviceStateReply * rep)
184{
185 swaps(&rep->sequenceNumber);
186 swapl(&rep->length);
187 WriteToClient(client, size, rep);
188}