Commit ead1e19a authored by Masami Hiramatsu's avatar Masami Hiramatsu Committed by Steven Rostedt (VMware)

lib/bootconfig: Fix a bug of breaking existing tree nodes

Fix a bug of breaking existing tree nodes by parsing the second
and subsequent braces. Since the bootconfig parser uses the
node.next field as a flag of current parent node, but this will
break the existing tree if the same key node is specified again
in the bootconfig.

For example, the following bootconfig should be foo.buz and bar.

foo
bar
foo { buz }

However, when parsing the brace "{", it breaks foo->bar link
by marking open-brace node. So the bootconfig unlinks bar
from the bootconfig internal tree.

This introduces a stack outside of the tree and record the
last open-brace on the stack instead of using node.next field.

Link: https://lkml.kernel.org/r/160068148267.1088739.8264704338030168660.stgit@devnote2

Fixes: 76db5a27 ("bootconfig: Add Extra Boot Config support")
Cc: Ingo Molnar <mingo@redhat.com>
Cc: stable@vger.kernel.org
Signed-off-by: default avatarMasami Hiramatsu <mhiramat@kernel.org>
Signed-off-by: default avatarSteven Rostedt (VMware) <rostedt@goodmis.org>
parent a27026e9
...@@ -31,6 +31,8 @@ static size_t xbc_data_size __initdata; ...@@ -31,6 +31,8 @@ static size_t xbc_data_size __initdata;
static struct xbc_node *last_parent __initdata; static struct xbc_node *last_parent __initdata;
static const char *xbc_err_msg __initdata; static const char *xbc_err_msg __initdata;
static int xbc_err_pos __initdata; static int xbc_err_pos __initdata;
static int open_brace[XBC_DEPTH_MAX] __initdata;
static int brace_index __initdata;
static int __init xbc_parse_error(const char *msg, const char *p) static int __init xbc_parse_error(const char *msg, const char *p)
{ {
...@@ -431,27 +433,27 @@ static char *skip_spaces_until_newline(char *p) ...@@ -431,27 +433,27 @@ static char *skip_spaces_until_newline(char *p)
return p; return p;
} }
static int __init __xbc_open_brace(void) static int __init __xbc_open_brace(char *p)
{ {
/* Mark the last key as open brace */ /* Push the last key as open brace */
last_parent->next = XBC_NODE_MAX; open_brace[brace_index++] = xbc_node_index(last_parent);
if (brace_index >= XBC_DEPTH_MAX)
return xbc_parse_error("Exceed max depth of braces", p);
return 0; return 0;
} }
static int __init __xbc_close_brace(char *p) static int __init __xbc_close_brace(char *p)
{ {
struct xbc_node *node; brace_index--;
if (!last_parent || brace_index < 0 ||
if (!last_parent || last_parent->next != XBC_NODE_MAX) (open_brace[brace_index] != xbc_node_index(last_parent)))
return xbc_parse_error("Unexpected closing brace", p); return xbc_parse_error("Unexpected closing brace", p);
node = last_parent; if (brace_index == 0)
node->next = 0; last_parent = NULL;
do { else
node = xbc_node_get_parent(node); last_parent = &xbc_nodes[open_brace[brace_index - 1]];
} while (node && node->next != XBC_NODE_MAX);
last_parent = node;
return 0; return 0;
} }
...@@ -661,7 +663,7 @@ static int __init xbc_open_brace(char **k, char *n) ...@@ -661,7 +663,7 @@ static int __init xbc_open_brace(char **k, char *n)
return ret; return ret;
*k = n; *k = n;
return __xbc_open_brace(); return __xbc_open_brace(n - 1);
} }
static int __init xbc_close_brace(char **k, char *n) static int __init xbc_close_brace(char **k, char *n)
...@@ -681,6 +683,13 @@ static int __init xbc_verify_tree(void) ...@@ -681,6 +683,13 @@ static int __init xbc_verify_tree(void)
int i, depth, len, wlen; int i, depth, len, wlen;
struct xbc_node *n, *m; struct xbc_node *n, *m;
/* Brace closing */
if (brace_index) {
n = &xbc_nodes[open_brace[brace_index]];
return xbc_parse_error("Brace is not closed",
xbc_node_get_data(n));
}
/* Empty tree */ /* Empty tree */
if (xbc_node_num == 0) { if (xbc_node_num == 0) {
xbc_parse_error("Empty config", xbc_data); xbc_parse_error("Empty config", xbc_data);
...@@ -745,6 +754,7 @@ void __init xbc_destroy_all(void) ...@@ -745,6 +754,7 @@ void __init xbc_destroy_all(void)
xbc_node_num = 0; xbc_node_num = 0;
memblock_free(__pa(xbc_nodes), sizeof(struct xbc_node) * XBC_NODE_MAX); memblock_free(__pa(xbc_nodes), sizeof(struct xbc_node) * XBC_NODE_MAX);
xbc_nodes = NULL; xbc_nodes = NULL;
brace_index = 0;
} }
/** /**
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment