summaryrefslogtreecommitdiff
path: root/lib/libcfg/rdcfg.c
blob: d3fc947e747a3f9e3831f1fd642a0f6120e567fc (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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
/* SPDX-License-Identifier: ISC */
#include "libcfg.h"

#include <string.h>
#include <stdio.h>
#include <ctype.h>

static const cfg_param_t *find_param(rdline_t *rd, const char *name,
				     const cfg_param_t *params, size_t count)
{
	size_t i;

	for (i = 0; i < count; ++i) {
		if (!strcmp(params[i].key, name))
			return params + i;
	}

	fprintf(stderr, "%s: %zu: unknown keyword '%s'\n",
		rd->filename, rd->lineno, name);
	return NULL;
}

static int splitkv(rdline_t *rd, char **k, char **v)
{
	char *key = rd->line, *value = rd->line;

	while (*value != ' ' && *value != '\0') {
		if (!isalpha(*value)) {
			fprintf(stderr,
				"%s: %zu: unexpected '%c' in keyword\n",
				rd->filename, rd->lineno, *value);
			return -1;
		}
		++value;
	}

	if (*value != ' ') {
		fprintf(stderr, "%s: %zu: expected argument after '%s'\n",
			rd->filename, rd->lineno, key);
		return -1;
	}

	*(value++) = '\0';

	*k = key;
	*v = value;
	return 0;
}

int rdcfg(void *cfgobj, rdline_t *rd, const cfg_param_t *params, size_t count,
	  int flags)
{
	const cfg_param_t *p;
	char *key, *value;
	int ret;

	while ((ret = rdline(rd)) == 0) {
		if (splitkv(rd, &key, &value))
			return -1;

		p = find_param(rd, key, params, count);
		if (p == NULL)
			return -1;

		if (p->allow_block && *value == '{') {
			for (++value; *value == ' '; ++value)
				;

			if (*value != '\0') {
				ret = p->handle(cfgobj, value, rd, flags);
				if (ret)
					return -1;
			}

			while ((ret = rdline(rd)) == 0) {
				if (strcmp(rd->line, "}") == 0)
					break;
				if (p->handle(cfgobj, rd->line, rd, flags))
					return -1;
			}

			if (ret < 0)
				return -1;
			if (ret > 0)
				goto fail_bra;
		} else if (p->handle(cfgobj, value, rd, flags)) {
			return -1;
		}
	}

	return ret < 0 ? -1 : 0;
fail_bra:
	fprintf(stderr, "%s: missing '}' before end-of-file\n", rd->filename);
	return -1;
}