Line | Count | Source (jump to first uncovered line) |
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 | | */ |
6 | | |
7 | | #include "fido.h" |
8 | | |
9 | | static int |
10 | | get_key_len(uint8_t tag, uint8_t *key, size_t *key_len) |
11 | 26.4k | { |
12 | 26.4k | *key = tag & 0xfc; |
13 | 26.4k | if ((*key & 0xf0) == 0xf0) { |
14 | 22 | fido_log_debug("%s: *key=0x%02x", __func__, *key); |
15 | 22 | return (-1); |
16 | 22 | } |
17 | 26.4k | |
18 | 26.4k | *key_len = tag & 0x3; |
19 | 26.4k | if (*key_len == 3) { |
20 | 48 | *key_len = 4; |
21 | 48 | } |
22 | 26.4k | |
23 | 26.4k | return (0); |
24 | 26.4k | } |
25 | | |
26 | | static int |
27 | | get_key_val(const void *body, size_t key_len, uint32_t *val) |
28 | 26.4k | { |
29 | 26.4k | const uint8_t *ptr = body; |
30 | 26.4k | |
31 | 26.4k | switch (key_len) { |
32 | 17.5k | case 0: |
33 | 17.5k | *val = 0; |
34 | 17.5k | break; |
35 | 6.90k | case 1: |
36 | 6.90k | *val = ptr[0]; |
37 | 6.90k | break; |
38 | 1.94k | case 2: |
39 | 1.94k | *val = (uint32_t)((ptr[1] << 8) | ptr[0]); |
40 | 1.94k | break; |
41 | 42 | default: |
42 | 42 | fido_log_debug("%s: key_len=%zu", __func__, key_len); |
43 | 42 | return (-1); |
44 | 26.3k | } |
45 | 26.3k | |
46 | 26.3k | return (0); |
47 | 26.3k | } |
48 | | |
49 | | int |
50 | | fido_hid_get_usage(const uint8_t *report_ptr, size_t report_len, |
51 | | uint32_t *usage_page) |
52 | 109 | { |
53 | 109 | const uint8_t *ptr = report_ptr; |
54 | 109 | size_t len = report_len; |
55 | 109 | |
56 | 13.2k | while (len > 0) { |
57 | 13.2k | const uint8_t tag = ptr[0]; |
58 | 13.2k | ptr++; |
59 | 13.2k | len--; |
60 | 13.2k | |
61 | 13.2k | uint8_t key; |
62 | 13.2k | size_t key_len; |
63 | 13.2k | uint32_t key_val; |
64 | 13.2k | |
65 | 13.2k | if (get_key_len(tag, &key, &key_len) < 0 || key_len > len || |
66 | 13.2k | get_key_val(ptr, key_len, &key_val) < 0) { |
67 | 40 | return (-1); |
68 | 40 | } |
69 | 13.1k | |
70 | 13.1k | if (key == 0x4) { |
71 | 250 | *usage_page = key_val; |
72 | 250 | } |
73 | 13.1k | |
74 | 13.1k | ptr += key_len; |
75 | 13.1k | len -= key_len; |
76 | 13.1k | } |
77 | 109 | |
78 | 109 | return (0); |
79 | 109 | } |
80 | | |
81 | | int |
82 | | fido_hid_get_report_len(const uint8_t *report_ptr, size_t report_len, |
83 | | size_t *report_in_len, size_t *report_out_len) |
84 | 109 | { |
85 | 109 | const uint8_t *ptr = report_ptr; |
86 | 109 | size_t len = report_len; |
87 | 109 | uint32_t report_size = 0; |
88 | 109 | |
89 | 13.2k | while (len > 0) { |
90 | 13.2k | const uint8_t tag = ptr[0]; |
91 | 13.2k | ptr++; |
92 | 13.2k | len--; |
93 | 13.2k | |
94 | 13.2k | uint8_t key; |
95 | 13.2k | size_t key_len; |
96 | 13.2k | uint32_t key_val; |
97 | 13.2k | |
98 | 13.2k | if (get_key_len(tag, &key, &key_len) < 0 || key_len > len || |
99 | 13.2k | get_key_val(ptr, key_len, &key_val) < 0) { |
100 | 40 | return (-1); |
101 | 40 | } |
102 | 13.1k | |
103 | 13.1k | if (key == 0x94) { |
104 | 335 | report_size = key_val; |
105 | 12.8k | } else if (key == 0x80) { |
106 | 343 | *report_in_len = (size_t)report_size; |
107 | 12.5k | } else if (key == 0x90) { |
108 | 206 | *report_out_len = (size_t)report_size; |
109 | 206 | } |
110 | 13.1k | |
111 | 13.1k | ptr += key_len; |
112 | 13.1k | len -= key_len; |
113 | 13.1k | } |
114 | 109 | |
115 | 109 | return (0); |
116 | 109 | } |
117 | | |
118 | | fido_dev_info_t * |
119 | | fido_dev_info_new(size_t n) |
120 | 0 | { |
121 | 0 | return (calloc(n, sizeof(fido_dev_info_t))); |
122 | 0 | } |
123 | | |
124 | | void |
125 | | fido_dev_info_free(fido_dev_info_t **devlist_p, size_t n) |
126 | 0 | { |
127 | 0 | fido_dev_info_t *devlist; |
128 | 0 |
|
129 | 0 | if (devlist_p == NULL || (devlist = *devlist_p) == NULL) |
130 | 0 | return; |
131 | 0 | |
132 | 0 | for (size_t i = 0; i < n; i++) { |
133 | 0 | const fido_dev_info_t *di = &devlist[i]; |
134 | 0 | free(di->path); |
135 | 0 | free(di->manufacturer); |
136 | 0 | free(di->product); |
137 | 0 | } |
138 | 0 |
|
139 | 0 | free(devlist); |
140 | 0 |
|
141 | 0 | *devlist_p = NULL; |
142 | 0 | } |
143 | | |
144 | | const fido_dev_info_t * |
145 | | fido_dev_info_ptr(const fido_dev_info_t *devlist, size_t i) |
146 | 0 | { |
147 | 0 | return (&devlist[i]); |
148 | 0 | } |
149 | | |
150 | | const char * |
151 | | fido_dev_info_path(const fido_dev_info_t *di) |
152 | 0 | { |
153 | 0 | return (di->path); |
154 | 0 | } |
155 | | |
156 | | int16_t |
157 | | fido_dev_info_vendor(const fido_dev_info_t *di) |
158 | 0 | { |
159 | 0 | return (di->vendor_id); |
160 | 0 | } |
161 | | |
162 | | int16_t |
163 | | fido_dev_info_product(const fido_dev_info_t *di) |
164 | 0 | { |
165 | 0 | return (di->product_id); |
166 | 0 | } |
167 | | |
168 | | const char * |
169 | | fido_dev_info_manufacturer_string(const fido_dev_info_t *di) |
170 | 0 | { |
171 | 0 | return (di->manufacturer); |
172 | 0 | } |
173 | | |
174 | | const char * |
175 | | fido_dev_info_product_string(const fido_dev_info_t *di) |
176 | 0 | { |
177 | 0 | return (di->product); |
178 | 0 | } |