xref: /openbsd-src/lib/libusbhid/parse.c (revision 2c53affbcc0119d6480b86c18f2790523b6a0aad)
1*2c53affbSjmc /*	$OpenBSD: parse.c,v 1.13 2022/12/27 17:10:07 jmc Exp $	*/
27517eab2Snate /*	$NetBSD: parse.c,v 1.2 2001/12/29 20:44:22 augustss Exp $	*/
3be8f1931Spvalchev 
4be8f1931Spvalchev /*
57517eab2Snate  * Copyright (c) 1999, 2001 Lennart Augustsson <augustss@netbsd.org>
6be8f1931Spvalchev  * All rights reserved.
7be8f1931Spvalchev  *
8be8f1931Spvalchev  * Redistribution and use in source and binary forms, with or without
9be8f1931Spvalchev  * modification, are permitted provided that the following conditions
10be8f1931Spvalchev  * are met:
11be8f1931Spvalchev  * 1. Redistributions of source code must retain the above copyright
12be8f1931Spvalchev  *    notice, this list of conditions and the following disclaimer.
13be8f1931Spvalchev  * 2. Redistributions in binary form must reproduce the above copyright
14be8f1931Spvalchev  *    notice, this list of conditions and the following disclaimer in the
15be8f1931Spvalchev  *    documentation and/or other materials provided with the distribution.
16be8f1931Spvalchev  *
17be8f1931Spvalchev  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18be8f1931Spvalchev  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19be8f1931Spvalchev  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20be8f1931Spvalchev  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21be8f1931Spvalchev  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22be8f1931Spvalchev  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23be8f1931Spvalchev  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24be8f1931Spvalchev  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25be8f1931Spvalchev  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26be8f1931Spvalchev  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27be8f1931Spvalchev  * SUCH DAMAGE.
28be8f1931Spvalchev  */
29be8f1931Spvalchev 
30be8f1931Spvalchev #include <stdlib.h>
31be8f1931Spvalchev #include <string.h>
32be8f1931Spvalchev 
33be8f1931Spvalchev #include <dev/usb/usb.h>
34be8f1931Spvalchev #include <dev/usb/usbhid.h>
35be8f1931Spvalchev 
36be8f1931Spvalchev #include "usbhid.h"
37be8f1931Spvalchev #include "usbvar.h"
38be8f1931Spvalchev 
39be8f1931Spvalchev #define	MAXUSAGE 100
40b3eedbb4Smpi #define	MAXPUSH 4
41b3eedbb4Smpi #define	MAXID 64
42b3eedbb4Smpi #define	ITEMTYPES 3
43674f1e1fSjasper 
44b3eedbb4Smpi struct hid_pos_data {
45b3eedbb4Smpi 	int32_t rid;
46b3eedbb4Smpi 	uint32_t pos[ITEMTYPES];
47be8f1931Spvalchev };
48be8f1931Spvalchev 
49b3eedbb4Smpi struct hid_data {
50b3eedbb4Smpi 	const uint8_t *start;
51b3eedbb4Smpi 	const uint8_t *end;
52b3eedbb4Smpi 	const uint8_t *p;
53b3eedbb4Smpi 	struct hid_item cur[MAXPUSH];
54b3eedbb4Smpi 	struct hid_pos_data last_pos[MAXID];
55b3eedbb4Smpi 	uint32_t pos[ITEMTYPES];
56b3eedbb4Smpi 	int32_t usages_min[MAXUSAGE];
57b3eedbb4Smpi 	int32_t usages_max[MAXUSAGE];
58b3eedbb4Smpi 	int32_t usage_last;	/* last seen usage */
59b3eedbb4Smpi 	uint32_t loc_size;	/* last seen size */
60b3eedbb4Smpi 	uint32_t loc_count;	/* last seen count */
61b3eedbb4Smpi 	uint8_t	kindset;	/* we have 5 kinds so 8 bits are enough */
62b3eedbb4Smpi 	uint8_t	pushlevel;	/* current pushlevel */
63b3eedbb4Smpi 	uint8_t	ncount;		/* end usage item count */
64b3eedbb4Smpi 	uint8_t icount;		/* current usage item count */
65b3eedbb4Smpi 	uint8_t	nusage;		/* end "usages_min/max" index */
66b3eedbb4Smpi 	uint8_t	iusage;		/* current "usages_min/max" index */
67b3eedbb4Smpi 	uint8_t ousage;		/* current "usages_min/max" offset */
68b3eedbb4Smpi 	uint8_t	susage;		/* usage set flags */
69b3eedbb4Smpi 	int32_t	reportid;	/* requested report ID */
7095fec9d1Smpi 	struct hid_item savedcoll; /* save coll until we know the ID */
7195fec9d1Smpi 	uint8_t hassavedcoll;
72b3eedbb4Smpi };
737517eab2Snate 
74be8f1931Spvalchev static void
hid_clear_local(hid_item_t * c)75be8f1931Spvalchev hid_clear_local(hid_item_t *c)
76be8f1931Spvalchev {
77674f1e1fSjasper 
78be8f1931Spvalchev 	c->usage = 0;
79be8f1931Spvalchev 	c->usage_minimum = 0;
80be8f1931Spvalchev 	c->usage_maximum = 0;
81be8f1931Spvalchev 	c->designator_index = 0;
82be8f1931Spvalchev 	c->designator_minimum = 0;
83be8f1931Spvalchev 	c->designator_maximum = 0;
84be8f1931Spvalchev 	c->string_index = 0;
85be8f1931Spvalchev 	c->string_minimum = 0;
86be8f1931Spvalchev 	c->string_maximum = 0;
87be8f1931Spvalchev 	c->set_delimiter = 0;
88be8f1931Spvalchev }
89be8f1931Spvalchev 
90b3eedbb4Smpi static void
hid_switch_rid(struct hid_data * s,struct hid_item * c,int32_t next_rID)91b3eedbb4Smpi hid_switch_rid(struct hid_data *s, struct hid_item *c, int32_t next_rID)
92b3eedbb4Smpi {
93b3eedbb4Smpi 	uint8_t i, j;
94b3eedbb4Smpi 
95b3eedbb4Smpi 	/* check for same report ID - optimise */
96b3eedbb4Smpi 
97b3eedbb4Smpi 	if (c->report_ID == next_rID)
98b3eedbb4Smpi 		return;
99b3eedbb4Smpi 
100b3eedbb4Smpi 	/* save current position for current rID */
101b3eedbb4Smpi 
102b3eedbb4Smpi 	if (c->report_ID == 0) {
103b3eedbb4Smpi 		i = 0;
104b3eedbb4Smpi 	} else {
105b3eedbb4Smpi 		for (i = 1; i != MAXID; i++) {
106b3eedbb4Smpi 			if (s->last_pos[i].rid == c->report_ID)
107b3eedbb4Smpi 				break;
108b3eedbb4Smpi 			if (s->last_pos[i].rid == 0)
109b3eedbb4Smpi 				break;
110b3eedbb4Smpi 		}
111b3eedbb4Smpi 	}
112b3eedbb4Smpi 	if (i != MAXID) {
113b3eedbb4Smpi 		s->last_pos[i].rid = c->report_ID;
114b3eedbb4Smpi 		for (j = 0; j < ITEMTYPES; j++)
115b3eedbb4Smpi 			s->last_pos[i].pos[j] = s->pos[j];
116b3eedbb4Smpi 	}
117b3eedbb4Smpi 
118b3eedbb4Smpi 	/* store next report ID */
119b3eedbb4Smpi 
120b3eedbb4Smpi 	c->report_ID = next_rID;
121b3eedbb4Smpi 
122b3eedbb4Smpi 	/* lookup last position for next rID */
123b3eedbb4Smpi 
124b3eedbb4Smpi 	if (next_rID == 0) {
125b3eedbb4Smpi 		i = 0;
126b3eedbb4Smpi 	} else {
127b3eedbb4Smpi 		for (i = 1; i != MAXID; i++) {
128b3eedbb4Smpi 			if (s->last_pos[i].rid == next_rID)
129b3eedbb4Smpi 				break;
130b3eedbb4Smpi 			if (s->last_pos[i].rid == 0)
131b3eedbb4Smpi 				break;
132b3eedbb4Smpi 		}
133b3eedbb4Smpi 	}
134b3eedbb4Smpi 	if (i != MAXID) {
135b3eedbb4Smpi 		s->last_pos[i].rid = next_rID;
136b3eedbb4Smpi 		for (j = 0; j < ITEMTYPES; j++)
137b3eedbb4Smpi 			s->pos[j] = s->last_pos[i].pos[j];
138b3eedbb4Smpi 	} else {
139b3eedbb4Smpi 		for (j = 0; j < ITEMTYPES; j++)
140b3eedbb4Smpi 			s->pos[j] = 0;	/* Out of RID entries. */
141b3eedbb4Smpi 	}
142b3eedbb4Smpi }
143b3eedbb4Smpi 
144be8f1931Spvalchev hid_data_t
hid_start_parse(report_desc_t d,int kindset,int id)145674f1e1fSjasper hid_start_parse(report_desc_t d, int kindset, int id)
146be8f1931Spvalchev {
1477517eab2Snate 	struct hid_data *s;
148be8f1931Spvalchev 
149ee353af9Stedu 	s = calloc(1, sizeof *s);
15078e5c336Sderaadt 	if (s == NULL)
15178e5c336Sderaadt 		return (NULL);
152be8f1931Spvalchev 	s->start = s->p = d->data;
153be8f1931Spvalchev 	s->end = d->data + d->size;
154be8f1931Spvalchev 	s->kindset = kindset;
155674f1e1fSjasper 	s->reportid = id;
15695fec9d1Smpi 	s->hassavedcoll = 0;
157be8f1931Spvalchev 	return (s);
158be8f1931Spvalchev }
159be8f1931Spvalchev 
160be8f1931Spvalchev void
hid_end_parse(hid_data_t s)161be8f1931Spvalchev hid_end_parse(hid_data_t s)
162be8f1931Spvalchev {
163be8f1931Spvalchev 
164b3eedbb4Smpi 	if (s == NULL)
165b3eedbb4Smpi 		return;
166b3eedbb4Smpi 
167674f1e1fSjasper 	free(s);
1689b8cf20dSyuo }
1699b8cf20dSyuo 
170b3eedbb4Smpi static uint8_t
hid_get_byte(struct hid_data * s,const uint16_t wSize)171b3eedbb4Smpi hid_get_byte(struct hid_data *s, const uint16_t wSize)
172be8f1931Spvalchev {
173b3eedbb4Smpi 	const uint8_t *ptr;
174b3eedbb4Smpi 	uint8_t retval;
1757517eab2Snate 
176b3eedbb4Smpi 	ptr = s->p;
177be8f1931Spvalchev 
178b3eedbb4Smpi 	/* check if end is reached */
179b3eedbb4Smpi 	if (ptr == s->end)
180b3eedbb4Smpi 		return (0);
181b3eedbb4Smpi 
182b3eedbb4Smpi 	/* read out a byte */
183b3eedbb4Smpi 	retval = *ptr;
184b3eedbb4Smpi 
185b3eedbb4Smpi 	/* check if data pointer can be advanced by "wSize" bytes */
186b3eedbb4Smpi 	if ((s->end - ptr) < wSize)
187b3eedbb4Smpi 		ptr = s->end;
188b3eedbb4Smpi 	else
189b3eedbb4Smpi 		ptr += wSize;
190b3eedbb4Smpi 
191b3eedbb4Smpi 	/* update pointer */
192b3eedbb4Smpi 	s->p = ptr;
193b3eedbb4Smpi 
194b3eedbb4Smpi 	return (retval);
195b3eedbb4Smpi }
196674f1e1fSjasper 
19795fec9d1Smpi #define REPORT_SAVED_COLL \
19895fec9d1Smpi 	do { \
19995fec9d1Smpi 		if (s->hassavedcoll) { \
20095fec9d1Smpi 			*h = s->savedcoll; \
20195fec9d1Smpi 			h->report_ID = c->report_ID; \
20295fec9d1Smpi 			s->hassavedcoll = 0; \
20395fec9d1Smpi 			return (1); \
20495fec9d1Smpi 		} \
20595fec9d1Smpi 	} while(0)
20695fec9d1Smpi 
207674f1e1fSjasper static int
hid_get_item_raw(hid_data_t s,hid_item_t * h)208674f1e1fSjasper hid_get_item_raw(hid_data_t s, hid_item_t *h)
209674f1e1fSjasper {
21095fec9d1Smpi 	hid_item_t nc, *c;
211b3eedbb4Smpi 	unsigned int bTag, bType, bSize;
212b3eedbb4Smpi 	int32_t mask;
213b3eedbb4Smpi 	int32_t dval;
2149b8cf20dSyuo 
215b3eedbb4Smpi 	if (s == NULL)
216674f1e1fSjasper 		return (0);
2179b8cf20dSyuo 
21863aecb60Sjsg 	if (s->pushlevel >= MAXPUSH)
21963aecb60Sjsg 		return (0);
22063aecb60Sjsg 
221b3eedbb4Smpi 	c = &s->cur[s->pushlevel];
222b3eedbb4Smpi 
223b3eedbb4Smpi  top:
224b3eedbb4Smpi 	/* check if there is an array of items */
225b3eedbb4Smpi 	if (s->icount < s->ncount) {
22695fec9d1Smpi 		REPORT_SAVED_COLL;
227b3eedbb4Smpi 		/* get current usage */
228b3eedbb4Smpi 		if (s->iusage < s->nusage) {
229b3eedbb4Smpi 			dval = s->usages_min[s->iusage] + s->ousage;
230b3eedbb4Smpi 			c->usage = dval;
231b3eedbb4Smpi 			s->usage_last = dval;
232b3eedbb4Smpi 			if (dval == s->usages_max[s->iusage]) {
233b3eedbb4Smpi 				s->iusage ++;
234b3eedbb4Smpi 				s->ousage = 0;
235b3eedbb4Smpi 			} else {
236b3eedbb4Smpi 				s->ousage ++;
237b3eedbb4Smpi 			}
238b3eedbb4Smpi 		} else {
239b3eedbb4Smpi 			/* Using last usage */
240b3eedbb4Smpi 			dval = s->usage_last;
241b3eedbb4Smpi 		}
242b3eedbb4Smpi 		s->icount ++;
243b3eedbb4Smpi 		/*
244b3eedbb4Smpi 		 * Only copy HID item, increment position and return
245b3eedbb4Smpi 		 * if correct kindset!
246b3eedbb4Smpi 		 */
247b3eedbb4Smpi 		if (s->kindset & (1 << c->kind)) {
248b3eedbb4Smpi 			*h = *c;
249b3eedbb4Smpi 			h->pos = s->pos[c->kind];
250b3eedbb4Smpi 			s->pos[c->kind] += c->report_size * c->report_count;
251b3eedbb4Smpi 			return (1);
252b3eedbb4Smpi 		}
253b3eedbb4Smpi 	}
254b3eedbb4Smpi 
255b3eedbb4Smpi 	/* reset state variables */
256b3eedbb4Smpi 	s->icount = 0;
257b3eedbb4Smpi 	s->ncount = 0;
258b3eedbb4Smpi 	s->iusage = 0;
259b3eedbb4Smpi 	s->nusage = 0;
260b3eedbb4Smpi 	s->susage = 0;
261b3eedbb4Smpi 	s->ousage = 0;
262b3eedbb4Smpi 	hid_clear_local(c);
263b3eedbb4Smpi 
264b3eedbb4Smpi 	/* get next item */
265b3eedbb4Smpi 	while (s->p != s->end) {
266b3eedbb4Smpi 
267b3eedbb4Smpi 		bSize = hid_get_byte(s, 1);
268be8f1931Spvalchev 		if (bSize == 0xfe) {
269be8f1931Spvalchev 			/* long item */
270b3eedbb4Smpi 			bSize = hid_get_byte(s, 1);
271b3eedbb4Smpi 			bSize |= hid_get_byte(s, 1) << 8;
272b3eedbb4Smpi 			bTag = hid_get_byte(s, 1);
273b3eedbb4Smpi 			bType = 0xff;	/* XXX what should it be */
274be8f1931Spvalchev 		} else {
275be8f1931Spvalchev 			/* short item */
276be8f1931Spvalchev 			bTag = bSize >> 4;
277be8f1931Spvalchev 			bType = (bSize >> 2) & 3;
278be8f1931Spvalchev 			bSize &= 3;
279b3eedbb4Smpi 			if (bSize == 3)
280b3eedbb4Smpi 				bSize = 4;
281be8f1931Spvalchev 		}
282b3eedbb4Smpi 
283be8f1931Spvalchev 		switch(bSize) {
284be8f1931Spvalchev 		case 0:
285be8f1931Spvalchev 			dval = 0;
286b3eedbb4Smpi 			mask = 0;
287be8f1931Spvalchev 			break;
288be8f1931Spvalchev 		case 1:
289b3eedbb4Smpi 			dval = (int8_t)hid_get_byte(s, 1);
290b3eedbb4Smpi 			mask = 0xFF;
291be8f1931Spvalchev 			break;
292be8f1931Spvalchev 		case 2:
293b3eedbb4Smpi 			dval = hid_get_byte(s, 1);
294b3eedbb4Smpi 			dval |= hid_get_byte(s, 1) << 8;
295b3eedbb4Smpi 			dval = (int16_t)dval;
296b3eedbb4Smpi 			mask = 0xFFFF;
297be8f1931Spvalchev 			break;
298be8f1931Spvalchev 		case 4:
299b3eedbb4Smpi 			dval = hid_get_byte(s, 1);
300b3eedbb4Smpi 			dval |= hid_get_byte(s, 1) << 8;
301b3eedbb4Smpi 			dval |= hid_get_byte(s, 1) << 16;
302b3eedbb4Smpi 			dval |= hid_get_byte(s, 1) << 24;
303b3eedbb4Smpi 			mask = 0xFFFFFFFF;
304be8f1931Spvalchev 			break;
305be8f1931Spvalchev 		default:
306b3eedbb4Smpi 			dval = hid_get_byte(s, bSize);
307b3eedbb4Smpi 			continue;
308be8f1931Spvalchev 		}
309be8f1931Spvalchev 
310be8f1931Spvalchev 		switch (bType) {
311be8f1931Spvalchev 		case 0:		/* Main */
312be8f1931Spvalchev 			switch (bTag) {
313be8f1931Spvalchev 			case 8:	/* Input */
314b3eedbb4Smpi 				c->kind = hid_input;
315674f1e1fSjasper 				c->flags = dval;
316b3eedbb4Smpi 		ret:
317b3eedbb4Smpi 				c->report_count = s->loc_count;
318b3eedbb4Smpi 				c->report_size = s->loc_size;
319b3eedbb4Smpi 
320be8f1931Spvalchev 				if (c->flags & HIO_VARIABLE) {
321b3eedbb4Smpi 					/* range check usage count */
322b3eedbb4Smpi 					if (c->report_count > 255) {
323b3eedbb4Smpi 						s->ncount = 255;
324b3eedbb4Smpi 					} else
325b3eedbb4Smpi 						s->ncount = c->report_count;
326b3eedbb4Smpi 
327b3eedbb4Smpi 					/*
328b3eedbb4Smpi 					 * The "top" loop will return
329b3eedbb4Smpi 					 * one and one item:
330b3eedbb4Smpi 					 */
331be8f1931Spvalchev 					c->report_count = 1;
3327517eab2Snate 					c->usage_minimum = 0;
3337517eab2Snate 					c->usage_maximum = 0;
334b3eedbb4Smpi 				} else {
335b3eedbb4Smpi 					s->ncount = 1;
336be8f1931Spvalchev 				}
337be8f1931Spvalchev 				goto top;
338b3eedbb4Smpi 
339be8f1931Spvalchev 			case 9:	/* Output */
340b3eedbb4Smpi 				c->kind = hid_output;
341b3eedbb4Smpi 				c->flags = dval;
342be8f1931Spvalchev 				goto ret;
343be8f1931Spvalchev 			case 10:	/* Collection */
344be8f1931Spvalchev 				c->kind = hid_collection;
345be8f1931Spvalchev 				c->collection = dval;
346be8f1931Spvalchev 				c->collevel++;
347b3eedbb4Smpi 				c->usage = s->usage_last;
34895fec9d1Smpi 				nc = *c;
34995fec9d1Smpi 				if (s->hassavedcoll) {
35095fec9d1Smpi 					*h = s->savedcoll;
35195fec9d1Smpi 					h->report_ID = nc.report_ID;
35295fec9d1Smpi 					s->savedcoll = nc;
353be8f1931Spvalchev 					return (1);
35495fec9d1Smpi 				} else {
35595fec9d1Smpi 					s->hassavedcoll = 1;
35695fec9d1Smpi 					s->savedcoll = nc;
35795fec9d1Smpi 				}
35895fec9d1Smpi 				goto top;
359be8f1931Spvalchev 			case 11:	/* Feature */
360b3eedbb4Smpi 				c->kind = hid_feature;
361b3eedbb4Smpi 				c->flags = dval;
362be8f1931Spvalchev 				goto ret;
363be8f1931Spvalchev 			case 12:	/* End collection */
36495fec9d1Smpi 				REPORT_SAVED_COLL;
365be8f1931Spvalchev 				c->kind = hid_endcollection;
366b3eedbb4Smpi 				if (c->collevel == 0) {
367b3eedbb4Smpi 					/* Invalid end collection. */
368b3eedbb4Smpi 					return (0);
369b3eedbb4Smpi 				}
370be8f1931Spvalchev 				c->collevel--;
371be8f1931Spvalchev 				*h = *c;
372be8f1931Spvalchev 				return (1);
373be8f1931Spvalchev 			default:
374b3eedbb4Smpi 				break;
375be8f1931Spvalchev 			}
3767517eab2Snate 			break;
377be8f1931Spvalchev 
378be8f1931Spvalchev 		case 1:		/* Global */
379be8f1931Spvalchev 			switch (bTag) {
380be8f1931Spvalchev 			case 0:
381be8f1931Spvalchev 				c->_usage_page = dval << 16;
382be8f1931Spvalchev 				break;
383be8f1931Spvalchev 			case 1:
384be8f1931Spvalchev 				c->logical_minimum = dval;
385be8f1931Spvalchev 				break;
386be8f1931Spvalchev 			case 2:
387be8f1931Spvalchev 				c->logical_maximum = dval;
388be8f1931Spvalchev 				break;
389be8f1931Spvalchev 			case 3:
39012c240a5Sjsg 				c->physical_minimum = dval;
391be8f1931Spvalchev 				break;
392be8f1931Spvalchev 			case 4:
393be8f1931Spvalchev 				c->physical_maximum = dval;
394be8f1931Spvalchev 				break;
395be8f1931Spvalchev 			case 5:
396be8f1931Spvalchev 				c->unit_exponent = dval;
397be8f1931Spvalchev 				break;
398be8f1931Spvalchev 			case 6:
399be8f1931Spvalchev 				c->unit = dval;
400be8f1931Spvalchev 				break;
401be8f1931Spvalchev 			case 7:
402b3eedbb4Smpi 				/* mask because value is unsigned */
403b3eedbb4Smpi 				s->loc_size = dval & mask;
404be8f1931Spvalchev 				break;
405be8f1931Spvalchev 			case 8:
406b3eedbb4Smpi 				hid_switch_rid(s, c, dval & mask);
407be8f1931Spvalchev 				break;
408be8f1931Spvalchev 			case 9:
409b3eedbb4Smpi 				/* mask because value is unsigned */
410b3eedbb4Smpi 				s->loc_count = dval & mask;
411be8f1931Spvalchev 				break;
412be8f1931Spvalchev 			case 10:	/* Push */
41363aecb60Sjsg 				if (s->pushlevel < MAXPUSH - 1) {
414b3eedbb4Smpi 					s->pushlevel++;
415b3eedbb4Smpi 					s->cur[s->pushlevel] = *c;
416b3eedbb4Smpi 					/* store size and count */
417b3eedbb4Smpi 					c->report_size = s->loc_size;
418b3eedbb4Smpi 					c->report_count = s->loc_count;
419b3eedbb4Smpi 					/* update current item pointer */
420b3eedbb4Smpi 					c = &s->cur[s->pushlevel];
421b3eedbb4Smpi 				}
422be8f1931Spvalchev 				break;
423be8f1931Spvalchev 			case 11:	/* Pop */
42463aecb60Sjsg 				if (s->pushlevel > 0) {
425b3eedbb4Smpi 					s->pushlevel--;
426b3eedbb4Smpi 					c = &s->cur[s->pushlevel];
427b3eedbb4Smpi 					/* restore size and count */
428b3eedbb4Smpi 					s->loc_size = c->report_size;
429b3eedbb4Smpi 					s->loc_count = c->report_count;
430b3eedbb4Smpi 					c->report_size = 0;
431b3eedbb4Smpi 					c->report_count = 0;
432b3eedbb4Smpi 				}
433be8f1931Spvalchev 				break;
434be8f1931Spvalchev 			default:
435b3eedbb4Smpi 				break;
436be8f1931Spvalchev 			}
437be8f1931Spvalchev 			break;
438be8f1931Spvalchev 		case 2:		/* Local */
439be8f1931Spvalchev 			switch (bTag) {
440be8f1931Spvalchev 			case 0:
441b3eedbb4Smpi 				if (bSize != 4)
442b3eedbb4Smpi 					dval = (dval & mask) | c->_usage_page;
443b3eedbb4Smpi 
444b3eedbb4Smpi 				/* set last usage, in case of a collection */
445b3eedbb4Smpi 				s->usage_last = dval;
446b3eedbb4Smpi 
447b3eedbb4Smpi 				if (s->nusage < MAXUSAGE) {
448b3eedbb4Smpi 					s->usages_min[s->nusage] = dval;
449b3eedbb4Smpi 					s->usages_max[s->nusage] = dval;
450b3eedbb4Smpi 					s->nusage ++;
451b3eedbb4Smpi 				}
452be8f1931Spvalchev 				/* else XXX */
453b3eedbb4Smpi 
454b3eedbb4Smpi 				/* clear any pending usage sets */
455b3eedbb4Smpi 				s->susage = 0;
456be8f1931Spvalchev 				break;
457be8f1931Spvalchev 			case 1:
458b3eedbb4Smpi 				s->susage |= 1;
459b3eedbb4Smpi 
460b3eedbb4Smpi 				if (bSize != 4)
461b3eedbb4Smpi 					dval = (dval & mask) | c->_usage_page;
462b3eedbb4Smpi 				c->usage_minimum = dval;
463b3eedbb4Smpi 
464b3eedbb4Smpi 				goto check_set;
465674f1e1fSjasper 			case 2:
466b3eedbb4Smpi 				s->susage |= 2;
467b3eedbb4Smpi 
468b3eedbb4Smpi 				if (bSize != 4)
469b3eedbb4Smpi 					dval = (dval & mask) | c->_usage_page;
470b3eedbb4Smpi 				c->usage_maximum = dval;
471b3eedbb4Smpi 
472b3eedbb4Smpi 			check_set:
473b3eedbb4Smpi 				if (s->susage != 3)
474b3eedbb4Smpi 					break;
475b3eedbb4Smpi 
476b3eedbb4Smpi 				/* sanity check */
477b3eedbb4Smpi 				if ((s->nusage < MAXUSAGE) &&
478b3eedbb4Smpi 				    (c->usage_minimum <= c->usage_maximum)) {
479b3eedbb4Smpi 					/* add usage range */
480b3eedbb4Smpi 					s->usages_min[s->nusage] =
481b3eedbb4Smpi 					    c->usage_minimum;
482b3eedbb4Smpi 					s->usages_max[s->nusage] =
483b3eedbb4Smpi 					    c->usage_maximum;
484b3eedbb4Smpi 					s->nusage ++;
485b3eedbb4Smpi 				}
486b3eedbb4Smpi 				/* else XXX */
487b3eedbb4Smpi 
488b3eedbb4Smpi 				s->susage = 0;
489be8f1931Spvalchev 				break;
490be8f1931Spvalchev 			case 3:
491be8f1931Spvalchev 				c->designator_index = dval;
492be8f1931Spvalchev 				break;
493be8f1931Spvalchev 			case 4:
494be8f1931Spvalchev 				c->designator_minimum = dval;
495be8f1931Spvalchev 				break;
496be8f1931Spvalchev 			case 5:
497be8f1931Spvalchev 				c->designator_maximum = dval;
498be8f1931Spvalchev 				break;
499be8f1931Spvalchev 			case 7:
500be8f1931Spvalchev 				c->string_index = dval;
501be8f1931Spvalchev 				break;
502be8f1931Spvalchev 			case 8:
503be8f1931Spvalchev 				c->string_minimum = dval;
504be8f1931Spvalchev 				break;
505be8f1931Spvalchev 			case 9:
506be8f1931Spvalchev 				c->string_maximum = dval;
507be8f1931Spvalchev 				break;
508be8f1931Spvalchev 			case 10:
509be8f1931Spvalchev 				c->set_delimiter = dval;
510be8f1931Spvalchev 				break;
511be8f1931Spvalchev 			default:
5129b8cf20dSyuo 				break;
513be8f1931Spvalchev 			}
514be8f1931Spvalchev 			break;
515be8f1931Spvalchev 		default:
5169b8cf20dSyuo 			break;
517be8f1931Spvalchev 		}
518be8f1931Spvalchev 	}
5199b8cf20dSyuo 	return (0);
520be8f1931Spvalchev }
521be8f1931Spvalchev 
522be8f1931Spvalchev int
hid_get_item(hid_data_t s,hid_item_t * h)523b3eedbb4Smpi hid_get_item(hid_data_t s, hid_item_t *h)
524b3eedbb4Smpi {
525b3eedbb4Smpi 	int r;
526b3eedbb4Smpi 
527b3eedbb4Smpi 	for (;;) {
528b3eedbb4Smpi 		r = hid_get_item_raw(s, h);
529b3eedbb4Smpi 		if (r <= 0 || s->reportid == -1 || h->report_ID == s->reportid)
530b3eedbb4Smpi 			break;
531b3eedbb4Smpi 	}
532b3eedbb4Smpi 	return (r);
533b3eedbb4Smpi }
534b3eedbb4Smpi 
535b3eedbb4Smpi int
hid_report_size(report_desc_t r,enum hid_kind k,int id)5367517eab2Snate hid_report_size(report_desc_t r, enum hid_kind k, int id)
537be8f1931Spvalchev {
538be8f1931Spvalchev 	struct hid_data *d;
5399b8cf20dSyuo 	struct hid_item h;
5409b8cf20dSyuo 	uint32_t temp;
5419b8cf20dSyuo 	uint32_t hpos;
5429b8cf20dSyuo 	uint32_t lpos;
5439b8cf20dSyuo 	int report_id = 0;
5449b8cf20dSyuo 
5459b8cf20dSyuo 	hpos = 0;
546b3eedbb4Smpi 	lpos = 0xFFFFFFFF;
547be8f1931Spvalchev 
548be8f1931Spvalchev 	memset(&h, 0, sizeof h);
5497517eab2Snate 	for (d = hid_start_parse(r, 1 << k, id); hid_get_item(d, &h); ) {
550b3eedbb4Smpi 		if (h.kind == k) {
5519b8cf20dSyuo 			/* compute minimum */
5529b8cf20dSyuo 			if (lpos > h.pos)
5539b8cf20dSyuo 				lpos = h.pos;
5549b8cf20dSyuo 			/* compute end position */
5559b8cf20dSyuo 			temp = h.pos + (h.report_size * h.report_count);
5569b8cf20dSyuo 			/* compute maximum */
5579b8cf20dSyuo 			if (hpos < temp)
5589b8cf20dSyuo 				hpos = temp;
5599b8cf20dSyuo 			if (h.report_ID != 0)
5609b8cf20dSyuo 				report_id = 1;
561be8f1931Spvalchev 		}
562be8f1931Spvalchev 	}
563be8f1931Spvalchev 	hid_end_parse(d);
5649b8cf20dSyuo 
565*2c53affbSjmc 	/* safety check - can happen in case of corrupt descriptors */
5669b8cf20dSyuo 	if (lpos > hpos)
5679b8cf20dSyuo 		temp = 0;
5689b8cf20dSyuo 	else
5699b8cf20dSyuo 		temp = hpos - lpos;
5709b8cf20dSyuo 
5714e4160e6Smpi 	/* No extra byte for the reportID because the kernel skips it. */
5724e4160e6Smpi 	return ((temp + 7) / 8);
573be8f1931Spvalchev }
574be8f1931Spvalchev 
575be8f1931Spvalchev int
hid_locate(report_desc_t desc,unsigned int u,enum hid_kind k,hid_item_t * h,int id)5767517eab2Snate hid_locate(report_desc_t desc, unsigned int u, enum hid_kind k,
5777517eab2Snate     hid_item_t *h, int id)
578be8f1931Spvalchev {
5799b8cf20dSyuo 	struct hid_data *d;
580be8f1931Spvalchev 
5817517eab2Snate 	for (d = hid_start_parse(desc, 1 << k, id); hid_get_item(d, h); ) {
582be8f1931Spvalchev 		if (h->kind == k && !(h->flags & HIO_CONST) && h->usage == u) {
583be8f1931Spvalchev 			hid_end_parse(d);
584be8f1931Spvalchev 			return (1);
585be8f1931Spvalchev 		}
586be8f1931Spvalchev 	}
587be8f1931Spvalchev 	hid_end_parse(d);
588be8f1931Spvalchev 	h->report_size = 0;
589be8f1931Spvalchev 	return (0);
590be8f1931Spvalchev }
591