xref: /openbsd-src/lib/libfido2/src/blob.c (revision c4a807ed13283181a6a84fdf571f4571a1b25eac)
1d75efeb7Sdjm /*
2d75efeb7Sdjm  * Copyright (c) 2018 Yubico AB. All rights reserved.
3d75efeb7Sdjm  * Use of this source code is governed by a BSD-style
4d75efeb7Sdjm  * license that can be found in the LICENSE file.
5d75efeb7Sdjm  */
6d75efeb7Sdjm 
7d75efeb7Sdjm #include "fido.h"
8d75efeb7Sdjm 
9d75efeb7Sdjm fido_blob_t *
fido_blob_new(void)10d75efeb7Sdjm fido_blob_new(void)
11d75efeb7Sdjm {
12*c4a807edSdjm 	return calloc(1, sizeof(fido_blob_t));
13*c4a807edSdjm }
14*c4a807edSdjm 
15*c4a807edSdjm void
fido_blob_reset(fido_blob_t * b)16*c4a807edSdjm fido_blob_reset(fido_blob_t *b)
17*c4a807edSdjm {
18*c4a807edSdjm 	freezero(b->ptr, b->len);
19*c4a807edSdjm 	explicit_bzero(b, sizeof(*b));
20d75efeb7Sdjm }
21d75efeb7Sdjm 
22d75efeb7Sdjm int
fido_blob_set(fido_blob_t * b,const u_char * ptr,size_t len)23*c4a807edSdjm fido_blob_set(fido_blob_t *b, const u_char *ptr, size_t len)
24d75efeb7Sdjm {
25*c4a807edSdjm 	fido_blob_reset(b);
26d75efeb7Sdjm 
27d75efeb7Sdjm 	if (ptr == NULL || len == 0) {
2832a20e26Sdjm 		fido_log_debug("%s: ptr=%p, len=%zu", __func__,
2932a20e26Sdjm 		    (const void *)ptr, len);
30*c4a807edSdjm 		return -1;
31d75efeb7Sdjm 	}
32d75efeb7Sdjm 
33d75efeb7Sdjm 	if ((b->ptr = malloc(len)) == NULL) {
3432a20e26Sdjm 		fido_log_debug("%s: malloc", __func__);
35*c4a807edSdjm 		return -1;
36d75efeb7Sdjm 	}
37d75efeb7Sdjm 
38d75efeb7Sdjm 	memcpy(b->ptr, ptr, len);
39d75efeb7Sdjm 	b->len = len;
40d75efeb7Sdjm 
41*c4a807edSdjm 	return 0;
42*c4a807edSdjm }
43*c4a807edSdjm 
44*c4a807edSdjm int
fido_blob_append(fido_blob_t * b,const u_char * ptr,size_t len)45*c4a807edSdjm fido_blob_append(fido_blob_t *b, const u_char *ptr, size_t len)
46*c4a807edSdjm {
47*c4a807edSdjm 	u_char *tmp;
48*c4a807edSdjm 
49*c4a807edSdjm 	if (ptr == NULL || len == 0) {
50*c4a807edSdjm 		fido_log_debug("%s: ptr=%p, len=%zu", __func__,
51*c4a807edSdjm 		    (const void *)ptr, len);
52*c4a807edSdjm 		return -1;
53*c4a807edSdjm 	}
54*c4a807edSdjm 	if (SIZE_MAX - b->len < len) {
55*c4a807edSdjm 		fido_log_debug("%s: overflow", __func__);
56*c4a807edSdjm 		return -1;
57*c4a807edSdjm 	}
58*c4a807edSdjm 	if ((tmp = realloc(b->ptr, b->len + len)) == NULL) {
59*c4a807edSdjm 		fido_log_debug("%s: realloc", __func__);
60*c4a807edSdjm 		return -1;
61*c4a807edSdjm 	}
62*c4a807edSdjm 	b->ptr = tmp;
63*c4a807edSdjm 	memcpy(&b->ptr[b->len], ptr, len);
64*c4a807edSdjm 	b->len += len;
65*c4a807edSdjm 
66*c4a807edSdjm 	return 0;
67d75efeb7Sdjm }
68d75efeb7Sdjm 
69d75efeb7Sdjm void
fido_blob_free(fido_blob_t ** bp)70d75efeb7Sdjm fido_blob_free(fido_blob_t **bp)
71d75efeb7Sdjm {
72d75efeb7Sdjm 	fido_blob_t *b;
73d75efeb7Sdjm 
74d75efeb7Sdjm 	if (bp == NULL || (b = *bp) == NULL)
75d75efeb7Sdjm 		return;
76d75efeb7Sdjm 
77*c4a807edSdjm 	fido_blob_reset(b);
78d75efeb7Sdjm 	free(b);
79d75efeb7Sdjm 	*bp = NULL;
80d75efeb7Sdjm }
81d75efeb7Sdjm 
82d75efeb7Sdjm void
fido_free_blob_array(fido_blob_array_t * array)8332a20e26Sdjm fido_free_blob_array(fido_blob_array_t *array)
84d75efeb7Sdjm {
85d75efeb7Sdjm 	if (array->ptr == NULL)
86d75efeb7Sdjm 		return;
87d75efeb7Sdjm 
88d75efeb7Sdjm 	for (size_t i = 0; i < array->len; i++) {
89d75efeb7Sdjm 		fido_blob_t *b = &array->ptr[i];
90*c4a807edSdjm 		freezero(b->ptr, b->len);
91d75efeb7Sdjm 		b->ptr = NULL;
92d75efeb7Sdjm 	}
93d75efeb7Sdjm 
94d75efeb7Sdjm 	free(array->ptr);
95d75efeb7Sdjm 	array->ptr = NULL;
96d75efeb7Sdjm 	array->len = 0;
97d75efeb7Sdjm }
98d75efeb7Sdjm 
99d75efeb7Sdjm cbor_item_t *
fido_blob_encode(const fido_blob_t * b)100d75efeb7Sdjm fido_blob_encode(const fido_blob_t *b)
101d75efeb7Sdjm {
102d75efeb7Sdjm 	if (b == NULL || b->ptr == NULL)
103*c4a807edSdjm 		return NULL;
104d75efeb7Sdjm 
105*c4a807edSdjm 	return cbor_build_bytestring(b->ptr, b->len);
106d75efeb7Sdjm }
107d75efeb7Sdjm 
108d75efeb7Sdjm int
fido_blob_decode(const cbor_item_t * item,fido_blob_t * b)109d75efeb7Sdjm fido_blob_decode(const cbor_item_t *item, fido_blob_t *b)
110d75efeb7Sdjm {
111*c4a807edSdjm 	return cbor_bytestring_copy(item, &b->ptr, &b->len);
112d75efeb7Sdjm }
113d75efeb7Sdjm 
114d75efeb7Sdjm int
fido_blob_is_empty(const fido_blob_t * b)115d75efeb7Sdjm fido_blob_is_empty(const fido_blob_t *b)
116d75efeb7Sdjm {
117*c4a807edSdjm 	return b->ptr == NULL || b->len == 0;
118*c4a807edSdjm }
119*c4a807edSdjm 
120*c4a807edSdjm int
fido_blob_serialise(fido_blob_t * b,const cbor_item_t * item)121*c4a807edSdjm fido_blob_serialise(fido_blob_t *b, const cbor_item_t *item)
122*c4a807edSdjm {
123*c4a807edSdjm 	size_t alloc;
124*c4a807edSdjm 
125*c4a807edSdjm 	if (!fido_blob_is_empty(b))
126*c4a807edSdjm 		return -1;
127*c4a807edSdjm 	if ((b->len = cbor_serialize_alloc(item, &b->ptr, &alloc)) == 0) {
128*c4a807edSdjm 		b->ptr = NULL;
129*c4a807edSdjm 		return -1;
130*c4a807edSdjm 	}
131*c4a807edSdjm 
132*c4a807edSdjm 	return 0;
133d75efeb7Sdjm }
134