summaryrefslogtreecommitdiff
path: root/contrib/libder/tests/fuzz_write.c
blob: 2ad5b5eb1764eaf03c03ee3b159e3f699891c4e7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/*-
 * Copyright (c) 2024 Kyle Evans <kevans@FreeBSD.org>
 *
 * SPDX-License-Identifier: BSD-2-Clause
 */

#include <sys/param.h>
#include <sys/socket.h>

#include <assert.h>
#include <signal.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

#include <libder.h>

#include "fuzzers.h"

int
LLVMFuzzerTestOneInput(const uint8_t *data, size_t sz)
{
	struct libder_ctx *ctx;
	struct libder_object *obj;
	size_t readsz;
	int ret;
	bool strict;

	if (sz < 2)
		return (-1);

	/*
	 * I worked this in originally by just using the high bit of the first
	 * byte, but then I realized that encoding it that way would make it
	 * impossible to get strict validation of universal and application
	 * tags.  The former is a bit more important than the latter.
	 */
	strict = !!data[0];
	data++;
	sz--;

	ctx = libder_open();
	libder_set_strict(ctx, strict);
	ret = -1;
	readsz = sz;
	obj = libder_read(ctx, data, &readsz);
	if (obj == NULL || readsz != sz)
		goto out;

	if (obj != NULL) {
		uint8_t *buf = NULL;
		size_t bufsz = 0;

		/*
		 * If we successfully read it, then it shouldn't
		 * overflow.  We're letting libder allocate the buffer,
		 * so we shouldn't be able to hit the 'too small' bit.
		 *
		 * I can't imagine what other errors might happen, so
		 * we'll just assert on it.
		 */
		buf = libder_write(ctx, obj, buf, &bufsz);
		if (buf == NULL)
			goto out;

		assert(bufsz != 0);

		free(buf);
	}

	ret = 0;

out:
	libder_obj_free(obj);
	libder_close(ctx);

	return (ret);
}