Coverage Report

Created: 2024-02-05 19:20

/libfido2/src/dev.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 2018-2022 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
#ifndef TLS
11
#define TLS
12
#endif
13
14
static TLS bool disable_u2f_fallback;
15
16
#ifdef FIDO_FUZZ
17
static void
18
set_random_report_len(fido_dev_t *dev)
19
1.51M
{
20
1.51M
        dev->rx_len = CTAP_MIN_REPORT_LEN +
21
1.51M
            uniform_random(CTAP_MAX_REPORT_LEN - CTAP_MIN_REPORT_LEN + 1);
22
1.51M
        dev->tx_len = CTAP_MIN_REPORT_LEN +
23
1.51M
            uniform_random(CTAP_MAX_REPORT_LEN - CTAP_MIN_REPORT_LEN + 1);
24
1.51M
}
25
#endif
26
27
static void
28
fido_dev_set_extension_flags(fido_dev_t *dev, const fido_cbor_info_t *info)
29
156k
{
30
156k
        char * const    *ptr = fido_cbor_info_extensions_ptr(info);
31
156k
        size_t           len = fido_cbor_info_extensions_len(info);
32
33
505k
        for (size_t i = 0; i < len; i++)
34
348k
                if (strcmp(ptr[i], "credProtect") == 0)
35
78.7k
                        dev->flags |= FIDO_DEV_CRED_PROT;
36
156k
}
37
38
static void
39
fido_dev_set_option_flags(fido_dev_t *dev, const fido_cbor_info_t *info)
40
156k
{
41
156k
        char * const    *ptr = fido_cbor_info_options_name_ptr(info);
42
156k
        const bool      *val = fido_cbor_info_options_value_ptr(info);
43
156k
        size_t           len = fido_cbor_info_options_len(info);
44
45
1.01M
        for (size_t i = 0; i < len; i++)
46
862k
                if (strcmp(ptr[i], "clientPin") == 0) {
47
73.8k
                        dev->flags |= val[i] ?
48
57.6k
                            FIDO_DEV_PIN_SET : FIDO_DEV_PIN_UNSET;
49
788k
                } else if (strcmp(ptr[i], "credMgmt") == 0) {
50
11.4k
                        if (val[i])
51
11.3k
                                dev->flags |= FIDO_DEV_CREDMAN;
52
776k
                } else if (strcmp(ptr[i], "credentialMgmtPreview") == 0) {
53
52.0k
                        if (val[i])
54
51.1k
                                dev->flags |= FIDO_DEV_CREDMAN_PRE;
55
724k
                } else if (strcmp(ptr[i], "uv") == 0) {
56
23.9k
                        dev->flags |= val[i] ?
57
23.7k
                            FIDO_DEV_UV_SET : FIDO_DEV_UV_UNSET;
58
700k
                } else if (strcmp(ptr[i], "pinUvAuthToken") == 0) {
59
15.0k
                        if (val[i])
60
12.8k
                                dev->flags |= FIDO_DEV_TOKEN_PERMS;
61
685k
                } else if (strcmp(ptr[i], "bioEnroll") == 0) {
62
10.9k
                        dev->flags |= val[i] ?
63
10.8k
                            FIDO_DEV_BIO_SET : FIDO_DEV_BIO_UNSET;
64
10.9k
                }
65
156k
}
66
67
static void
68
fido_dev_set_protocol_flags(fido_dev_t *dev, const fido_cbor_info_t *info)
69
156k
{
70
156k
        const uint8_t   *ptr = fido_cbor_info_protocols_ptr(info);
71
156k
        size_t           len = fido_cbor_info_protocols_len(info);
72
73
435k
        for (size_t i = 0; i < len; i++)
74
278k
                switch (ptr[i]) {
75
154k
                case CTAP_PIN_PROTOCOL1:
76
154k
                        dev->flags |= FIDO_DEV_PIN_PROTOCOL1;
77
154k
                        break;
78
58.4k
                case CTAP_PIN_PROTOCOL2:
79
58.4k
                        dev->flags |= FIDO_DEV_PIN_PROTOCOL2;
80
58.4k
                        break;
81
66.0k
                default:
82
66.0k
                        fido_log_debug("%s: unknown protocol %u", __func__,
83
66.0k
                            ptr[i]);
84
66.0k
                        break;
85
278k
                }
86
156k
}
87
88
static void
89
fido_dev_set_flags(fido_dev_t *dev, const fido_cbor_info_t *info)
90
156k
{
91
156k
        fido_dev_set_extension_flags(dev, info);
92
156k
        fido_dev_set_option_flags(dev, info);
93
156k
        fido_dev_set_protocol_flags(dev, info);
94
156k
}
95
96
static int
97
fido_dev_open_tx(fido_dev_t *dev, const char *path, int *ms)
98
6.68M
{
99
6.68M
        int r;
100
101
6.68M
        if (dev->io_handle != NULL) {
102
0
                fido_log_debug("%s: handle=%p", __func__, dev->io_handle);
103
0
                return (FIDO_ERR_INVALID_ARGUMENT);
104
0
        }
105
106
6.68M
        if (dev->io.open == NULL || dev->io.close == NULL) {
107
0
                fido_log_debug("%s: NULL open/close", __func__);
108
0
                return (FIDO_ERR_INVALID_ARGUMENT);
109
0
        }
110
111
6.68M
        if (dev->cid != CTAP_CID_BROADCAST) {
112
0
                fido_log_debug("%s: cid=0x%x", __func__, dev->cid);
113
0
                return (FIDO_ERR_INVALID_ARGUMENT);
114
0
        }
115
116
6.68M
        if (fido_get_random(&dev->nonce, sizeof(dev->nonce)) < 0) {
117
12.6k
                fido_log_debug("%s: fido_get_random", __func__);
118
12.6k
                return (FIDO_ERR_INTERNAL);
119
12.6k
        }
120
121
6.67M
        if ((dev->io_handle = dev->io.open(path)) == NULL) {
122
5.15M
                fido_log_debug("%s: dev->io.open", __func__);
123
5.15M
                return (FIDO_ERR_INTERNAL);
124
5.15M
        }
125
126
1.51M
        if (dev->io_own) {
127
1.51M
                dev->rx_len = CTAP_MAX_REPORT_LEN;
128
1.51M
                dev->tx_len = CTAP_MAX_REPORT_LEN;
129
1.51M
        } else {
130
0
                dev->rx_len = fido_hid_report_in_len(dev->io_handle);
131
0
                dev->tx_len = fido_hid_report_out_len(dev->io_handle);
132
0
        }
133
134
1.51M
#ifdef FIDO_FUZZ
135
1.51M
        set_random_report_len(dev);
136
1.51M
#endif
137
138
1.51M
        if (dev->rx_len < CTAP_MIN_REPORT_LEN ||
139
1.51M
            dev->rx_len > CTAP_MAX_REPORT_LEN) {
140
0
                fido_log_debug("%s: invalid rx_len %zu", __func__, dev->rx_len);
141
0
                r = FIDO_ERR_RX;
142
0
                goto fail;
143
0
        }
144
145
1.51M
        if (dev->tx_len < CTAP_MIN_REPORT_LEN ||
146
1.51M
            dev->tx_len > CTAP_MAX_REPORT_LEN) {
147
0
                fido_log_debug("%s: invalid tx_len %zu", __func__, dev->tx_len);
148
0
                r = FIDO_ERR_TX;
149
0
                goto fail;
150
0
        }
151
152
1.51M
        if (fido_tx(dev, CTAP_CMD_INIT, &dev->nonce, sizeof(dev->nonce),
153
1.51M
            ms) < 0) {
154
24.9k
                fido_log_debug("%s: fido_tx", __func__);
155
24.9k
                r = FIDO_ERR_TX;
156
24.9k
                goto fail;
157
24.9k
        }
158
159
1.49M
        return (FIDO_OK);
160
24.9k
fail:
161
24.9k
        dev->io.close(dev->io_handle);
162
24.9k
        dev->io_handle = NULL;
163
164
24.9k
        return (r);
165
1.51M
}
166
167
static int
168
fido_dev_open_rx(fido_dev_t *dev, int *ms)
169
1.49M
{
170
1.49M
        fido_cbor_info_t        *info = NULL;
171
1.49M
        int                      reply_len;
172
1.49M
        int                      r;
173
174
1.49M
        if ((reply_len = fido_rx(dev, CTAP_CMD_INIT, &dev->attr,
175
1.49M
            sizeof(dev->attr), ms)) < 0) {
176
918k
                fido_log_debug("%s: fido_rx", __func__);
177
918k
                r = FIDO_ERR_RX;
178
918k
                goto fail;
179
918k
        }
180
181
576k
#ifdef FIDO_FUZZ
182
576k
        dev->attr.nonce = dev->nonce;
183
576k
#endif
184
185
576k
        if ((size_t)reply_len != sizeof(dev->attr) ||
186
576k
            dev->attr.nonce != dev->nonce) {
187
34.6k
                fido_log_debug("%s: invalid nonce", __func__);
188
34.6k
                r = FIDO_ERR_RX;
189
34.6k
                goto fail;
190
34.6k
        }
191
192
541k
        dev->flags = 0;
193
541k
        dev->cid = dev->attr.cid;
194
195
541k
        if (fido_dev_is_fido2(dev)) {
196
428k
                if ((info = fido_cbor_info_new()) == NULL) {
197
3.71k
                        fido_log_debug("%s: fido_cbor_info_new", __func__);
198
3.71k
                        r = FIDO_ERR_INTERNAL;
199
3.71k
                        goto fail;
200
3.71k
                }
201
424k
                if ((r = fido_dev_get_cbor_info_wait(dev, info,
202
424k
                    ms)) != FIDO_OK) {
203
268k
                        fido_log_debug("%s: fido_dev_cbor_info_wait: %d",
204
268k
                            __func__, r);
205
268k
                        if (disable_u2f_fallback)
206
0
                                goto fail;
207
268k
                        fido_log_debug("%s: falling back to u2f", __func__);
208
268k
                        fido_dev_force_u2f(dev);
209
268k
                } else {
210
156k
                        fido_dev_set_flags(dev, info);
211
156k
                }
212
424k
        }
213
214
537k
        if (fido_dev_is_fido2(dev) && info != NULL) {
215
156k
                dev->maxmsgsize = fido_cbor_info_maxmsgsiz(info);
216
156k
                fido_log_debug("%s: FIDO_MAXMSG=%d, maxmsgsiz=%lu", __func__,
217
156k
                    FIDO_MAXMSG, (unsigned long)dev->maxmsgsize);
218
156k
        }
219
220
537k
        r = FIDO_OK;
221
1.49M
fail:
222
1.49M
        fido_cbor_info_free(&info);
223
224
1.49M
        if (r != FIDO_OK) {
225
956k
                dev->io.close(dev->io_handle);
226
956k
                dev->io_handle = NULL;
227
956k
        }
228
229
1.49M
        return (r);
230
537k
}
231
232
static int
233
fido_dev_open_wait(fido_dev_t *dev, const char *path, int *ms)
234
6.68M
{
235
6.68M
        int r;
236
237
#ifdef USE_WINHELLO
238
        if (strcmp(path, FIDO_WINHELLO_PATH) == 0)
239
                return (fido_winhello_open(dev));
240
#endif
241
6.68M
        if ((r = fido_dev_open_tx(dev, path, ms)) != FIDO_OK ||
242
6.68M
            (r = fido_dev_open_rx(dev, ms)) != FIDO_OK)
243
6.14M
                return (r);
244
245
537k
        return (FIDO_OK);
246
6.68M
}
247
248
static void
249
run_manifest(fido_dev_info_t *devlist, size_t ilen, size_t *olen,
250
    const char *type, int (*manifest)(fido_dev_info_t *, size_t, size_t *))
251
27.6k
{
252
27.6k
        size_t ndevs = 0;
253
27.6k
        int r;
254
255
27.6k
        if (*olen >= ilen) {
256
586
                fido_log_debug("%s: skipping %s", __func__, type);
257
586
                return;
258
586
        }
259
27.0k
        if ((r = manifest(devlist + *olen, ilen - *olen, &ndevs)) != FIDO_OK)
260
148
                fido_log_debug("%s: %s: 0x%x", __func__, type, r);
261
27.0k
        fido_log_debug("%s: found %zu %s device%s", __func__, ndevs, type,
262
27.0k
            ndevs == 1 ? "" : "s");
263
27.0k
        *olen += ndevs;
264
27.0k
}
265
266
int
267
fido_dev_info_manifest(fido_dev_info_t *devlist, size_t ilen, size_t *olen)
268
9.20k
{
269
9.20k
        *olen = 0;
270
271
9.20k
        run_manifest(devlist, ilen, olen, "hid", fido_hid_manifest);
272
9.20k
#ifdef USE_NFC
273
9.20k
        run_manifest(devlist, ilen, olen, "nfc", fido_nfc_manifest);
274
9.20k
#endif
275
9.20k
#ifdef USE_PCSC
276
9.20k
        run_manifest(devlist, ilen, olen, "pcsc", fido_pcsc_manifest);
277
9.20k
#endif
278
#ifdef USE_WINHELLO
279
        run_manifest(devlist, ilen, olen, "winhello", fido_winhello_manifest);
280
#endif
281
282
9.20k
        return (FIDO_OK);
283
9.20k
}
284
285
int
286
fido_dev_open_with_info(fido_dev_t *dev)
287
0
{
288
0
        int ms = dev->timeout_ms;
289
290
0
        if (dev->path == NULL)
291
0
                return (FIDO_ERR_INVALID_ARGUMENT);
292
293
0
        return (fido_dev_open_wait(dev, dev->path, &ms));
294
0
}
295
296
int
297
fido_dev_open(fido_dev_t *dev, const char *path)
298
6.68M
{
299
6.68M
        int ms = dev->timeout_ms;
300
301
6.68M
#ifdef USE_NFC
302
6.68M
        if (fido_is_nfc(path) && fido_dev_set_nfc(dev) < 0) {
303
0
                fido_log_debug("%s: fido_dev_set_nfc", __func__);
304
0
                return FIDO_ERR_INTERNAL;
305
0
        }
306
6.68M
#endif
307
6.68M
#ifdef USE_PCSC
308
6.68M
        if (fido_is_pcsc(path) && fido_dev_set_pcsc(dev) < 0) {
309
0
                fido_log_debug("%s: fido_dev_set_pcsc", __func__);
310
0
                return FIDO_ERR_INTERNAL;
311
0
        }
312
6.68M
#endif
313
314
6.68M
        return (fido_dev_open_wait(dev, path, &ms));
315
6.68M
}
316
317
int
318
fido_dev_close(fido_dev_t *dev)
319
537k
{
320
#ifdef USE_WINHELLO
321
        if (dev->flags & FIDO_DEV_WINHELLO)
322
                return (fido_winhello_close(dev));
323
#endif
324
537k
        if (dev->io_handle == NULL || dev->io.close == NULL)
325
0
                return (FIDO_ERR_INVALID_ARGUMENT);
326
327
537k
        dev->io.close(dev->io_handle);
328
537k
        dev->io_handle = NULL;
329
537k
        dev->cid = CTAP_CID_BROADCAST;
330
331
537k
        return (FIDO_OK);
332
537k
}
333
334
int
335
fido_dev_set_sigmask(fido_dev_t *dev, const fido_sigset_t *sigmask)
336
0
{
337
0
        if (dev->io_handle == NULL || sigmask == NULL)
338
0
                return (FIDO_ERR_INVALID_ARGUMENT);
339
340
0
#ifdef USE_NFC
341
0
        if (dev->transport.rx == fido_nfc_rx && dev->io.read == fido_nfc_read)
342
0
                return (fido_nfc_set_sigmask(dev->io_handle, sigmask));
343
0
#endif
344
0
        if (dev->transport.rx == NULL && dev->io.read == fido_hid_read)
345
0
                return (fido_hid_set_sigmask(dev->io_handle, sigmask));
346
347
0
        return (FIDO_ERR_INVALID_ARGUMENT);
348
0
}
349
350
int
351
fido_dev_cancel(fido_dev_t *dev)
352
31.5k
{
353
31.5k
        int ms = dev->timeout_ms;
354
355
#ifdef USE_WINHELLO
356
        if (dev->flags & FIDO_DEV_WINHELLO)
357
                return (fido_winhello_cancel(dev));
358
#endif
359
31.5k
        if (fido_dev_is_fido2(dev) == false)
360
20.1k
                return (FIDO_ERR_INVALID_ARGUMENT);
361
11.3k
        if (fido_tx(dev, CTAP_CMD_CANCEL, NULL, 0, &ms) < 0)
362
410
                return (FIDO_ERR_TX);
363
364
10.9k
        return (FIDO_OK);
365
11.3k
}
366
367
int
368
fido_dev_set_io_functions(fido_dev_t *dev, const fido_dev_io_t *io)
369
1.41M
{
370
1.41M
        if (dev->io_handle != NULL) {
371
0
                fido_log_debug("%s: non-NULL handle", __func__);
372
0
                return (FIDO_ERR_INVALID_ARGUMENT);
373
0
        }
374
375
1.41M
        if (io == NULL || io->open == NULL || io->close == NULL ||
376
1.41M
            io->read == NULL || io->write == NULL) {
377
0
                fido_log_debug("%s: NULL function", __func__);
378
0
                return (FIDO_ERR_INVALID_ARGUMENT);
379
0
        }
380
381
1.41M
        dev->io = *io;
382
1.41M
        dev->io_own = true;
383
384
1.41M
        return (FIDO_OK);
385
1.41M
}
386
387
int
388
fido_dev_set_transport_functions(fido_dev_t *dev, const fido_dev_transport_t *t)
389
2.57k
{
390
2.57k
        if (dev->io_handle != NULL) {
391
0
                fido_log_debug("%s: non-NULL handle", __func__);
392
0
                return (FIDO_ERR_INVALID_ARGUMENT);
393
0
        }
394
395
2.57k
        dev->transport = *t;
396
2.57k
        dev->io_own = true;
397
398
2.57k
        return (FIDO_OK);
399
2.57k
}
400
401
void *
402
fido_dev_io_handle(const fido_dev_t *dev)
403
0
{
404
405
0
        return (dev->io_handle);
406
0
}
407
408
void
409
fido_init(int flags)
410
274k
{
411
274k
        if (flags & FIDO_DEBUG || getenv("FIDO_DEBUG") != NULL)
412
274k
                fido_log_init();
413
414
274k
        disable_u2f_fallback = (flags & FIDO_DISABLE_U2F_FALLBACK);
415
274k
}
416
417
fido_dev_t *
418
fido_dev_new(void)
419
6.70M
{
420
6.70M
        fido_dev_t *dev;
421
422
6.70M
        if ((dev = calloc(1, sizeof(*dev))) == NULL)
423
16.0k
                return (NULL);
424
425
6.68M
        dev->cid = CTAP_CID_BROADCAST;
426
6.68M
        dev->timeout_ms = -1;
427
6.68M
        dev->io = (fido_dev_io_t) {
428
6.68M
                &fido_hid_open,
429
6.68M
                &fido_hid_close,
430
6.68M
                &fido_hid_read,
431
6.68M
                &fido_hid_write,
432
6.68M
        };
433
434
6.68M
        return (dev);
435
6.70M
}
436
437
fido_dev_t *
438
fido_dev_new_with_info(const fido_dev_info_t *di)
439
0
{
440
0
        fido_dev_t *dev;
441
442
0
        if ((dev = calloc(1, sizeof(*dev))) == NULL)
443
0
                return (NULL);
444
445
#if 0
446
        if (di->io.open == NULL || di->io.close == NULL ||
447
            di->io.read == NULL || di->io.write == NULL) {
448
                fido_log_debug("%s: NULL function", __func__);
449
                fido_dev_free(&dev);
450
                return (NULL);
451
        }
452
#endif
453
454
0
        dev->io = di->io;
455
0
        dev->io_own = di->transport.tx != NULL || di->transport.rx != NULL;
456
0
        dev->transport = di->transport;
457
0
        dev->cid = CTAP_CID_BROADCAST;
458
0
        dev->timeout_ms = -1;
459
460
0
        if ((dev->path = strdup(di->path)) == NULL) {
461
0
                fido_log_debug("%s: strdup", __func__);
462
0
                fido_dev_free(&dev);
463
0
                return (NULL);
464
0
        }
465
466
0
        return (dev);
467
0
}
468
469
void
470
fido_dev_free(fido_dev_t **dev_p)
471
6.88M
{
472
6.88M
        fido_dev_t *dev;
473
474
6.88M
        if (dev_p == NULL || (dev = *dev_p) == NULL)
475
202k
                return;
476
477
6.68M
        free(dev->path);
478
6.68M
        free(dev);
479
480
6.68M
        *dev_p = NULL;
481
6.68M
}
482
483
uint8_t
484
fido_dev_protocol(const fido_dev_t *dev)
485
17.1k
{
486
17.1k
        return (dev->attr.protocol);
487
17.1k
}
488
489
uint8_t
490
fido_dev_major(const fido_dev_t *dev)
491
17.1k
{
492
17.1k
        return (dev->attr.major);
493
17.1k
}
494
495
uint8_t
496
fido_dev_minor(const fido_dev_t *dev)
497
17.1k
{
498
17.1k
        return (dev->attr.minor);
499
17.1k
}
500
501
uint8_t
502
fido_dev_build(const fido_dev_t *dev)
503
17.1k
{
504
17.1k
        return (dev->attr.build);
505
17.1k
}
506
507
uint8_t
508
fido_dev_flags(const fido_dev_t *dev)
509
17.1k
{
510
17.1k
        return (dev->attr.flags);
511
17.1k
}
512
513
bool
514
fido_dev_is_fido2(const fido_dev_t *dev)
515
1.42M
{
516
1.42M
        return (dev->attr.flags & FIDO_CAP_CBOR);
517
1.42M
}
518
519
bool
520
fido_dev_is_winhello(const fido_dev_t *dev)
521
0
{
522
0
        return (dev->flags & FIDO_DEV_WINHELLO);
523
0
}
524
525
bool
526
fido_dev_supports_pin(const fido_dev_t *dev)
527
85.5k
{
528
85.5k
        return (dev->flags & (FIDO_DEV_PIN_SET|FIDO_DEV_PIN_UNSET));
529
85.5k
}
530
531
bool
532
fido_dev_has_pin(const fido_dev_t *dev)
533
84.4k
{
534
84.4k
        return (dev->flags & FIDO_DEV_PIN_SET);
535
84.4k
}
536
537
bool
538
fido_dev_supports_cred_prot(const fido_dev_t *dev)
539
77.2k
{
540
77.2k
        return (dev->flags & FIDO_DEV_CRED_PROT);
541
77.2k
}
542
543
bool
544
fido_dev_supports_credman(const fido_dev_t *dev)
545
77.2k
{
546
77.2k
        return (dev->flags & (FIDO_DEV_CREDMAN|FIDO_DEV_CREDMAN_PRE));
547
77.2k
}
548
549
bool
550
fido_dev_supports_uv(const fido_dev_t *dev)
551
84.4k
{
552
84.4k
        return (dev->flags & (FIDO_DEV_UV_SET|FIDO_DEV_UV_UNSET));
553
84.4k
}
554
555
bool
556
fido_dev_has_uv(const fido_dev_t *dev)
557
88.9k
{
558
88.9k
        return (dev->flags & FIDO_DEV_UV_SET);
559
88.9k
}
560
561
bool
562
fido_dev_supports_permissions(const fido_dev_t *dev)
563
118k
{
564
118k
        return (dev->flags & FIDO_DEV_TOKEN_PERMS);
565
118k
}
566
567
void
568
fido_dev_force_u2f(fido_dev_t *dev)
569
276k
{
570
276k
        dev->attr.flags &= (uint8_t)~FIDO_CAP_CBOR;
571
276k
        dev->flags = 0;
572
276k
}
573
574
void
575
fido_dev_force_fido2(fido_dev_t *dev)
576
0
{
577
0
        dev->attr.flags |= FIDO_CAP_CBOR;
578
0
}
579
580
uint8_t
581
fido_dev_get_pin_protocol(const fido_dev_t *dev)
582
483k
{
583
483k
        if (dev->flags & FIDO_DEV_PIN_PROTOCOL2)
584
115k
                return (CTAP_PIN_PROTOCOL2);
585
367k
        else if (dev->flags & FIDO_DEV_PIN_PROTOCOL1)
586
273k
                return (CTAP_PIN_PROTOCOL1);
587
588
94.3k
        return (0);
589
483k
}
590
591
uint64_t
592
fido_dev_maxmsgsize(const fido_dev_t *dev)
593
59.1k
{
594
59.1k
        return (dev->maxmsgsize);
595
59.1k
}
596
597
int
598
fido_dev_set_timeout(fido_dev_t *dev, int ms)
599
1.41M
{
600
1.41M
        if (ms < -1)
601
0
                return (FIDO_ERR_INVALID_ARGUMENT);
602
603
1.41M
        dev->timeout_ms = ms;
604
605
1.41M
        return (FIDO_OK);
606
1.41M
}