GCC Code Coverage Report
Directory: ./ Exec Total Coverage
File: usbx_host_classes/src/ux_host_class_hid_local_item_parse.c Lines: 55 55 100.0 %
Date: 2024-12-12 17:16:36 Branches: 17 17 100.0 %

Line Branch Exec Source
1
/***************************************************************************
2
 * Copyright (c) 2024 Microsoft Corporation
3
 *
4
 * This program and the accompanying materials are made available under the
5
 * terms of the MIT License which is available at
6
 * https://opensource.org/licenses/MIT.
7
 *
8
 * SPDX-License-Identifier: MIT
9
 **************************************************************************/
10
11
12
/**************************************************************************/
13
/**************************************************************************/
14
/**                                                                       */
15
/** USBX Component                                                        */
16
/**                                                                       */
17
/**   HID Class                                                           */
18
/**                                                                       */
19
/**************************************************************************/
20
/**************************************************************************/
21
22
23
/* Include necessary system files.  */
24
25
#define UX_SOURCE_CODE
26
27
#include "ux_api.h"
28
#include "ux_host_class_hid.h"
29
#include "ux_host_stack.h"
30
31
32
/**************************************************************************/
33
/*                                                                        */
34
/*  FUNCTION                                               RELEASE        */
35
/*                                                                        */
36
/*    _ux_host_class_hid_local_item_parse                 PORTABLE C      */
37
/*                                                           6.1          */
38
/*  AUTHOR                                                                */
39
/*                                                                        */
40
/*    Chaoqiong Xiao, Microsoft Corporation                               */
41
/*                                                                        */
42
/*  DESCRIPTION                                                           */
43
/*                                                                        */
44
/*    This function parses a local item from the report descriptor.       */
45
/*                                                                        */
46
/*  INPUT                                                                 */
47
/*                                                                        */
48
/*    hid                                   Pointer to HID class          */
49
/*    item                                  Pointer to item               */
50
/*    descriptor                            Pointer to descriptor         */
51
/*                                                                        */
52
/*  OUTPUT                                                                */
53
/*                                                                        */
54
/*    Completion Status                                                   */
55
/*                                                                        */
56
/*  CALLS                                                                 */
57
/*                                                                        */
58
/*    _ux_host_class_hid_item_data_get      Get data item                 */
59
/*                                                                        */
60
/*  CALLED BY                                                             */
61
/*                                                                        */
62
/*    HID Class                                                           */
63
/*                                                                        */
64
/*  RELEASE HISTORY                                                       */
65
/*                                                                        */
66
/*    DATE              NAME                      DESCRIPTION             */
67
/*                                                                        */
68
/*  05-19-2020     Chaoqiong Xiao           Initial Version 6.0           */
69
/*  09-30-2020     Chaoqiong Xiao           Modified comment(s),          */
70
/*                                            resulting in version 6.1    */
71
/*                                                                        */
72
/**************************************************************************/
73
3423
UINT  _ux_host_class_hid_local_item_parse(UX_HOST_CLASS_HID *hid, UX_HOST_CLASS_HID_ITEM *item, UCHAR *descriptor)
74
{
75
76
UX_HOST_CLASS_HID_PARSER    *hid_parser;
77
ULONG                       usage;
78
ULONG                       usage_min;
79
ULONG                       usage_max;
80
ULONG                       delimiter_set;
81
82
    /* Get the temporary parser structure pointer.  */
83
3423
    hid_parser =  &hid -> ux_host_class_hid_parser;
84
85
    /* Analyze the tag.  */
86

3423
    switch (item -> ux_host_class_hid_item_report_tag)
87
    {
88
89
1075
    case UX_HOST_CLASS_HID_LOCAL_TAG_USAGE:
90
91
        /* Local usage tag, check if we have an overflow.  */
92
1075
        if (hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_number_usage == UX_HOST_CLASS_HID_USAGES)
93
        {
94
95
            /* Error trap. */
96
3
            _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_USAGE_OVERFLOW);
97
98
            /* If trace is enabled, insert this event into the trace buffer.  */
99
            UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_HOST_CLASS_HID_USAGE_OVERFLOW, hid, 0, 0, UX_TRACE_ERRORS, 0, 0)
100
101
3
            return(UX_HOST_CLASS_HID_USAGE_OVERFLOW);
102
        }
103
104
        /* Obtain the usage from the descriptor.  */
105
1072
        usage =  _ux_host_class_hid_item_data_get(descriptor, item);
106
107
        /* Combine the global usage with the local usage to form a unique usage ID.  */
108
1072
        usage |= (hid_parser -> ux_host_class_hid_parser_global.ux_host_class_hid_global_item_usage_page << 16);
109
110
        /* Add the usage to the local usage table.  */
111
1072
        hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usages[hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_number_usage] =  usage;
112
113
        /* We have one more usage now.  */
114
1072
        hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_number_usage++;
115
116
1072
        break;
117
118
119
1152
    case UX_HOST_CLASS_HID_LOCAL_TAG_USAGE_MINIMUM:
120
121
        /* Usage Minimum tag.  */
122
1152
        hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usage_min =
123
1152
                                                (ULONG) _ux_host_class_hid_item_data_get(descriptor, item);
124
125
1152
        break;
126
127
128
1153
    case UX_HOST_CLASS_HID_LOCAL_TAG_USAGE_MAXIMUM:
129
130
        /* Usage Maximum tag.  */
131
1153
        hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usage_max =
132
1153
                                                (ULONG) _ux_host_class_hid_item_data_get(descriptor, item);
133
134
        /* Check if the maximum value is coherent with the minimum.  */
135
1153
        if (hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usage_max < hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usage_min)
136
        {
137
138
            /* Error trap. */
139
3
            _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_MIN_MAX_ERROR);
140
141
            /* If trace is enabled, insert this event into the trace buffer.  */
142
            UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_HOST_CLASS_HID_MIN_MAX_ERROR, hid, 0, 0, UX_TRACE_ERRORS, 0, 0)
143
144
3
            return(UX_HOST_CLASS_HID_MIN_MAX_ERROR);
145
        }
146
147
        /* Get the boundaries for the usage values which are defined when encountering the USAGE MAX tag.  */
148
1150
        usage_min =  (ULONG)(hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usage_min);
149
1150
        usage_max =  (ULONG)(hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usage_max);
150
151
35694
        while (usage_min <= usage_max)
152
        {
153
154
            /* Check if we can still add this usage.  */
155
34547
            if (hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_number_usage == UX_HOST_CLASS_HID_USAGES)
156
            {
157
158
                /* Error trap. */
159
3
                _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_USAGE_OVERFLOW);
160
161
                /* If trace is enabled, insert this event into the trace buffer.  */
162
                UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_HOST_CLASS_HID_USAGE_OVERFLOW, hid, 0, 0, UX_TRACE_ERRORS, 0, 0)
163
164
3
                return(UX_HOST_CLASS_HID_USAGE_OVERFLOW);
165
166
            }
167
168
            /* Combine the global usage with the local usage to form a unique usage ID.  */
169
34544
            usage =  usage_min | (hid_parser -> ux_host_class_hid_parser_global.ux_host_class_hid_global_item_usage_page << 16);
170
171
            /* Add the usage to the local usage table.  */
172
34544
            hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_usages[hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_number_usage] =  usage;
173
174
            /* We have one more usage now.  */
175
34544
            hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_number_usage++;
176
177
            /* Next usage value.  */
178
34544
            usage_min++;
179
        }
180
181
1147
        break;
182
183
40
    case UX_HOST_CLASS_HID_LOCAL_TAG_DELIMITER:
184
185
        /* Obtain the delimiter set from the descriptor.  */
186
40
        delimiter_set =  _ux_host_class_hid_item_data_get(descriptor, item);
187
188
        /* We should have either an open or a close.  */
189
        switch (delimiter_set)
190
        {
191
192
23
        case UX_HOST_CLASS_HID_DELIMITER_OPEN:
193
194
            /* Recursive delimiter opens are not supported.  */
195
23
            if (hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_delimiter_level == 1)
196
            {
197
198
                /* Error trap. */
199
3
                _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_DELIMITER_ERROR);
200
201
3
                return(UX_HOST_CLASS_HID_DELIMITER_ERROR);
202
            }
203
204
            /* Mark the opening of the delimiter.  */
205
20
            hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_delimiter_level =  1;
206
207
20
            break;
208
209
14
        case UX_HOST_CLASS_HID_DELIMITER_CLOSE:
210
211
            /* Ensure we had an open delimiter before.  */
212
14
            if (hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_delimiter_level == 0)
213
            {
214
215
                /* Error trap. */
216
3
                _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_DELIMITER_ERROR);
217
218
3
                return(UX_HOST_CLASS_HID_DELIMITER_ERROR);
219
            }
220
221
            /* Mark the closing of the delimiter.  */
222
11
            hid_parser -> ux_host_class_hid_parser_local.ux_host_class_hid_local_item_delimiter_level =  0;
223
224
11
            break;
225
226
3
        default:
227
228
            /* Error trap. */
229
3
            _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_DELIMITER_ERROR);
230
231
            /* We got a wrong delimiter set.  */
232
3
            return(UX_HOST_CLASS_HID_DELIMITER_ERROR);
233
        }
234
31
        break;
235
236
3
    default:
237
238
        /* Error trap. */
239
3
        _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_HOST_CLASS_HID_TAG_UNSUPPORTED);
240
241
        /* If trace is enabled, insert this event into the trace buffer.  */
242
        UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_HOST_CLASS_HID_TAG_UNSUPPORTED, hid, 0, 0, UX_TRACE_ERRORS, 0, 0)
243
244
        /* This tag is either unknown or unsupported.  */
245
3
        return(UX_HOST_CLASS_HID_TAG_UNSUPPORTED);
246
    }
247
248
    /* Return status. Always SUCCESS if we get here.*/
249
3402
    return(UX_SUCCESS);
250
}
251