source: mainline/uspace/lib/usbhid/src/hidpath.c@ 1a5eca4

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 1a5eca4 was 07525cd, checked in by Martin Decky <martin@…>, 12 years ago

cstyle

  • Property mode set to 100644
File size: 10.7 KB
RevLine 
[9d05599]1/*
2 * Copyright (c) 2011 Matej Klonfar
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
[160b75e]29/** @addtogroup libusbhid
[9d05599]30 * @{
31 */
32/** @file
33 * HID report descriptor and report data parser implementation.
34 */
[faa44e58]35#include <usb/hid/hidparser.h>
[9d05599]36#include <errno.h>
37#include <stdio.h>
38#include <malloc.h>
39#include <mem.h>
40#include <usb/debug.h>
41#include <assert.h>
42
[a76b01b4]43
[5499a8b]44/**
45 * Compares two usages if they are same or not or one of the usages is not
46 * set.
47 *
48 * @param usage1
49 * @param usage2
50 * @return boolean
51 */
52#define USB_HID_SAME_USAGE(usage1, usage2) \
53 ((usage1 == usage2) || (usage1 == 0) || (usage2 == 0))
[9d05599]54
[5499a8b]55/**
56 * Compares two usage pages if they are same or not or one of them is not set.
57 *
58 * @param page1
59 * @param page2
60 * @return boolean
61 */
62#define USB_HID_SAME_USAGE_PAGE(page1, page2) \
63 ((page1 == page2) || (page1 == 0) || (page2 == 0))
[3a6e423]64
[a76b01b4]65
[9d05599]66/**
67 * Appends one item (couple of usage_path and usage) into the usage path
68 * structure
69 *
70 * @param usage_path Usage path structure
71 * @param usage_page Usage page constant
72 * @param usage Usage constant
73 * @return Error code
74 */
75int usb_hid_report_path_append_item(usb_hid_report_path_t *usage_path,
76 int32_t usage_page, int32_t usage)
77{
[4bb7ffe]78 usb_hid_report_usage_path_t *item
79 = malloc(sizeof(usb_hid_report_usage_path_t));
[9d05599]80
[4bb7ffe]81 if (item == NULL) {
[9d05599]82 return ENOMEM;
83 }
[b72efe8]84 link_initialize(&item->rpath_items_link);
[9d05599]85
86 item->usage = usage;
87 item->usage_page = usage_page;
88 item->flags = 0;
89
[b72efe8]90 list_append (&item->rpath_items_link, &usage_path->items);
[9d05599]91 usage_path->depth++;
92 return EOK;
93}
94
[a76b01b4]95
[9d05599]96/**
97 * Removes last item from the usage path structure
98 * @param usage_path
99 * @return void
100 */
101void usb_hid_report_remove_last_item(usb_hid_report_path_t *usage_path)
102{
[b72efe8]103 link_t *item_link;
[9d05599]104 usb_hid_report_usage_path_t *item;
105
[b72efe8]106 if(!list_empty(&usage_path->items)){
107 item_link = list_last(&usage_path->items);
108 item = list_get_instance(item_link,
109 usb_hid_report_usage_path_t, rpath_items_link);
110 list_remove(item_link);
[9d05599]111 usage_path->depth--;
112 free(item);
113 }
114}
115
[a76b01b4]116
[9d05599]117/**
118 * Nulls last item of the usage path structure.
119 *
120 * @param usage_path
121 * @return void
122 */
123void usb_hid_report_null_last_item(usb_hid_report_path_t *usage_path)
124{
125 usb_hid_report_usage_path_t *item;
126
[b72efe8]127 if(!list_empty(&usage_path->items)){
128 item = list_get_instance(list_last(&usage_path->items),
129 usb_hid_report_usage_path_t, rpath_items_link);
[5499a8b]130
[9d05599]131 memset(item, 0, sizeof(usb_hid_report_usage_path_t));
132 }
133}
134
[a76b01b4]135
[9d05599]136/**
137 * Modifies last item of usage path structure by given usage page or usage
138 *
139 * @param usage_path Opaque usage path structure
140 * @param tag Class of currently processed tag (Usage page tag falls into Global
141 * class but Usage tag into the Local)
142 * @param data Value of the processed tag
143 * @return void
144 */
145void usb_hid_report_set_last_item(usb_hid_report_path_t *usage_path,
146 int32_t tag, int32_t data)
147{
148 usb_hid_report_usage_path_t *item;
149
[b72efe8]150 if(!list_empty(&usage_path->items)){
151 item = list_get_instance(list_last(&usage_path->items),
152 usb_hid_report_usage_path_t, rpath_items_link);
[9d05599]153
154 switch(tag) {
155 case USB_HID_TAG_CLASS_GLOBAL:
156 item->usage_page = data;
157 break;
158 case USB_HID_TAG_CLASS_LOCAL:
159 item->usage = data;
160 break;
161 }
162 }
163
164}
165
[a76b01b4]166
[5499a8b]167/**
168 *
169 *
170 *
171 *
172 */
[9d05599]173void usb_hid_print_usage_path(usb_hid_report_path_t *path)
174{
175 usb_log_debug("USAGE_PATH FOR RId(%d):\n", path->report_id);
176 usb_log_debug("\tLENGTH: %d\n", path->depth);
177
[feeac0d]178 list_foreach(path->items, rpath_items_link,
179 usb_hid_report_usage_path_t, path_item) {
[5499a8b]180
[9d05599]181 usb_log_debug("\tUSAGE_PAGE: %X\n", path_item->usage_page);
182 usb_log_debug("\tUSAGE: %X\n", path_item->usage);
[b72efe8]183 usb_log_debug("\tFLAGS: %d\n", path_item->flags);
[9d05599]184 }
185}
186
[a76b01b4]187
[9d05599]188/**
189 * Compares two usage paths structures
190 *
191 *
[5f7b75a]192 * @param report_path usage path structure to compare with @path
[9d05599]193 * @param path usage patrh structure to compare
194 * @param flags Flags determining the mode of comparison
195 * @return EOK if both paths are identical, non zero number otherwise
196 */
197int usb_hid_report_compare_usage_path(usb_hid_report_path_t *report_path,
198 usb_hid_report_path_t *path,
199 int flags)
200{
201 usb_hid_report_usage_path_t *report_item;
202 usb_hid_report_usage_path_t *path_item;
203
204 link_t *report_link;
205 link_t *path_link;
206
207 int only_page;
208
209 if(report_path->report_id != path->report_id) {
[dcb7d7cd]210 if(path->report_id != 0) {
211 return 1;
212 }
[9d05599]213 }
214
[5f7b75a]215 // Empty path match all others
[9d05599]216 if(path->depth == 0){
217 return EOK;
218 }
219
220
221 if((only_page = flags & USB_HID_PATH_COMPARE_USAGE_PAGE_ONLY) != 0){
222 flags -= USB_HID_PATH_COMPARE_USAGE_PAGE_ONLY;
223 }
224
225 switch(flags){
[f3b39b4]226 /* path is somewhere in report_path */
227 case USB_HID_PATH_COMPARE_ANYWHERE:
228 if(path->depth != 1){
229 return 1;
230 }
[5f7b75a]231
[b72efe8]232 path_link = list_first(&path->items);
233 path_item = list_get_instance(path_link,
234 usb_hid_report_usage_path_t, rpath_items_link);
[5f7b75a]235
[feeac0d]236 list_foreach(report_path->items, rpath_items_link,
237 usb_hid_report_usage_path_t, report_item) {
238
[07525cd]239 if (USB_HID_SAME_USAGE_PAGE(report_item->usage_page,
240 path_item->usage_page)) {
241
242 if (only_page == 0) {
243 if (USB_HID_SAME_USAGE(report_item->usage,
244 path_item->usage))
[5f7b75a]245 return EOK;
246 }
[f3b39b4]247 else {
248 return EOK;
249 }
[5f7b75a]250 }
[f3b39b4]251 }
252
253 return 1;
254 break;
255
256 /* the paths must be identical */
257 case USB_HID_PATH_COMPARE_STRICT:
258 if(report_path->depth != path->depth){
[5f7b75a]259 return 1;
[f3b39b4]260 }
[feeac0d]261
[f3b39b4]262 /* path is prefix of the report_path */
263 case USB_HID_PATH_COMPARE_BEGIN:
[feeac0d]264
[b72efe8]265 report_link = report_path->items.head.next;
266 path_link = path->items.head.next;
[07525cd]267
268 while ((report_link != &report_path->items.head) &&
269 (path_link != &path->items.head)) {
270
[f3b39b4]271 report_item = list_get_instance(report_link,
[07525cd]272 usb_hid_report_usage_path_t, rpath_items_link);
273
[f3b39b4]274 path_item = list_get_instance(path_link,
[b72efe8]275 usb_hid_report_usage_path_t, rpath_items_link);
[f3b39b4]276
[07525cd]277 if (!USB_HID_SAME_USAGE_PAGE(report_item->usage_page,
278 path_item->usage_page) || ((only_page == 0) &&
[f3b39b4]279 !USB_HID_SAME_USAGE(report_item->usage,
[07525cd]280 path_item->usage))) {
[f3b39b4]281 return 1;
[feeac0d]282 } else {
[f3b39b4]283 report_link = report_link->next;
[feeac0d]284 path_link = path_link->next;
[f3b39b4]285 }
[b72efe8]286 }
[9d05599]287
[07525cd]288 if ((((flags & USB_HID_PATH_COMPARE_BEGIN) != 0) &&
289 (path_link == &path->items.head)) ||
290 ((report_link == &report_path->items.head) &&
291 (path_link == &path->items.head))) {
[f3b39b4]292 return EOK;
[07525cd]293 } else {
[f3b39b4]294 return 1;
[07525cd]295 }
[f3b39b4]296 break;
[9d05599]297
[f3b39b4]298 /* path is suffix of report_path */
299 case USB_HID_PATH_COMPARE_END:
[9d05599]300
[b72efe8]301 report_link = report_path->items.head.prev;
302 path_link = path->items.head.prev;
[9d05599]303
[b72efe8]304 if(list_empty(&path->items)){
[f3b39b4]305 return EOK;
306 }
[9d05599]307
[b72efe8]308 while((report_link != &report_path->items.head) &&
309 (path_link != &path->items.head)) {
[9d05599]310
[b72efe8]311 report_item = list_get_instance(report_link,
312 usb_hid_report_usage_path_t, rpath_items_link);
[5499a8b]313
[f3b39b4]314 path_item = list_get_instance(path_link,
[b72efe8]315 usb_hid_report_usage_path_t, rpath_items_link);
[3a6e423]316
[f3b39b4]317 if(!USB_HID_SAME_USAGE_PAGE(report_item->usage_page,
318 path_item->usage_page) || ((only_page == 0) &&
319 !USB_HID_SAME_USAGE(report_item->usage,
320 path_item->usage))) {
[5499a8b]321
[9d05599]322 return 1;
[f3b39b4]323 } else {
324 report_link = report_link->prev;
325 path_link = path_link->prev;
326 }
327
328 }
329
[b72efe8]330 if(path_link == &path->items.head) {
[f3b39b4]331 return EOK;
332 }
333 else {
334 return 1;
335 }
[9d05599]336
[f3b39b4]337 break;
[9d05599]338
[f3b39b4]339 default:
340 return EINVAL;
[9d05599]341 }
342}
343
[a76b01b4]344
[9d05599]345/**
346 * Allocates and initializes new usage path structure.
347 *
348 * @return Initialized usage path structure
349 */
350usb_hid_report_path_t *usb_hid_report_path(void)
351{
352 usb_hid_report_path_t *path;
353 path = malloc(sizeof(usb_hid_report_path_t));
354 if(path == NULL){
355 return NULL;
356 }
357 else {
358 path->depth = 0;
359 path->report_id = 0;
[b72efe8]360 link_initialize(&path->cpath_link);
361 list_initialize(&path->items);
[9d05599]362 return path;
363 }
364}
365
[a76b01b4]366
[9d05599]367/**
368 * Releases given usage path structure.
369 *
370 * @param path usage path structure to release
371 * @return void
372 */
373void usb_hid_report_path_free(usb_hid_report_path_t *path)
374{
[0cda600]375 if (path == NULL)
376 return;
[b72efe8]377 while(!list_empty(&path->items)){
[9d05599]378 usb_hid_report_remove_last_item(path);
379 }
380
[b72efe8]381 assert_link_not_used(&path->cpath_link);
[9d05599]382 free(path);
383}
384
[a76b01b4]385
[9d05599]386/**
387 * Clone content of given usage path to the new one
388 *
389 * @param usage_path Usage path structure to clone
390 * @return New copy of given usage path structure
391 */
[5499a8b]392usb_hid_report_path_t *usb_hid_report_path_clone(
393 usb_hid_report_path_t *usage_path)
[9d05599]394{
395 usb_hid_report_usage_path_t *new_path_item;
396 usb_hid_report_path_t *new_usage_path = usb_hid_report_path ();
397
398 if(new_usage_path == NULL){
399 return NULL;
400 }
401
402 new_usage_path->report_id = usage_path->report_id;
403
[b72efe8]404 if(list_empty(&usage_path->items)){
[9d05599]405 return new_usage_path;
406 }
407
[feeac0d]408 list_foreach(usage_path->items, rpath_items_link,
409 usb_hid_report_usage_path_t, path_item) {
[f3b39b4]410
[9d05599]411 new_path_item = malloc(sizeof(usb_hid_report_usage_path_t));
412 if(new_path_item == NULL) {
413 return NULL;
414 }
415
[b72efe8]416 link_initialize(&new_path_item->rpath_items_link);
[9d05599]417 new_path_item->usage_page = path_item->usage_page;
418 new_path_item->usage = path_item->usage;
419 new_path_item->flags = path_item->flags;
420
[b72efe8]421 list_append(&new_path_item->rpath_items_link,
422 &new_usage_path->items);
[9d05599]423 new_usage_path->depth++;
424 }
425
426 return new_usage_path;
427}
428
[a76b01b4]429
[9d05599]430/**
431 * Sets report id in usage path structure
432 *
433 * @param path Usage path structure
434 * @param report_id Report id to set
435 * @return Error code
436 */
[5499a8b]437int usb_hid_report_path_set_report_id(usb_hid_report_path_t *path,
438 uint8_t report_id)
[9d05599]439{
440 if(path == NULL){
441 return EINVAL;
442 }
443
444 path->report_id = report_id;
445 return EOK;
446}
447
448/**
449 * @}
450 */
Note: See TracBrowser for help on using the repository browser.