Path: blob/main/contrib/libfido2/fuzz/fuzz_assert.c
39586 views
/*1* Copyright (c) 2019-2023 Yubico AB. All rights reserved.2* Use of this source code is governed by a BSD-style3* license that can be found in the LICENSE file.4* SPDX-License-Identifier: BSD-2-Clause5*/67#include <assert.h>8#include <stdint.h>9#include <stdio.h>10#include <stdlib.h>11#include <string.h>1213#include "mutator_aux.h"14#include "wiredata_fido2.h"15#include "wiredata_u2f.h"16#include "dummy.h"1718#include "../openbsd-compat/openbsd-compat.h"1920/* Parameter set defining a FIDO2 get assertion operation. */21struct param {22char pin[MAXSTR];23char rp_id[MAXSTR];24int ext;25int seed;26struct blob cdh;27struct blob cred;28struct blob es256;29struct blob rs256;30struct blob eddsa;31struct blob wire_data;32uint8_t cred_count;33uint8_t type;34uint8_t opt;35uint8_t up;36uint8_t uv;37};3839/*40* Collection of HID reports from an authenticator issued with a FIDO241* get assertion using the example parameters above.42*/43static const uint8_t dummy_wire_data_fido[] = {44WIREDATA_CTAP_INIT,45WIREDATA_CTAP_CBOR_INFO,46WIREDATA_CTAP_CBOR_AUTHKEY,47WIREDATA_CTAP_CBOR_PINTOKEN,48WIREDATA_CTAP_CBOR_ASSERT,49};5051/*52* Collection of HID reports from an authenticator issued with a U2F53* authentication using the example parameters above.54*/55static const uint8_t dummy_wire_data_u2f[] = {56WIREDATA_CTAP_INIT,57WIREDATA_CTAP_U2F_6985,58WIREDATA_CTAP_U2F_6985,59WIREDATA_CTAP_U2F_6985,60WIREDATA_CTAP_U2F_6985,61WIREDATA_CTAP_U2F_AUTH,62};6364struct param *65unpack(const uint8_t *ptr, size_t len)66{67cbor_item_t *item = NULL, **v;68struct cbor_load_result cbor;69struct param *p;70int ok = -1;7172if ((p = calloc(1, sizeof(*p))) == NULL ||73(item = cbor_load(ptr, len, &cbor)) == NULL ||74cbor.read != len ||75cbor_isa_array(item) == false ||76cbor_array_is_definite(item) == false ||77cbor_array_size(item) != 15 ||78(v = cbor_array_handle(item)) == NULL)79goto fail;8081if (unpack_byte(v[0], &p->uv) < 0 ||82unpack_byte(v[1], &p->up) < 0 ||83unpack_byte(v[2], &p->opt) < 0 ||84unpack_byte(v[3], &p->type) < 0 ||85unpack_byte(v[4], &p->cred_count) < 0 ||86unpack_int(v[5], &p->ext) < 0 ||87unpack_int(v[6], &p->seed) < 0 ||88unpack_string(v[7], p->rp_id) < 0 ||89unpack_string(v[8], p->pin) < 0 ||90unpack_blob(v[9], &p->wire_data) < 0 ||91unpack_blob(v[10], &p->rs256) < 0 ||92unpack_blob(v[11], &p->es256) < 0 ||93unpack_blob(v[12], &p->eddsa) < 0 ||94unpack_blob(v[13], &p->cred) < 0 ||95unpack_blob(v[14], &p->cdh) < 0)96goto fail;9798ok = 0;99fail:100if (ok < 0) {101free(p);102p = NULL;103}104105if (item)106cbor_decref(&item);107108return p;109}110111size_t112pack(uint8_t *ptr, size_t len, const struct param *p)113{114cbor_item_t *argv[15], *array = NULL;115size_t cbor_alloc_len, cbor_len = 0;116unsigned char *cbor = NULL;117118memset(argv, 0, sizeof(argv));119120if ((array = cbor_new_definite_array(15)) == NULL ||121(argv[0] = pack_byte(p->uv)) == NULL ||122(argv[1] = pack_byte(p->up)) == NULL ||123(argv[2] = pack_byte(p->opt)) == NULL ||124(argv[3] = pack_byte(p->type)) == NULL ||125(argv[4] = pack_byte(p->cred_count)) == NULL ||126(argv[5] = pack_int(p->ext)) == NULL ||127(argv[6] = pack_int(p->seed)) == NULL ||128(argv[7] = pack_string(p->rp_id)) == NULL ||129(argv[8] = pack_string(p->pin)) == NULL ||130(argv[9] = pack_blob(&p->wire_data)) == NULL ||131(argv[10] = pack_blob(&p->rs256)) == NULL ||132(argv[11] = pack_blob(&p->es256)) == NULL ||133(argv[12] = pack_blob(&p->eddsa)) == NULL ||134(argv[13] = pack_blob(&p->cred)) == NULL ||135(argv[14] = pack_blob(&p->cdh)) == NULL)136goto fail;137138for (size_t i = 0; i < 15; i++)139if (cbor_array_push(array, argv[i]) == false)140goto fail;141142if ((cbor_len = cbor_serialize_alloc(array, &cbor,143&cbor_alloc_len)) == 0 || cbor_len > len) {144cbor_len = 0;145goto fail;146}147148memcpy(ptr, cbor, cbor_len);149fail:150for (size_t i = 0; i < 15; i++)151if (argv[i])152cbor_decref(&argv[i]);153154if (array)155cbor_decref(&array);156157free(cbor);158159return cbor_len;160}161162size_t163pack_dummy(uint8_t *ptr, size_t len)164{165struct param dummy;166uint8_t blob[MAXCORPUS];167size_t blob_len;168169memset(&dummy, 0, sizeof(dummy));170171dummy.type = 1; /* rsa */172dummy.ext = FIDO_EXT_HMAC_SECRET;173174strlcpy(dummy.pin, dummy_pin, sizeof(dummy.pin));175strlcpy(dummy.rp_id, dummy_rp_id, sizeof(dummy.rp_id));176177dummy.cred.len = sizeof(dummy_cdh); /* XXX */178dummy.cdh.len = sizeof(dummy_cdh);179dummy.es256.len = sizeof(dummy_es256);180dummy.rs256.len = sizeof(dummy_rs256);181dummy.eddsa.len = sizeof(dummy_eddsa);182dummy.wire_data.len = sizeof(dummy_wire_data_fido);183184memcpy(&dummy.cred.body, &dummy_cdh, dummy.cred.len); /* XXX */185memcpy(&dummy.cdh.body, &dummy_cdh, dummy.cdh.len);186memcpy(&dummy.wire_data.body, &dummy_wire_data_fido,187dummy.wire_data.len);188memcpy(&dummy.es256.body, &dummy_es256, dummy.es256.len);189memcpy(&dummy.rs256.body, &dummy_rs256, dummy.rs256.len);190memcpy(&dummy.eddsa.body, &dummy_eddsa, dummy.eddsa.len);191192assert((blob_len = pack(blob, sizeof(blob), &dummy)) != 0);193194if (blob_len > len) {195memcpy(ptr, blob, len);196return len;197}198199memcpy(ptr, blob, blob_len);200201return blob_len;202}203204static void205get_assert(fido_assert_t *assert, uint8_t opt, const struct blob *cdh,206const char *rp_id, int ext, uint8_t up, uint8_t uv, const char *pin,207uint8_t cred_count, const struct blob *cred)208{209fido_dev_t *dev;210211if ((dev = open_dev(opt & 2)) == NULL)212return;213if (opt & 1)214fido_dev_force_u2f(dev);215if (ext & FIDO_EXT_HMAC_SECRET)216fido_assert_set_extensions(assert, FIDO_EXT_HMAC_SECRET);217if (ext & FIDO_EXT_CRED_BLOB)218fido_assert_set_extensions(assert, FIDO_EXT_CRED_BLOB);219if (ext & FIDO_EXT_LARGEBLOB_KEY)220fido_assert_set_extensions(assert, FIDO_EXT_LARGEBLOB_KEY);221if (up & 1)222fido_assert_set_up(assert, FIDO_OPT_TRUE);223else if (opt & 1)224fido_assert_set_up(assert, FIDO_OPT_FALSE);225if (uv & 1)226fido_assert_set_uv(assert, FIDO_OPT_TRUE);227228for (uint8_t i = 0; i < cred_count; i++)229fido_assert_allow_cred(assert, cred->body, cred->len);230231fido_assert_set_clientdata_hash(assert, cdh->body, cdh->len);232fido_assert_set_rp(assert, rp_id);233/* XXX reuse cred as hmac salt */234fido_assert_set_hmac_salt(assert, cred->body, cred->len);235236/* repeat memory operations to trigger reallocation paths */237fido_assert_set_clientdata_hash(assert, cdh->body, cdh->len);238fido_assert_set_rp(assert, rp_id);239fido_assert_set_hmac_salt(assert, cred->body, cred->len);240241if (strlen(pin) == 0)242pin = NULL;243244fido_dev_get_assert(dev, assert, (opt & 1) ? NULL : pin);245246fido_dev_cancel(dev);247fido_dev_close(dev);248fido_dev_free(&dev);249}250251static void252verify_assert(int type, const unsigned char *cdh_ptr, size_t cdh_len,253const char *rp_id, const unsigned char *authdata_ptr, size_t authdata_len,254const unsigned char *sig_ptr, size_t sig_len, uint8_t up, uint8_t uv,255int ext, void *pk)256{257fido_assert_t *assert = NULL;258int r;259260if ((assert = fido_assert_new()) == NULL)261return;262263fido_assert_set_clientdata_hash(assert, cdh_ptr, cdh_len);264fido_assert_set_rp(assert, rp_id);265fido_assert_set_count(assert, 1);266267if (fido_assert_set_authdata(assert, 0, authdata_ptr,268authdata_len) != FIDO_OK) {269fido_assert_set_authdata_raw(assert, 0, authdata_ptr,270authdata_len);271}272273if (up & 1)274fido_assert_set_up(assert, FIDO_OPT_TRUE);275if (uv & 1)276fido_assert_set_uv(assert, FIDO_OPT_TRUE);277278fido_assert_set_extensions(assert, ext);279fido_assert_set_sig(assert, 0, sig_ptr, sig_len);280281/* repeat memory operations to trigger reallocation paths */282if (fido_assert_set_authdata(assert, 0, authdata_ptr,283authdata_len) != FIDO_OK) {284fido_assert_set_authdata_raw(assert, 0, authdata_ptr,285authdata_len);286}287fido_assert_set_sig(assert, 0, sig_ptr, sig_len);288289r = fido_assert_verify(assert, 0, type, pk);290consume(&r, sizeof(r));291292fido_assert_free(&assert);293}294295/*296* Do a dummy conversion to exercise es256_pk_from_EVP_PKEY().297*/298static void299es256_convert(const es256_pk_t *k)300{301EVP_PKEY *pkey = NULL;302es256_pk_t *pk = NULL;303int r;304305if ((pkey = es256_pk_to_EVP_PKEY(k)) == NULL ||306(pk = es256_pk_new()) == NULL)307goto out;308309r = es256_pk_from_EVP_PKEY(pk, pkey);310consume(&r, sizeof(r));311out:312es256_pk_free(&pk);313EVP_PKEY_free(pkey);314}315316/*317* Do a dummy conversion to exercise es384_pk_from_EVP_PKEY().318*/319static void320es384_convert(const es384_pk_t *k)321{322EVP_PKEY *pkey = NULL;323es384_pk_t *pk = NULL;324int r;325326if ((pkey = es384_pk_to_EVP_PKEY(k)) == NULL ||327(pk = es384_pk_new()) == NULL)328goto out;329330r = es384_pk_from_EVP_PKEY(pk, pkey);331consume(&r, sizeof(r));332out:333es384_pk_free(&pk);334EVP_PKEY_free(pkey);335}336337/*338* Do a dummy conversion to exercise rs256_pk_from_EVP_PKEY().339*/340static void341rs256_convert(const rs256_pk_t *k)342{343EVP_PKEY *pkey = NULL;344rs256_pk_t *pk = NULL;345int r;346347if ((pkey = rs256_pk_to_EVP_PKEY(k)) == NULL ||348(pk = rs256_pk_new()) == NULL)349goto out;350351r = rs256_pk_from_EVP_PKEY(pk, pkey);352consume(&r, sizeof(r));353out:354rs256_pk_free(&pk);355EVP_PKEY_free(pkey);356}357358/*359* Do a dummy conversion to exercise eddsa_pk_from_EVP_PKEY().360*/361static void362eddsa_convert(const eddsa_pk_t *k)363{364EVP_PKEY *pkey = NULL;365eddsa_pk_t *pk = NULL;366int r;367368if ((pkey = eddsa_pk_to_EVP_PKEY(k)) == NULL ||369(pk = eddsa_pk_new()) == NULL)370goto out;371372r = eddsa_pk_from_EVP_PKEY(pk, pkey);373consume(&r, sizeof(r));374out:375if (pk)376eddsa_pk_free(&pk);377if (pkey)378EVP_PKEY_free(pkey);379}380381void382test(const struct param *p)383{384fido_assert_t *assert = NULL;385es256_pk_t *es256_pk = NULL;386es384_pk_t *es384_pk = NULL;387rs256_pk_t *rs256_pk = NULL;388eddsa_pk_t *eddsa_pk = NULL;389uint8_t flags;390uint32_t sigcount;391int cose_alg = 0;392void *pk;393394prng_init((unsigned int)p->seed);395fuzz_clock_reset();396fido_init(FIDO_DEBUG);397fido_set_log_handler(consume_str);398399switch (p->type & 3) {400case 0:401cose_alg = COSE_ES256;402403if ((es256_pk = es256_pk_new()) == NULL)404return;405406es256_pk_from_ptr(es256_pk, p->es256.body, p->es256.len);407pk = es256_pk;408409es256_convert(pk);410411break;412case 1:413cose_alg = COSE_RS256;414415if ((rs256_pk = rs256_pk_new()) == NULL)416return;417418rs256_pk_from_ptr(rs256_pk, p->rs256.body, p->rs256.len);419pk = rs256_pk;420421rs256_convert(pk);422423break;424case 2:425cose_alg = COSE_ES384;426427if ((es384_pk = es384_pk_new()) == NULL)428return;429430/* XXX reuse p->es256 as es384 */431es384_pk_from_ptr(es384_pk, p->es256.body, p->es256.len);432pk = es384_pk;433434es384_convert(pk);435436break;437default:438cose_alg = COSE_EDDSA;439440if ((eddsa_pk = eddsa_pk_new()) == NULL)441return;442443eddsa_pk_from_ptr(eddsa_pk, p->eddsa.body, p->eddsa.len);444pk = eddsa_pk;445446eddsa_convert(pk);447448break;449}450451if ((assert = fido_assert_new()) == NULL)452goto out;453454set_wire_data(p->wire_data.body, p->wire_data.len);455456get_assert(assert, p->opt, &p->cdh, p->rp_id, p->ext, p->up, p->uv,457p->pin, p->cred_count, &p->cred);458459/* XXX +1 on purpose */460for (size_t i = 0; i <= fido_assert_count(assert); i++) {461verify_assert(cose_alg,462fido_assert_clientdata_hash_ptr(assert),463fido_assert_clientdata_hash_len(assert),464fido_assert_rp_id(assert),465fido_assert_authdata_ptr(assert, i),466fido_assert_authdata_len(assert, i),467fido_assert_sig_ptr(assert, i),468fido_assert_sig_len(assert, i), p->up, p->uv, p->ext, pk);469consume(fido_assert_authdata_raw_ptr(assert, i),470fido_assert_authdata_raw_len(assert, i));471consume(fido_assert_id_ptr(assert, i),472fido_assert_id_len(assert, i));473consume(fido_assert_user_id_ptr(assert, i),474fido_assert_user_id_len(assert, i));475consume(fido_assert_hmac_secret_ptr(assert, i),476fido_assert_hmac_secret_len(assert, i));477consume_str(fido_assert_user_icon(assert, i));478consume_str(fido_assert_user_name(assert, i));479consume_str(fido_assert_user_display_name(assert, i));480consume(fido_assert_blob_ptr(assert, i),481fido_assert_blob_len(assert, i));482consume(fido_assert_largeblob_key_ptr(assert, i),483fido_assert_largeblob_key_len(assert, i));484flags = fido_assert_flags(assert, i);485consume(&flags, sizeof(flags));486sigcount = fido_assert_sigcount(assert, i);487consume(&sigcount, sizeof(sigcount));488}489490out:491es256_pk_free(&es256_pk);492es384_pk_free(&es384_pk);493rs256_pk_free(&rs256_pk);494eddsa_pk_free(&eddsa_pk);495496fido_assert_free(&assert);497}498499void500mutate(struct param *p, unsigned int seed, unsigned int flags) NO_MSAN501{502if (flags & MUTATE_SEED)503p->seed = (int)seed;504505if (flags & MUTATE_PARAM) {506mutate_byte(&p->uv);507mutate_byte(&p->up);508mutate_byte(&p->opt);509mutate_byte(&p->type);510mutate_byte(&p->cred_count);511mutate_int(&p->ext);512mutate_blob(&p->rs256);513mutate_blob(&p->es256);514mutate_blob(&p->eddsa);515mutate_blob(&p->cred);516mutate_blob(&p->cdh);517mutate_string(p->rp_id);518mutate_string(p->pin);519}520521if (flags & MUTATE_WIREDATA) {522if (p->opt & 1) {523p->wire_data.len = sizeof(dummy_wire_data_u2f);524memcpy(&p->wire_data.body, &dummy_wire_data_u2f,525p->wire_data.len);526} else {527p->wire_data.len = sizeof(dummy_wire_data_fido);528memcpy(&p->wire_data.body, &dummy_wire_data_fido,529p->wire_data.len);530}531mutate_blob(&p->wire_data);532}533}534535536