/* SPDX-License-Identifier: GPL-3.0-or-later */ /* * Copyright (C) 2018 - David Oberhollenzer * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "libcfg.h" #include #include 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; } 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->buffer, "}") == 0) break; if (p->handle(cfgobj, rd->buffer, 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; }