| /* |
| * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved. |
| * Copyright (C) 2004-2011 Red Hat, Inc. All rights reserved. |
| * |
| * This file is part of LVM2. |
| * |
| * This copyrighted material is made available to anyone wishing to use, |
| * modify, copy, or redistribute it subject to the terms and conditions |
| * of the GNU Lesser General Public License v.2.1. |
| * |
| * You should have received a copy of the GNU Lesser General Public License |
| * along with this program; if not, write to the Free Software Foundation, |
| * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include "dmlib.h" |
| |
| #include <sys/stat.h> |
| #include <sys/mman.h> |
| #include <unistd.h> |
| #include <fcntl.h> |
| #include <ctype.h> |
| #include <stdarg.h> |
| |
| #define SECTION_B_CHAR '{' |
| #define SECTION_E_CHAR '}' |
| |
| enum { |
| TOK_INT, |
| TOK_FLOAT, |
| TOK_STRING, /* Single quotes */ |
| TOK_STRING_ESCAPED, /* Double quotes */ |
| TOK_STRING_BARE, /* No quotes */ |
| TOK_EQ, |
| TOK_SECTION_B, |
| TOK_SECTION_E, |
| TOK_ARRAY_B, |
| TOK_ARRAY_E, |
| TOK_IDENTIFIER, |
| TOK_COMMA, |
| TOK_EOF |
| }; |
| |
| struct parser { |
| const char *fb, *fe; /* file limits */ |
| |
| int t; /* token limits and type */ |
| const char *tb, *te; |
| |
| int line; /* line number we are on */ |
| |
| struct dm_pool *mem; |
| }; |
| |
| struct config_output { |
| struct dm_pool *mem; |
| dm_putline_fn putline; |
| const struct dm_config_node_out_spec *spec; |
| void *baton; |
| }; |
| |
| static void _get_token(struct parser *p, int tok_prev); |
| static void _eat_space(struct parser *p); |
| static struct dm_config_node *_file(struct parser *p); |
| static struct dm_config_node *_section(struct parser *p, struct dm_config_node *parent); |
| static struct dm_config_value *_value(struct parser *p); |
| static struct dm_config_value *_type(struct parser *p); |
| static int _match_aux(struct parser *p, int t); |
| static struct dm_config_value *_create_value(struct dm_pool *mem); |
| static struct dm_config_node *_create_node(struct dm_pool *mem); |
| static char *_dup_tok(struct parser *p); |
| static char *_dup_token(struct dm_pool *mem, const char *b, const char *e); |
| |
| static const int sep = '/'; |
| |
| #define MAX_INDENT 32 |
| |
| #define match(t) do {\ |
| if (!_match_aux(p, (t))) {\ |
| log_error("Parse error at byte %" PRIptrdiff_t " (line %d): unexpected token", \ |
| p->tb - p->fb + 1, p->line); \ |
| return 0;\ |
| } \ |
| } while(0) |
| |
| static int _tok_match(const char *str, const char *b, const char *e) |
| { |
| while (*str && (b != e)) { |
| if (*str++ != *b++) |
| return 0; |
| } |
| |
| return !(*str || (b != e)); |
| } |
| |
| struct dm_config_tree *dm_config_create(void) |
| { |
| struct dm_config_tree *cft; |
| struct dm_pool *mem = dm_pool_create("config", 10 * 1024); |
| |
| if (!mem) { |
| log_error("Failed to allocate config pool."); |
| return 0; |
| } |
| |
| if (!(cft = dm_pool_zalloc(mem, sizeof(*cft)))) { |
| log_error("Failed to allocate config tree."); |
| dm_pool_destroy(mem); |
| return 0; |
| } |
| cft->mem = mem; |
| |
| return cft; |
| } |
| |
| void dm_config_set_custom(struct dm_config_tree *cft, void *custom) |
| { |
| cft->custom = custom; |
| } |
| |
| void *dm_config_get_custom(struct dm_config_tree *cft) |
| { |
| return cft->custom; |
| } |
| |
| void dm_config_destroy(struct dm_config_tree *cft) |
| { |
| dm_pool_destroy(cft->mem); |
| } |
| |
| /* |
| * If there's a cascaded dm_config_tree, remove and return it, otherwise |
| * return NULL. |
| */ |
| struct dm_config_tree *dm_config_remove_cascaded_tree(struct dm_config_tree *cft) |
| { |
| struct dm_config_tree *second_cft; |
| |
| if (!cft) |
| return NULL; |
| |
| second_cft = cft->cascade; |
| cft->cascade = NULL; |
| |
| return second_cft; |
| } |
| |
| /* |
| * When searching, first_cft is checked before second_cft. |
| */ |
| struct dm_config_tree *dm_config_insert_cascaded_tree(struct dm_config_tree *first_cft, struct dm_config_tree *second_cft) |
| { |
| first_cft->cascade = second_cft; |
| |
| return first_cft; |
| } |
| |
| static struct dm_config_node *_config_reverse(struct dm_config_node *head) |
| { |
| if (!head) |
| return NULL; |
| |
| struct dm_config_node *left = head, *middle = NULL, *right = NULL; |
| |
| do { |
| right = middle; |
| middle = left; |
| left = left->sib; |
| middle->sib = right; |
| middle->child = _config_reverse(middle->child); |
| } while (left); |
| |
| return middle; |
| }; |
| |
| int dm_config_parse(struct dm_config_tree *cft, const char *start, const char *end) |
| { |
| /* TODO? if (start == end) return 1; */ |
| |
| struct parser *p; |
| if (!(p = dm_pool_alloc(cft->mem, sizeof(*p)))) |
| return_0; |
| |
| p->mem = cft->mem; |
| p->fb = start; |
| p->fe = end; |
| p->tb = p->te = p->fb; |
| p->line = 1; |
| |
| _get_token(p, TOK_SECTION_E); |
| if (!(cft->root = _file(p))) |
| return_0; |
| |
| cft->root = _config_reverse(cft->root); |
| |
| return 1; |
| } |
| |
| struct dm_config_tree *dm_config_from_string(const char *config_settings) |
| { |
| struct dm_config_tree *cft; |
| |
| if (!(cft = dm_config_create())) |
| return_NULL; |
| |
| if (!dm_config_parse(cft, config_settings, config_settings + strlen(config_settings))) { |
| dm_config_destroy(cft); |
| return_NULL; |
| } |
| |
| return cft; |
| } |
| |
| static int _line_start(struct config_output *out) |
| { |
| if (!dm_pool_begin_object(out->mem, 128)) { |
| log_error("dm_pool_begin_object failed for config line"); |
| return 0; |
| } |
| |
| return 1; |
| } |
| |
| __attribute__ ((format(printf, 2, 3))) |
| static int _line_append(struct config_output *out, const char *fmt, ...) |
| { |
| char buf[4096]; |
| va_list ap; |
| int n; |
| |
| va_start(ap, fmt); |
| n = vsnprintf(&buf[0], sizeof buf - 1, fmt, ap); |
| va_end(ap); |
| |
| if (n < 0 || n > (int) sizeof buf - 1) { |
| log_error("vsnprintf failed for config line"); |
| return 0; |
| } |
| |
| if (!dm_pool_grow_object(out->mem, &buf[0], strlen(buf))) { |
| log_error("dm_pool_grow_object failed for config line"); |
| return 0; |
| } |
| |
| return 1; |
| } |
| |
| #define line_append(args...) do {if (!_line_append(out, args)) {return_0;}} while (0) |
| |
| static int _line_end(const struct dm_config_node *cn, struct config_output *out) |
| { |
| const char *line; |
| |
| if (!dm_pool_grow_object(out->mem, "\0", 1)) { |
| log_error("dm_pool_grow_object failed for config line"); |
| return 0; |
| } |
| |
| line = dm_pool_end_object(out->mem); |
| |
| if (!out->putline && !out->spec) |
| return 0; |
| |
| if (out->putline) |
| out->putline(line, out->baton); |
| |
| if (out->spec && out->spec->line_fn) |
| out->spec->line_fn(cn, line, out->baton); |
| |
| return 1; |
| } |
| |
| static int _write_value(struct config_output *out, const struct dm_config_value *v) |
| { |
| char *buf; |
| const char *s; |
| |
| switch (v->type) { |
| case DM_CFG_STRING: |
| buf = alloca(dm_escaped_len(v->v.str)); |
| s = (v->format_flags & DM_CONFIG_VALUE_FMT_STRING_NO_QUOTES) ? "" : "\""; |
| line_append("%s%s%s", s, dm_escape_double_quotes(buf, v->v.str), s); |
| break; |
| |
| case DM_CFG_FLOAT: |
| line_append("%f", v->v.f); |
| break; |
| |
| case DM_CFG_INT: |
| if (v->format_flags & DM_CONFIG_VALUE_FMT_INT_OCTAL) |
| line_append("0%" PRIo64, v->v.i); |
| else |
| line_append(FMTd64, v->v.i); |
| break; |
| |
| case DM_CFG_EMPTY_ARRAY: |
| s = (v->format_flags & DM_CONFIG_VALUE_FMT_COMMON_EXTRA_SPACES) ? " " : ""; |
| line_append("[%s]", s); |
| break; |
| |
| default: |
| log_error("_write_value: Unknown value type: %d", v->type); |
| |
| } |
| |
| return 1; |
| } |
| |
| static int _write_config(const struct dm_config_node *n, int only_one, |
| struct config_output *out, int level) |
| { |
| const char *extra_space; |
| int format_array; |
| char space[MAX_INDENT + 1]; |
| int l = (level < MAX_INDENT) ? level : MAX_INDENT; |
| int i; |
| char *escaped_key = NULL; |
| |
| if (!n) |
| return 1; |
| |
| for (i = 0; i < l; i++) |
| space[i] = '\t'; |
| space[i] = '\0'; |
| |
| do { |
| extra_space = (n->v && (n->v->format_flags & DM_CONFIG_VALUE_FMT_COMMON_EXTRA_SPACES)) ? " " : ""; |
| format_array = (n->v && (n->v->format_flags & DM_CONFIG_VALUE_FMT_COMMON_ARRAY)); |
| |
| if (out->spec && out->spec->prefix_fn) |
| out->spec->prefix_fn(n, space, out->baton); |
| |
| if (!_line_start(out)) |
| return_0; |
| if (strchr(n->key, '#') || strchr(n->key, '"') || strchr(n->key, '!')) { |
| escaped_key = alloca(dm_escaped_len(n->key) + 2); |
| *escaped_key = '"'; |
| dm_escape_double_quotes(escaped_key + 1, n->key); |
| strcat(escaped_key, "\""); |
| } |
| line_append("%s%s", space, escaped_key ? escaped_key : n->key); |
| escaped_key = NULL; |
| if (!n->v) { |
| /* it's a sub section */ |
| line_append(" {"); |
| if (!_line_end(n, out)) |
| return_0; |
| _write_config(n->child, 0, out, level + 1); |
| if (!_line_start(out)) |
| return_0; |
| line_append("%s}", space); |
| } else { |
| /* it's a value */ |
| const struct dm_config_value *v = n->v; |
| line_append("%s=%s", extra_space, extra_space); |
| if (v->next) { |
| line_append("[%s", extra_space); |
| while (v && v->type != DM_CFG_EMPTY_ARRAY) { |
| if (!_write_value(out, v)) |
| return_0; |
| v = v->next; |
| if (v && v->type != DM_CFG_EMPTY_ARRAY) |
| line_append(",%s", extra_space); |
| } |
| line_append("%s]", extra_space); |
| } else { |
| if (format_array && (v->type != DM_CFG_EMPTY_ARRAY)) |
| line_append("[%s", extra_space); |
| if (!_write_value(out, v)) |
| return_0; |
| if (format_array && (v->type != DM_CFG_EMPTY_ARRAY)) |
| line_append("%s]", extra_space); |
| } |
| } |
| if (!_line_end(n, out)) |
| return_0; |
| |
| if (out->spec && out->spec->suffix_fn) |
| out->spec->suffix_fn(n, space, out->baton); |
| |
| n = n->sib; |
| } while (n && !only_one); |
| /* FIXME: add error checking */ |
| return 1; |
| } |
| |
| static int _write_node(const struct dm_config_node *cn, int only_one, |
| dm_putline_fn putline, |
| const struct dm_config_node_out_spec *out_spec, |
| void *baton) |
| { |
| struct config_output out = { |
| .mem = dm_pool_create("config_output", 1024), |
| .putline = putline, |
| .spec = out_spec, |
| .baton = baton |
| }; |
| |
| if (!out.mem) |
| return_0; |
| |
| if (!_write_config(cn, only_one, &out, 0)) { |
| dm_pool_destroy(out.mem); |
| return_0; |
| } |
| dm_pool_destroy(out.mem); |
| return 1; |
| } |
| |
| int dm_config_write_one_node(const struct dm_config_node *cn, dm_putline_fn putline, void *baton) |
| { |
| return _write_node(cn, 1, putline, NULL, baton); |
| } |
| |
| int dm_config_write_node(const struct dm_config_node *cn, dm_putline_fn putline, void *baton) |
| { |
| return _write_node(cn, 0, putline, NULL, baton); |
| } |
| |
| int dm_config_write_one_node_out(const struct dm_config_node *cn, |
| const struct dm_config_node_out_spec *out_spec, |
| void *baton) |
| { |
| return _write_node(cn, 1, NULL, out_spec, baton); |
| } |
| |
| int dm_config_write_node_out(const struct dm_config_node *cn, |
| const struct dm_config_node_out_spec *out_spec, |
| void *baton) |
| { |
| return _write_node(cn, 0, NULL, out_spec, baton); |
| } |
| |
| /* |
| * parser |
| */ |
| static char *_dup_string_tok(struct parser *p) |
| { |
| char *str; |
| |
| p->tb++, p->te--; /* strip "'s */ |
| |
| if (p->te < p->tb) { |
| log_error("Parse error at byte %" PRIptrdiff_t " (line %d): " |
| "expected a string token.", |
| p->tb - p->fb + 1, p->line); |
| return NULL; |
| } |
| |
| if (!(str = _dup_tok(p))) |
| return_NULL; |
| |
| p->te++; |
| |
| return str; |
| } |
| |
| static struct dm_config_node *_file(struct parser *p) |
| { |
| struct dm_config_node root = { 0 }; |
| root.key = "<root>"; |
| |
| while (p->t != TOK_EOF) |
| if (!_section(p, &root)) |
| return_NULL; |
| return root.child; |
| } |
| |
| static struct dm_config_node *_make_node(struct dm_pool *mem, |
| const char *key_b, const char *key_e, |
| struct dm_config_node *parent) |
| { |
| struct dm_config_node *n; |
| |
| if (!(n = _create_node(mem))) |
| return_NULL; |
| |
| n->key = _dup_token(mem, key_b, key_e); |
| if (parent) { |
| n->parent = parent; |
| n->sib = parent->child; |
| parent->child = n; |
| } |
| return n; |
| } |
| |
| /* when mem is not NULL, we create the path if it doesn't exist yet */ |
| static struct dm_config_node *_find_or_make_node(struct dm_pool *mem, |
| struct dm_config_node *parent, |
| const char *path) |
| { |
| const char *e; |
| struct dm_config_node *cn = parent ? parent->child : NULL; |
| struct dm_config_node *cn_found = NULL; |
| |
| while (cn || mem) { |
| /* trim any leading slashes */ |
| while (*path && (*path == sep)) |
| path++; |
| |
| /* find the end of this segment */ |
| for (e = path; *e && (*e != sep); e++) ; |
| |
| /* hunt for the node */ |
| cn_found = NULL; |
| |
| while (cn) { |
| if (_tok_match(cn->key, path, e)) { |
| /* Inefficient */ |
| if (!cn_found) |
| cn_found = cn; |
| else |
| log_warn("WARNING: Ignoring duplicate" |
| " config node: %s (" |
| "seeking %s)", cn->key, path); |
| } |
| |
| cn = cn->sib; |
| } |
| |
| if (!cn_found && mem) { |
| if (!(cn_found = _make_node(mem, path, e, parent))) |
| return_NULL; |
| } |
| |
| if (cn_found && *e) { |
| parent = cn_found; |
| cn = cn_found->child; |
| } else |
| return cn_found; |
| path = e; |
| } |
| |
| return NULL; |
| } |
| |
| static struct dm_config_node *_section(struct parser *p, struct dm_config_node *parent) |
| { |
| /* IDENTIFIER SECTION_B_CHAR VALUE* SECTION_E_CHAR */ |
| |
| struct dm_config_node *root; |
| struct dm_config_value *value; |
| char *str; |
| |
| if (p->t == TOK_STRING_ESCAPED) { |
| if (!(str = _dup_string_tok(p))) |
| return_NULL; |
| dm_unescape_double_quotes(str); |
| |
| match(TOK_STRING_ESCAPED); |
| } else if (p->t == TOK_STRING) { |
| if (!(str = _dup_string_tok(p))) |
| return_NULL; |
| |
| match(TOK_STRING); |
| } else { |
| if (!(str = _dup_tok(p))) |
| return_NULL; |
| |
| match(TOK_IDENTIFIER); |
| } |
| |
| if (!strlen(str)) { |
| log_error("Parse error at byte %" PRIptrdiff_t " (line %d): empty section identifier", |
| p->tb - p->fb + 1, p->line); |
| return NULL; |
| } |
| |
| root = _find_or_make_node(p->mem, parent, str); |
| |
| if (p->t == TOK_SECTION_B) { |
| match(TOK_SECTION_B); |
| while (p->t != TOK_SECTION_E) { |
| if (!(_section(p, root))) |
| return_NULL; |
| } |
| match(TOK_SECTION_E); |
| } else { |
| match(TOK_EQ); |
| if (!(value = _value(p))) |
| return_NULL; |
| if (root->v) |
| log_warn("WARNING: Ignoring duplicate" |
| " config value: %s", str); |
| root->v = value; |
| } |
| |
| return root; |
| } |
| |
| static struct dm_config_value *_value(struct parser *p) |
| { |
| /* '[' TYPE* ']' | TYPE */ |
| struct dm_config_value *h = NULL, *l, *ll = NULL; |
| if (p->t == TOK_ARRAY_B) { |
| match(TOK_ARRAY_B); |
| while (p->t != TOK_ARRAY_E) { |
| if (!(l = _type(p))) |
| return_NULL; |
| |
| if (!h) |
| h = l; |
| else |
| ll->next = l; |
| ll = l; |
| |
| if (p->t == TOK_COMMA) |
| match(TOK_COMMA); |
| } |
| match(TOK_ARRAY_E); |
| /* |
| * Special case for an empty array. |
| */ |
| if (!h) { |
| if (!(h = _create_value(p->mem))) { |
| log_error("Failed to allocate value"); |
| return NULL; |
| } |
| |
| h->type = DM_CFG_EMPTY_ARRAY; |
| } |
| |
| } else |
| if (!(h = _type(p))) |
| return_NULL; |
| |
| return h; |
| } |
| |
| static struct dm_config_value *_type(struct parser *p) |
| { |
| /* [+-]{0,1}[0-9]+ | [0-9]*\.[0-9]* | ".*" */ |
| struct dm_config_value *v = _create_value(p->mem); |
| char *str; |
| |
| if (!v) { |
| log_error("Failed to allocate type value"); |
| return NULL; |
| } |
| |
| switch (p->t) { |
| case TOK_INT: |
| v->type = DM_CFG_INT; |
| v->v.i = strtoll(p->tb, NULL, 0); /* FIXME: check error */ |
| match(TOK_INT); |
| break; |
| |
| case TOK_FLOAT: |
| v->type = DM_CFG_FLOAT; |
| v->v.f = strtod(p->tb, NULL); /* FIXME: check error */ |
| match(TOK_FLOAT); |
| break; |
| |
| case TOK_STRING: |
| v->type = DM_CFG_STRING; |
| |
| if (!(v->v.str = _dup_string_tok(p))) |
| return_NULL; |
| |
| match(TOK_STRING); |
| break; |
| |
| case TOK_STRING_BARE: |
| v->type = DM_CFG_STRING; |
| |
| if (!(v->v.str = _dup_tok(p))) |
| return_NULL; |
| |
| match(TOK_STRING_BARE); |
| break; |
| |
| case TOK_STRING_ESCAPED: |
| v->type = DM_CFG_STRING; |
| |
| if (!(str = _dup_string_tok(p))) |
| return_NULL; |
| dm_unescape_double_quotes(str); |
| v->v.str = str; |
| match(TOK_STRING_ESCAPED); |
| break; |
| |
| default: |
| log_error("Parse error at byte %" PRIptrdiff_t " (line %d): expected a value", |
| p->tb - p->fb + 1, p->line); |
| return NULL; |
| } |
| return v; |
| } |
| |
| static int _match_aux(struct parser *p, int t) |
| { |
| if (p->t != t) |
| return 0; |
| |
| _get_token(p, t); |
| return 1; |
| } |
| |
| /* |
| * tokeniser |
| */ |
| static void _get_token(struct parser *p, int tok_prev) |
| { |
| int values_allowed = 0; |
| |
| const char *te; |
| |
| p->tb = p->te; |
| _eat_space(p); |
| if (p->tb == p->fe || !*p->tb) { |
| p->t = TOK_EOF; |
| return; |
| } |
| |
| /* Should next token be interpreted as value instead of identifier? */ |
| if (tok_prev == TOK_EQ || tok_prev == TOK_ARRAY_B || |
| tok_prev == TOK_COMMA) |
| values_allowed = 1; |
| |
| p->t = TOK_INT; /* fudge so the fall through for |
| floats works */ |
| |
| te = p->te; |
| switch (*te) { |
| case SECTION_B_CHAR: |
| p->t = TOK_SECTION_B; |
| te++; |
| break; |
| |
| case SECTION_E_CHAR: |
| p->t = TOK_SECTION_E; |
| te++; |
| break; |
| |
| case '[': |
| p->t = TOK_ARRAY_B; |
| te++; |
| break; |
| |
| case ']': |
| p->t = TOK_ARRAY_E; |
| te++; |
| break; |
| |
| case ',': |
| p->t = TOK_COMMA; |
| te++; |
| break; |
| |
| case '=': |
| p->t = TOK_EQ; |
| te++; |
| break; |
| |
| case '"': |
| p->t = TOK_STRING_ESCAPED; |
| te++; |
| while ((te != p->fe) && (*te) && (*te != '"')) { |
| if ((*te == '\\') && (te + 1 != p->fe) && |
| *(te + 1)) |
| te++; |
| te++; |
| } |
| |
| if ((te != p->fe) && (*te)) |
| te++; |
| break; |
| |
| case '\'': |
| p->t = TOK_STRING; |
| te++; |
| while ((te != p->fe) && (*te) && (*te != '\'')) |
| te++; |
| |
| if ((te != p->fe) && (*te)) |
| te++; |
| break; |
| |
| case '.': |
| p->t = TOK_FLOAT; |
| /* Fall through */ |
| case '0': |
| case '1': |
| case '2': |
| case '3': |
| case '4': |
| case '5': |
| case '6': |
| case '7': |
| case '8': |
| case '9': |
| case '+': |
| case '-': |
| if (values_allowed) { |
| while (++te != p->fe) { |
| if (!isdigit((int) *te)) { |
| if (*te == '.') { |
| if (p->t != TOK_FLOAT) { |
| p->t = TOK_FLOAT; |
| continue; |
| } |
| } |
| break; |
| } |
| } |
| break; |
| } |
| /* fall through */ |
| |
| default: |
| p->t = TOK_IDENTIFIER; |
| while ((te != p->fe) && (*te) && !isspace(*te) && |
| (*te != '#') && (*te != '=') && |
| (*te != SECTION_B_CHAR) && |
| (*te != SECTION_E_CHAR)) |
| te++; |
| if (values_allowed) |
| p->t = TOK_STRING_BARE; |
| break; |
| } |
| |
| p->te = te; |
| } |
| |
| static void _eat_space(struct parser *p) |
| { |
| while (p->tb != p->fe) { |
| if (*p->te == '#') |
| while ((p->te != p->fe) && (*p->te != '\n') && (*p->te)) |
| ++p->te; |
| |
| else if (!isspace(*p->te)) |
| break; |
| |
| while ((p->te != p->fe) && isspace(*p->te)) { |
| if (*p->te == '\n') |
| ++p->line; |
| ++p->te; |
| } |
| |
| p->tb = p->te; |
| } |
| } |
| |
| /* |
| * memory management |
| */ |
| static struct dm_config_value *_create_value(struct dm_pool *mem) |
| { |
| return dm_pool_zalloc(mem, sizeof(struct dm_config_value)); |
| } |
| |
| static struct dm_config_node *_create_node(struct dm_pool *mem) |
| { |
| return dm_pool_zalloc(mem, sizeof(struct dm_config_node)); |
| } |
| |
| static char *_dup_token(struct dm_pool *mem, const char *b, const char *e) |
| { |
| size_t len = e - b; |
| char *str = dm_pool_alloc(mem, len + 1); |
| if (!str) { |
| log_error("Failed to duplicate token."); |
| return 0; |
| } |
| memcpy(str, b, len); |
| str[len] = '\0'; |
| return str; |
| } |
| |
| static char *_dup_tok(struct parser *p) |
| { |
| return _dup_token(p->mem, p->tb, p->te); |
| } |
| |
| /* |
| * Utility functions |
| */ |
| |
| /* |
| * node_lookup_fn is either: |
| * _find_config_node to perform a lookup starting from a given config_node |
| * in a config_tree; |
| * or |
| * _find_first_config_node to find the first config_node in a set of |
| * cascaded trees. |
| */ |
| typedef const struct dm_config_node *node_lookup_fn(const void *start, const char *path); |
| |
| static const struct dm_config_node *_find_config_node(const void *start, const char *path) { |
| struct dm_config_node dummy = { .child = (void *) start }; |
| return _find_or_make_node(NULL, &dummy, path); |
| } |
| |
| static const struct dm_config_node *_find_first_config_node(const void *start, const char *path) |
| { |
| const struct dm_config_tree *cft = start; |
| const struct dm_config_node *cn = NULL; |
| |
| while (cft) { |
| if ((cn = _find_config_node(cft->root, path))) |
| return cn; |
| cft = cft->cascade; |
| } |
| |
| return NULL; |
| } |
| |
| static const char *_find_config_str(const void *start, node_lookup_fn find_fn, |
| const char *path, const char *fail, int allow_empty) |
| { |
| const struct dm_config_node *n = find_fn(start, path); |
| |
| /* Empty strings are ignored if allow_empty is set */ |
| if (n && n->v) { |
| if ((n->v->type == DM_CFG_STRING) && |
| (allow_empty || (*n->v->v.str))) { |
| log_very_verbose("Setting %s to %s", path, n->v->v.str); |
| return n->v->v.str; |
| } |
| if ((n->v->type != DM_CFG_STRING) || (!allow_empty && fail)) |
| log_warn("WARNING: Ignoring unsupported value for %s.", path); |
| } |
| |
| if (fail) |
| log_very_verbose("%s not found in config: defaulting to %s", |
| path, fail); |
| return fail; |
| } |
| |
| const char *dm_config_find_str(const struct dm_config_node *cn, |
| const char *path, const char *fail) |
| { |
| return _find_config_str(cn, _find_config_node, path, fail, 0); |
| } |
| |
| const char *dm_config_find_str_allow_empty(const struct dm_config_node *cn, |
| const char *path, const char *fail) |
| { |
| return _find_config_str(cn, _find_config_node, path, fail, 1); |
| } |
| |
| static int64_t _find_config_int64(const void *start, node_lookup_fn find, |
| const char *path, int64_t fail) |
| { |
| const struct dm_config_node *n = find(start, path); |
| |
| if (n && n->v && n->v->type == DM_CFG_INT) { |
| log_very_verbose("Setting %s to %" PRId64, path, n->v->v.i); |
| return n->v->v.i; |
| } |
| |
| log_very_verbose("%s not found in config: defaulting to %" PRId64, |
| path, fail); |
| return fail; |
| } |
| |
| static float _find_config_float(const void *start, node_lookup_fn find, |
| const char *path, float fail) |
| { |
| const struct dm_config_node *n = find(start, path); |
| |
| if (n && n->v && n->v->type == DM_CFG_FLOAT) { |
| log_very_verbose("Setting %s to %f", path, n->v->v.f); |
| return n->v->v.f; |
| } |
| |
| log_very_verbose("%s not found in config: defaulting to %f", |
| path, fail); |
| |
| return fail; |
| |
| } |
| |
| static int _str_in_array(const char *str, const char * const values[]) |
| { |
| int i; |
| |
| for (i = 0; values[i]; i++) |
| if (!strcasecmp(str, values[i])) |
| return 1; |
| |
| return 0; |
| } |
| |
| static int _str_to_bool(const char *str, int fail) |
| { |
| const char * const _true_values[] = { "y", "yes", "on", "true", NULL }; |
| const char * const _false_values[] = { "n", "no", "off", "false", NULL }; |
| |
| if (_str_in_array(str, _true_values)) |
| return 1; |
| |
| if (_str_in_array(str, _false_values)) |
| return 0; |
| |
| return fail; |
| } |
| |
| static int _find_config_bool(const void *start, node_lookup_fn find, |
| const char *path, int fail) |
| { |
| const struct dm_config_node *n = find(start, path); |
| const struct dm_config_value *v; |
| int b; |
| |
| if (n) { |
| v = n->v; |
| |
| switch (v->type) { |
| case DM_CFG_INT: |
| b = v->v.i ? 1 : 0; |
| log_very_verbose("Setting %s to %d", path, b); |
| return b; |
| |
| case DM_CFG_STRING: |
| b = _str_to_bool(v->v.str, fail); |
| log_very_verbose("Setting %s to %d", path, b); |
| return b; |
| default: |
| ; |
| } |
| } |
| |
| log_very_verbose("%s not found in config: defaulting to %d", |
| path, fail); |
| |
| return fail; |
| } |
| |
| /*********************************** |
| * node-based lookup |
| **/ |
| |
| struct dm_config_node *dm_config_find_node(const struct dm_config_node *cn, |
| const char *path) |
| { |
| return (struct dm_config_node *) _find_config_node(cn, path); |
| } |
| |
| int dm_config_find_int(const struct dm_config_node *cn, const char *path, int fail) |
| { |
| /* FIXME Add log_error message on overflow */ |
| return (int) _find_config_int64(cn, _find_config_node, path, (int64_t) fail); |
| } |
| |
| int64_t dm_config_find_int64(const struct dm_config_node *cn, const char *path, int64_t fail) |
| { |
| return _find_config_int64(cn, _find_config_node, path, fail); |
| } |
| |
| float dm_config_find_float(const struct dm_config_node *cn, const char *path, |
| float fail) |
| { |
| return _find_config_float(cn, _find_config_node, path, fail); |
| } |
| |
| int dm_config_find_bool(const struct dm_config_node *cn, const char *path, int fail) |
| { |
| return _find_config_bool(cn, _find_config_node, path, fail); |
| } |
| |
| int dm_config_value_is_bool(const struct dm_config_value *v) { |
| if (!v) |
| return 0; |
| |
| switch(v->type) { |
| case DM_CFG_INT: |
| return 1; |
| case DM_CFG_STRING: |
| return _str_to_bool(v->v.str, -1) != -1; |
| default: |
| return 0; |
| } |
| } |
| |
| /*********************************** |
| * tree-based lookup |
| **/ |
| |
| const struct dm_config_node *dm_config_tree_find_node(const struct dm_config_tree *cft, |
| const char *path) |
| { |
| return _find_first_config_node(cft, path); |
| } |
| |
| const char *dm_config_tree_find_str(const struct dm_config_tree *cft, const char *path, |
| const char *fail) |
| { |
| return _find_config_str(cft, _find_first_config_node, path, fail, 0); |
| } |
| |
| const char *dm_config_tree_find_str_allow_empty(const struct dm_config_tree *cft, const char *path, |
| const char *fail) |
| { |
| return _find_config_str(cft, _find_first_config_node, path, fail, 1); |
| } |
| |
| int dm_config_tree_find_int(const struct dm_config_tree *cft, const char *path, int fail) |
| { |
| /* FIXME Add log_error message on overflow */ |
| return (int) _find_config_int64(cft, _find_first_config_node, path, (int64_t) fail); |
| } |
| |
| int64_t dm_config_tree_find_int64(const struct dm_config_tree *cft, const char *path, int64_t fail) |
| { |
| return _find_config_int64(cft, _find_first_config_node, path, fail); |
| } |
| |
| float dm_config_tree_find_float(const struct dm_config_tree *cft, const char *path, |
| float fail) |
| { |
| return _find_config_float(cft, _find_first_config_node, path, fail); |
| } |
| |
| int dm_config_tree_find_bool(const struct dm_config_tree *cft, const char *path, int fail) |
| { |
| return _find_config_bool(cft, _find_first_config_node, path, fail); |
| } |
| |
| /************************************/ |
| |
| |
| int dm_config_get_uint32(const struct dm_config_node *cn, const char *path, |
| uint32_t *result) |
| { |
| const struct dm_config_node *n; |
| |
| n = _find_config_node(cn, path); |
| |
| if (!n || !n->v || n->v->type != DM_CFG_INT) |
| return 0; |
| |
| if (result) |
| *result = n->v->v.i; |
| return 1; |
| } |
| |
| int dm_config_get_uint64(const struct dm_config_node *cn, const char *path, |
| uint64_t *result) |
| { |
| const struct dm_config_node *n; |
| |
| n = _find_config_node(cn, path); |
| |
| if (!n || !n->v || n->v->type != DM_CFG_INT) |
| return 0; |
| |
| if (result) |
| *result = (uint64_t) n->v->v.i; |
| return 1; |
| } |
| |
| int dm_config_get_str(const struct dm_config_node *cn, const char *path, |
| const char **result) |
| { |
| const struct dm_config_node *n; |
| |
| n = _find_config_node(cn, path); |
| |
| if (!n || !n->v || n->v->type != DM_CFG_STRING) |
| return 0; |
| |
| if (result) |
| *result = n->v->v.str; |
| return 1; |
| } |
| |
| int dm_config_get_list(const struct dm_config_node *cn, const char *path, |
| const struct dm_config_value **result) |
| { |
| const struct dm_config_node *n; |
| |
| n = _find_config_node(cn, path); |
| /* TODO when we represent single-item lists consistently, add a check |
| * for n->v->next != NULL */ |
| if (!n || !n->v) |
| return 0; |
| |
| if (result) |
| *result = n->v; |
| return 1; |
| } |
| |
| int dm_config_get_section(const struct dm_config_node *cn, const char *path, |
| const struct dm_config_node **result) |
| { |
| const struct dm_config_node *n; |
| |
| n = _find_config_node(cn, path); |
| if (!n || n->v) |
| return 0; |
| |
| if (result) |
| *result = n; |
| return 1; |
| } |
| |
| int dm_config_has_node(const struct dm_config_node *cn, const char *path) |
| { |
| return _find_config_node(cn, path) ? 1 : 0; |
| } |
| |
| /* |
| * Convert a token type to the char it represents. |
| */ |
| static char _token_type_to_char(int type) |
| { |
| switch (type) { |
| case TOK_SECTION_B: |
| return SECTION_B_CHAR; |
| case TOK_SECTION_E: |
| return SECTION_E_CHAR; |
| default: |
| return 0; |
| } |
| } |
| |
| /* |
| * Returns: |
| * # of 'type' tokens in 'str'. |
| */ |
| static unsigned _count_tokens(const char *str, unsigned len, int type) |
| { |
| char c; |
| |
| c = _token_type_to_char(type); |
| |
| return dm_count_chars(str, len, c); |
| } |
| |
| const char *dm_config_parent_name(const struct dm_config_node *n) |
| { |
| return (n->parent ? n->parent->key : "(root)"); |
| } |
| /* |
| * Heuristic function to make a quick guess as to whether a text |
| * region probably contains a valid config "section". (Useful for |
| * scanning areas of the disk for old metadata.) |
| * Config sections contain various tokens, may contain other sections |
| * and strings, and are delimited by begin (type 'TOK_SECTION_B') and |
| * end (type 'TOK_SECTION_E') tokens. As a quick heuristic, we just |
| * count the number of begin and end tokens, and see if they are |
| * non-zero and the counts match. |
| * Full validation of the section should be done with another function |
| * (for example, read_config_fd). |
| * |
| * Returns: |
| * 0 - probably is not a valid config section |
| * 1 - probably _is_ a valid config section |
| */ |
| unsigned dm_config_maybe_section(const char *str, unsigned len) |
| { |
| int begin_count; |
| int end_count; |
| |
| begin_count = _count_tokens(str, len, TOK_SECTION_B); |
| end_count = _count_tokens(str, len, TOK_SECTION_E); |
| |
| if (begin_count && end_count && (begin_count == end_count)) |
| return 1; |
| else |
| return 0; |
| } |
| |
| __attribute__((nonnull(1, 2))) |
| static struct dm_config_value *_clone_config_value(struct dm_pool *mem, |
| const struct dm_config_value *v) |
| { |
| struct dm_config_value *new_cv; |
| |
| if (!(new_cv = _create_value(mem))) { |
| log_error("Failed to clone config value."); |
| return NULL; |
| } |
| |
| new_cv->type = v->type; |
| if (v->type == DM_CFG_STRING) { |
| if (!(new_cv->v.str = dm_pool_strdup(mem, v->v.str))) { |
| log_error("Failed to clone config string value."); |
| return NULL; |
| } |
| } else |
| new_cv->v = v->v; |
| |
| if (v->next && !(new_cv->next = _clone_config_value(mem, v->next))) |
| return_NULL; |
| |
| return new_cv; |
| } |
| |
| struct dm_config_node *dm_config_clone_node_with_mem(struct dm_pool *mem, const struct dm_config_node *cn, int siblings) |
| { |
| struct dm_config_node *new_cn; |
| |
| if (!cn) { |
| log_error("Cannot clone NULL config node."); |
| return NULL; |
| } |
| |
| if (!(new_cn = _create_node(mem))) { |
| log_error("Failed to clone config node."); |
| return NULL; |
| } |
| |
| if ((cn->key && !(new_cn->key = dm_pool_strdup(mem, cn->key)))) { |
| log_error("Failed to clone config node key."); |
| return NULL; |
| } |
| |
| new_cn->id = cn->id; |
| |
| if ((cn->v && !(new_cn->v = _clone_config_value(mem, cn->v))) || |
| (cn->child && !(new_cn->child = dm_config_clone_node_with_mem(mem, cn->child, 1))) || |
| (siblings && cn->sib && !(new_cn->sib = dm_config_clone_node_with_mem(mem, cn->sib, siblings)))) |
| return_NULL; /* 'new_cn' released with mem pool */ |
| |
| return new_cn; |
| } |
| |
| struct dm_config_node *dm_config_clone_node(struct dm_config_tree *cft, const struct dm_config_node *node, int sib) |
| { |
| return dm_config_clone_node_with_mem(cft->mem, node, sib); |
| } |
| |
| struct dm_config_node *dm_config_create_node(struct dm_config_tree *cft, const char *key) |
| { |
| struct dm_config_node *cn; |
| |
| if (!(cn = _create_node(cft->mem))) { |
| log_error("Failed to create config node."); |
| return NULL; |
| } |
| if (!(cn->key = dm_pool_strdup(cft->mem, key))) { |
| log_error("Failed to create config node's key."); |
| return NULL; |
| } |
| cn->parent = NULL; |
| cn->v = NULL; |
| |
| return cn; |
| } |
| |
| struct dm_config_value *dm_config_create_value(struct dm_config_tree *cft) |
| { |
| return _create_value(cft->mem); |
| } |
| |
| void dm_config_value_set_format_flags(struct dm_config_value *cv, uint32_t format_flags) |
| { |
| if (!cv) |
| return; |
| |
| cv->format_flags = format_flags; |
| } |
| |
| uint32_t dm_config_value_get_format_flags(struct dm_config_value *cv) |
| { |
| if (!cv) |
| return 0; |
| |
| return cv->format_flags; |
| } |
| |
| struct dm_pool *dm_config_memory(struct dm_config_tree *cft) |
| { |
| return cft->mem; |
| } |
| |
| static int _override_path(const char *path, struct dm_config_node *node, void *baton) |
| { |
| struct dm_config_tree *cft = baton; |
| struct dm_config_node dummy, *target; |
| dummy.child = cft->root; |
| if (!(target = _find_or_make_node(cft->mem, &dummy, path))) |
| return_0; |
| if (!(target->v = _clone_config_value(cft->mem, node->v))) |
| return_0; |
| cft->root = dummy.child; |
| return 1; |
| } |
| |
| static int _enumerate(const char *path, struct dm_config_node *cn, int (*cb)(const char *, struct dm_config_node *, void *), void *baton) |
| { |
| char *sub = NULL; |
| |
| while (cn) { |
| if (dm_asprintf(&sub, "%s/%s", path, cn->key) < 0) |
| return_0; |
| if (cn->child) { |
| if (!_enumerate(sub, cn->child, cb, baton)) |
| goto_bad; |
| } else |
| if (!cb(sub, cn, baton)) |
| goto_bad; |
| dm_free(sub); |
| cn = cn->sib; |
| } |
| return 1; |
| bad: |
| dm_free(sub); |
| return 0; |
| } |
| |
| struct dm_config_tree *dm_config_flatten(struct dm_config_tree *cft) |
| { |
| struct dm_config_tree *res = dm_config_create(), *done = NULL, *current = NULL; |
| |
| if (!res) |
| return_NULL; |
| |
| while (done != cft) { |
| current = cft; |
| while (current->cascade != done) |
| current = current->cascade; |
| _enumerate("", current->root, _override_path, res); |
| done = current; |
| } |
| |
| return res; |
| } |
| |
| int dm_config_remove_node(struct dm_config_node *parent, struct dm_config_node *rem_node) |
| { |
| struct dm_config_node *cn = parent->child, *last = NULL; |
| while (cn) { |
| if (cn == rem_node) { |
| if (last) |
| last->sib = cn->sib; |
| else |
| parent->child = cn->sib; |
| return 1; |
| } |
| last = cn; |
| cn = cn->sib; |
| } |
| return 0; |
| } |