Coverage Report

Created: 2026-04-08 06:32

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/libfido2/src/hid.c
Line
Count
Source
1
/*
2
 * Copyright (c) 2018 Yubico AB. All rights reserved.
3
 * Use of this source code is governed by a BSD-style
4
 * license that can be found in the LICENSE file.
5
 * SPDX-License-Identifier: BSD-2-Clause
6
 */
7
8
#include "fido.h"
9
10
static int
11
get_key_len(uint8_t tag, uint8_t *key, size_t *key_len)
12
38.3M
{
13
38.3M
        *key = tag & 0xfc;
14
38.3M
        if ((*key & 0xf0) == 0xf0) {
15
153k
                fido_log_debug("%s: *key=0x%02x", __func__, *key);
16
153k
                return (-1);
17
153k
        }
18
19
38.1M
        *key_len = tag & 0x3;
20
38.1M
        if (*key_len == 3) {
21
330k
                *key_len = 4;
22
330k
        }
23
24
38.1M
        return (0);
25
38.3M
}
26
27
static int
28
get_key_val(const void *body, size_t key_len, uint32_t *val)
29
37.2M
{
30
37.2M
        const uint8_t *ptr = body;
31
32
37.2M
        switch (key_len) {
33
23.7M
        case 0:
34
23.7M
                *val = 0;
35
23.7M
                break;
36
6.76M
        case 1:
37
6.76M
                *val = ptr[0];
38
6.76M
                break;
39
6.74M
        case 2:
40
6.74M
                *val = (uint32_t)((ptr[1] << 8) | ptr[0]);
41
6.74M
                break;
42
76.1k
        default:
43
76.1k
                fido_log_debug("%s: key_len=%zu", __func__, key_len);
44
76.1k
                return (-1);
45
37.2M
        }
46
47
37.2M
        return (0);
48
37.2M
}
49
50
int
51
fido_hid_get_usage(const uint8_t *report_ptr, size_t report_len,
52
    uint32_t *usage_page)
53
2.20M
{
54
2.20M
        const uint8_t   *ptr = report_ptr;
55
2.20M
        size_t           len = report_len;
56
57
39.3M
        while (len > 0) {
58
38.2M
                const uint8_t tag = ptr[0];
59
38.2M
                ptr++;
60
38.2M
                len--;
61
62
38.2M
                uint8_t  key;
63
38.2M
                size_t   key_len;
64
38.2M
                uint32_t key_val;
65
66
38.2M
                if (get_key_len(tag, &key, &key_len) < 0 || key_len > len ||
67
38.2M
                    get_key_val(ptr, key_len, &key_val) < 0) {
68
1.12M
                        return (-1);
69
1.12M
                }
70
71
37.1M
                if (key == 0x4) {
72
592k
                        *usage_page = key_val;
73
592k
                }
74
75
37.1M
                ptr += key_len;
76
37.1M
                len -= key_len;
77
37.1M
        }
78
79
1.07M
        return (0);
80
2.20M
}
81
82
int
83
fido_hid_get_report_len(const uint8_t *report_ptr, size_t report_len,
84
    size_t *report_in_len, size_t *report_out_len)
85
5.16k
{
86
5.16k
        const uint8_t   *ptr = report_ptr;
87
5.16k
        size_t           len = report_len;
88
5.16k
        uint32_t         report_size = 0;
89
90
67.1k
        while (len > 0) {
91
64.7k
                const uint8_t tag = ptr[0];
92
64.7k
                ptr++;
93
64.7k
                len--;
94
95
64.7k
                uint8_t  key;
96
64.7k
                size_t   key_len;
97
64.7k
                uint32_t key_val;
98
99
64.7k
                if (get_key_len(tag, &key, &key_len) < 0 || key_len > len ||
100
64.7k
                    get_key_val(ptr, key_len, &key_val) < 0) {
101
2.76k
                        return (-1);
102
2.76k
                }
103
104
62.0k
                if (key == 0x94) {
105
1.59k
                        report_size = key_val;
106
60.4k
                } else if (key == 0x80) {
107
1.21k
                        *report_in_len = (size_t)report_size;
108
59.2k
                } else if (key == 0x90) {
109
1.23k
                        *report_out_len = (size_t)report_size;
110
1.23k
                }
111
112
62.0k
                ptr += key_len;
113
62.0k
                len -= key_len;
114
62.0k
        }
115
116
2.40k
        return (0);
117
5.16k
}
118
119
fido_dev_info_t *
120
fido_dev_info_new(size_t n)
121
36.7k
{
122
36.7k
        return (calloc(n, sizeof(fido_dev_info_t)));
123
36.7k
}
124
125
static void
126
fido_dev_info_reset(fido_dev_info_t *di)
127
1.00M
{
128
1.00M
        free(di->path);
129
1.00M
        free(di->manufacturer);
130
1.00M
        free(di->product);
131
1.00M
        memset(di, 0, sizeof(*di));
132
1.00M
}
133
134
void
135
fido_dev_info_free(fido_dev_info_t **devlist_p, size_t n)
136
36.7k
{
137
36.7k
        fido_dev_info_t *devlist;
138
139
36.7k
        if (devlist_p == NULL || (devlist = *devlist_p) == NULL)
140
75
                return;
141
142
1.03M
        for (size_t i = 0; i < n; i++)
143
1.00M
                fido_dev_info_reset(&devlist[i]);
144
145
36.6k
        free(devlist);
146
147
36.6k
        *devlist_p = NULL;
148
36.6k
}
149
150
const fido_dev_info_t *
151
fido_dev_info_ptr(const fido_dev_info_t *devlist, size_t i)
152
15.0k
{
153
15.0k
        return (&devlist[i]);
154
15.0k
}
155
156
int
157
fido_dev_info_set(fido_dev_info_t *devlist, size_t i,
158
    const char *path, const char *manufacturer, const char *product,
159
    const fido_dev_io_t *io, const fido_dev_transport_t *transport)
160
9.43k
{
161
9.43k
        char *path_copy = NULL, *manu_copy = NULL, *prod_copy = NULL;
162
9.43k
        int r;
163
164
9.43k
        if (path == NULL || manufacturer == NULL || product == NULL ||
165
9.43k
            io == NULL) {
166
0
                r = FIDO_ERR_INVALID_ARGUMENT;
167
0
                goto out;
168
0
        }
169
170
9.43k
        if ((path_copy = strdup(path)) == NULL ||
171
9.43k
            (manu_copy = strdup(manufacturer)) == NULL ||
172
9.43k
            (prod_copy = strdup(product)) == NULL) {
173
173
                r = FIDO_ERR_INTERNAL;
174
173
                goto out;
175
173
        }
176
177
9.25k
        fido_dev_info_reset(&devlist[i]);
178
9.25k
        devlist[i].path = path_copy;
179
9.25k
        devlist[i].manufacturer = manu_copy;
180
9.25k
        devlist[i].product = prod_copy;
181
9.25k
        devlist[i].io = *io;
182
9.25k
        if (transport)
183
9.25k
                devlist[i].transport = *transport;
184
9.25k
        r = FIDO_OK;
185
9.43k
out:
186
9.43k
        if (r != FIDO_OK) {
187
173
                free(prod_copy);
188
173
                free(manu_copy);
189
173
                free(path_copy);
190
173
        }
191
9.43k
        return (r);
192
9.25k
}
193
194
const char *
195
fido_dev_info_path(const fido_dev_info_t *di)
196
24.4k
{
197
24.4k
        return (di->path);
198
24.4k
}
199
200
int16_t
201
fido_dev_info_vendor(const fido_dev_info_t *di)
202
15.0k
{
203
15.0k
        return (di->vendor_id);
204
15.0k
}
205
206
int16_t
207
fido_dev_info_product(const fido_dev_info_t *di)
208
15.0k
{
209
15.0k
        return (di->product_id);
210
15.0k
}
211
212
const char *
213
fido_dev_info_manufacturer_string(const fido_dev_info_t *di)
214
24.4k
{
215
24.4k
        return (di->manufacturer);
216
24.4k
}
217
218
const char *
219
fido_dev_info_product_string(const fido_dev_info_t *di)
220
24.4k
{
221
24.4k
        return (di->product);
222
24.4k
}