From 7dae98803a6fc5d08041d1387e2e0d83fc0eb0ed Mon Sep 17 00:00:00 2001 From: Ulf Magnusson Date: Wed, 22 Aug 2018 22:08:06 +0200 Subject: Add a generic node iterator Suggested by Mitja Horvat (pinkfluid) in https://github.com/ulfalizer/Kconfiglib/pull/50. Kconfig.node_iter() iterates through all menu nodes in the menu tree in Kconfig order. This saves scripts the trouble of implementing their own tree walking code. Have node_iter() take a 'unique_syms' flag that can be enabled to only include symbols defined in multiple locations once. This is often what you want when generating output (and is used by write_config()). Order is still preserved. Piggyback a fix to a syntax error test comment. Parsing has been tightened up now. --- kconfiglib.py | 92 +++++++++++++++++++++++++++++++++-------------------------- 1 file changed, 52 insertions(+), 40 deletions(-) (limited to 'kconfiglib.py') diff --git a/kconfiglib.py b/kconfiglib.py index a943f2f..8abec53 100644 --- a/kconfiglib.py +++ b/kconfiglib.py @@ -1087,49 +1087,11 @@ class Kconfig(object): with self._open(filename, "w") as f: f.write(header) - # Symbol._visited is set to True when we visit a symbol, so that - # symbols defined in multiple locations only get one .config entry. - # We reset it prior to writing out a new .config. It only needs to - # be reset for defined symbols, because undefined symbols will - # never be written out (because they do not appear in the menu tree - # rooted at Kconfig.top_node). - # - # The C tools reuse _write_to_conf for this, but we cache - # _write_to_conf together with the value and don't invalidate - # cached values when writing .config files, so that won't work. - # - # Note: The usage of _visited here is completely independent from - # the usage during dependency loop detection (which runs at the end - # of parsing). The attribute is just reused. - for sym in self.unique_defined_syms: - sym._visited = False - - # The 'top_node' menu node itself doesn't generate any output, so - # it's skipped over below - node = self.top_node - while 1: - # Jump to the next node with an iterative tree walk - if node.list: - node = node.list - elif node.next: - node = node.next - else: - while node.parent: - node = node.parent - if node.next: - node = node.next - break - else: - return - - # Write node - + for node in self.node_iter(unique_syms=True): item = node.item if isinstance(item, Symbol): - if not item._visited: - item._visited = True - f.write(item.config_string) + f.write(item.config_string) elif expr_value(node.dep) and \ ((item == MENU and expr_value(node.visibility)) or @@ -1354,6 +1316,56 @@ class Kconfig(object): self.syms[name]._old_val = val + def node_iter(self, unique_syms=False): + """ + Returns a generator for iterating through all MenuNode's in the Kconfig + tree. The iteration is done in Kconfig definition order (the children + of a node are visited before the next node is visited). + + The Kconfig.top_node menu node is skipped. It contains an implicit menu + that holds the top-level items. + + As an example, the following code will produce a list equal to + Kconfig.defined_syms: + + defined_syms = [node.item for node in kconf.node_iter() + if isinstance(node.item, Symbol)] + + unique_syms (default: False): + If True, only the first MenuNode will be included for symbols defined + in multiple locations. + + Using kconf.node_iter(True) in the example above would give a list + equal to unique_defined_syms. + """ + if unique_syms: + for sym in self.unique_defined_syms: + sym._visited = False + + node = self.top_node + while 1: + # Jump to the next node with an iterative tree walk + if node.list: + node = node.list + elif node.next: + node = node.next + else: + while node.parent: + node = node.parent + if node.next: + node = node.next + break + else: + # No more nodes + return + + if unique_syms and isinstance(node.item, Symbol): + if node.item._visited: + continue + node.item._visited = True + + yield node + def eval_string(self, s): """ Returns the tristate value of the expression 's', represented as 0, 1, -- cgit v1.2.3