2010-08-17 03:33:07 +02:00
|
|
|
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
|
2009-11-19 00:48:48 +01:00
|
|
|
|
2012-07-18 19:07:51 +02:00
|
|
|
#pragma once
|
2009-11-19 00:48:48 +01:00
|
|
|
|
2010-02-03 13:03:47 +01:00
|
|
|
/***
|
|
|
|
This file is part of systemd.
|
|
|
|
|
|
|
|
Copyright 2010 Lennart Poettering
|
|
|
|
|
|
|
|
systemd is free software; you can redistribute it and/or modify it
|
2012-04-12 00:20:58 +02:00
|
|
|
under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
2010-02-03 13:03:47 +01:00
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
systemd 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
|
2012-04-12 00:20:58 +02:00
|
|
|
Lesser General Public License for more details.
|
2010-02-03 13:03:47 +01:00
|
|
|
|
2012-04-12 00:20:58 +02:00
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
2010-02-03 13:03:47 +01:00
|
|
|
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
***/
|
|
|
|
|
2009-11-19 00:48:48 +01:00
|
|
|
#include <stdio.h>
|
2010-06-16 01:58:50 +02:00
|
|
|
#include <stdbool.h>
|
2009-11-19 00:48:48 +01:00
|
|
|
|
2013-04-16 04:25:58 +02:00
|
|
|
#include "macro.h"
|
|
|
|
|
2009-11-19 00:48:48 +01:00
|
|
|
/* An abstract parser for simple, line based, shallow configuration
|
|
|
|
* files consisting of variable assignments only. */
|
|
|
|
|
2011-08-01 00:43:05 +02:00
|
|
|
/* Prototype for a parser for a specific configuration setting */
|
2013-04-16 04:25:58 +02:00
|
|
|
typedef int (*ConfigParserCallback)(const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
2013-11-19 16:17:55 +01:00
|
|
|
unsigned section_line,
|
2013-04-16 04:25:58 +02:00
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata);
|
2011-08-01 00:43:05 +02:00
|
|
|
|
|
|
|
/* Wraps information for parsing a specific configuration variable, to
|
|
|
|
* be stored in a simple array */
|
|
|
|
typedef struct ConfigTableItem {
|
|
|
|
const char *section; /* Section */
|
|
|
|
const char *lvalue; /* Name of the variable */
|
|
|
|
ConfigParserCallback parse; /* Function that is called to parse the variable's value */
|
|
|
|
int ltype; /* Distinguish different variables passed to the same callback */
|
|
|
|
void *data; /* Where to store the variable's data */
|
|
|
|
} ConfigTableItem;
|
|
|
|
|
|
|
|
/* Wraps information for parsing a specific configuration variable, to
|
|
|
|
* ve srored in a gperf perfect hashtable */
|
|
|
|
typedef struct ConfigPerfItem {
|
|
|
|
const char *section_and_lvalue; /* Section + "." + name of the variable */
|
|
|
|
ConfigParserCallback parse; /* Function that is called to parse the variable's value */
|
|
|
|
int ltype; /* Distinguish different variables passed to the same callback */
|
|
|
|
size_t offset; /* Offset where to store data, from the beginning of userdata */
|
|
|
|
} ConfigPerfItem;
|
|
|
|
|
|
|
|
/* Prototype for a low-level gperf lookup function */
|
|
|
|
typedef const ConfigPerfItem* (*ConfigPerfItemLookup)(const char *section_and_lvalue, unsigned length);
|
|
|
|
|
|
|
|
/* Prototype for a generic high-level lookup function */
|
|
|
|
typedef int (*ConfigItemLookup)(
|
|
|
|
void *table,
|
|
|
|
const char *section,
|
|
|
|
const char *lvalue,
|
|
|
|
ConfigParserCallback *func,
|
|
|
|
int *ltype,
|
|
|
|
void **data,
|
|
|
|
void *userdata);
|
|
|
|
|
|
|
|
/* Linear table search implementation of ConfigItemLookup, based on
|
|
|
|
* ConfigTableItem arrays */
|
|
|
|
int config_item_table_lookup(void *table, const char *section, const char *lvalue, ConfigParserCallback *func, int *ltype, void **data, void *userdata);
|
|
|
|
|
|
|
|
/* gperf implementation of ConfigItemLookup, based on gperf
|
|
|
|
* ConfigPerfItem tables */
|
|
|
|
int config_item_perf_lookup(void *table, const char *section, const char *lvalue, ConfigParserCallback *func, int *ltype, void **data, void *userdata);
|
|
|
|
|
2013-04-16 04:25:58 +02:00
|
|
|
int config_parse(const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
FILE *f,
|
|
|
|
const char *sections, /* nulstr */
|
|
|
|
ConfigItemLookup lookup,
|
|
|
|
void *table,
|
|
|
|
bool relaxed,
|
2013-04-25 00:53:16 +02:00
|
|
|
bool allow_include,
|
2013-04-16 04:25:58 +02:00
|
|
|
void *userdata);
|
2009-11-19 00:48:48 +01:00
|
|
|
|
|
|
|
/* Generic parsers */
|
2013-11-19 16:17:55 +01:00
|
|
|
int config_parse_int(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_unsigned(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_long(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_uint64(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_double(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_bytes_size(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_bytes_off(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_bool(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_string(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_path(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_strv(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_path_strv(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_sec(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_nsec(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_mode(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_facility(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_level(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
|
|
|
int config_parse_set_status(const char *unit, const char *filename, unsigned line, const char *section, unsigned section_line, const char *lvalue, int ltype, const char *rvalue, void *data, void *userdata);
|
2013-04-16 04:25:58 +02:00
|
|
|
|
|
|
|
int log_syntax_internal(const char *unit, int level,
|
|
|
|
const char *file, unsigned line, const char *func,
|
|
|
|
const char *config_file, unsigned config_line,
|
2013-10-16 03:17:09 +02:00
|
|
|
int error, const char *format, ...) _printf_(9, 10);
|
2013-04-16 04:25:58 +02:00
|
|
|
|
|
|
|
#define log_syntax(unit, level, config_file, config_line, error, ...) \
|
|
|
|
log_syntax_internal(unit, level, \
|
|
|
|
__FILE__, __LINE__, __func__, \
|
|
|
|
config_file, config_line, \
|
|
|
|
error, __VA_ARGS__)
|
2009-11-19 00:48:48 +01:00
|
|
|
|
2010-07-07 01:10:27 +02:00
|
|
|
#define DEFINE_CONFIG_PARSE_ENUM(function,name,type,msg) \
|
2013-04-16 04:25:58 +02:00
|
|
|
int function(const char *unit, \
|
|
|
|
const char *filename, \
|
|
|
|
unsigned line, \
|
|
|
|
const char *section, \
|
2013-11-19 16:17:55 +01:00
|
|
|
unsigned section_line, \
|
2013-04-16 04:25:58 +02:00
|
|
|
const char *lvalue, \
|
|
|
|
int ltype, \
|
|
|
|
const char *rvalue, \
|
|
|
|
void *data, \
|
|
|
|
void *userdata) { \
|
2010-07-07 01:10:27 +02:00
|
|
|
\
|
|
|
|
type *i = data, x; \
|
|
|
|
\
|
|
|
|
assert(filename); \
|
|
|
|
assert(lvalue); \
|
|
|
|
assert(rvalue); \
|
|
|
|
assert(data); \
|
|
|
|
\
|
|
|
|
if ((x = name##_from_string(rvalue)) < 0) { \
|
2013-04-16 04:25:58 +02:00
|
|
|
log_syntax(unit, LOG_ERR, filename, line, -x, \
|
|
|
|
msg ", ignoring: %s", rvalue); \
|
2010-08-17 03:30:53 +02:00
|
|
|
return 0; \
|
2010-07-07 01:10:27 +02:00
|
|
|
} \
|
|
|
|
\
|
|
|
|
*i = x; \
|
|
|
|
return 0; \
|
|
|
|
}
|
2013-10-29 13:03:13 +01:00
|
|
|
|
|
|
|
#define DEFINE_CONFIG_PARSE_ENUMV(function,name,type,invalid,msg) \
|
|
|
|
int function(const char *unit, \
|
|
|
|
const char *filename, \
|
|
|
|
unsigned line, \
|
|
|
|
const char *section, \
|
2013-11-19 16:17:55 +01:00
|
|
|
unsigned section_line, \
|
2013-10-29 13:03:13 +01:00
|
|
|
const char *lvalue, \
|
|
|
|
int ltype, \
|
|
|
|
const char *rvalue, \
|
|
|
|
void *data, \
|
|
|
|
void *userdata) { \
|
|
|
|
\
|
|
|
|
type **enums = data, *xs, x, *ys; \
|
|
|
|
char *w, *state; \
|
|
|
|
size_t l, i = 0; \
|
|
|
|
\
|
|
|
|
assert(filename); \
|
|
|
|
assert(lvalue); \
|
|
|
|
assert(rvalue); \
|
|
|
|
assert(data); \
|
|
|
|
\
|
|
|
|
xs = new0(type, 1); \
|
|
|
|
*xs = invalid; \
|
|
|
|
\
|
|
|
|
FOREACH_WORD(w, l, rvalue, state) { \
|
|
|
|
_cleanup_free_ char *en = NULL; \
|
|
|
|
\
|
|
|
|
en = strndup(w, l); \
|
|
|
|
if (!en) \
|
|
|
|
return -ENOMEM; \
|
|
|
|
\
|
|
|
|
if ((x = name##_from_string(en)) < 0) { \
|
|
|
|
log_syntax(unit, LOG_ERR, filename, line, \
|
|
|
|
-x, msg ", ignoring: %s", en); \
|
|
|
|
continue; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
for (ys = xs; x != invalid && *ys != invalid; ys++) { \
|
|
|
|
if (*ys == x) { \
|
|
|
|
log_syntax(unit, LOG_ERR, filename, \
|
|
|
|
line, -x, \
|
|
|
|
"Duplicate entry, ignoring: %s", \
|
|
|
|
en); \
|
|
|
|
x = invalid; \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
if (x == invalid) \
|
|
|
|
continue; \
|
|
|
|
\
|
|
|
|
*(xs + i) = x; \
|
|
|
|
xs = realloc(xs, ++i + 1); \
|
|
|
|
if (!xs) \
|
|
|
|
return -ENOMEM; \
|
|
|
|
*(xs + i) = invalid; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
free(*enums); \
|
|
|
|
*enums = xs; \
|
|
|
|
return 0; \
|
|
|
|
}
|