| # SPDX-License-Identifier: GPL-2.0 |
| # |
| # Builds a .config from a kunitconfig. |
| # |
| # Copyright (C) 2019, Google LLC. |
| # Author: Felix Guo <felixguoxiuping@gmail.com> |
| # Author: Brendan Higgins <brendanhiggins@google.com> |
| |
| from dataclasses import dataclass |
| import re |
| from typing import Any, Dict, Iterable, List, Tuple |
| |
| CONFIG_IS_NOT_SET_PATTERN = r'^# CONFIG_(\w+) is not set$' |
| CONFIG_PATTERN = r'^CONFIG_(\w+)=(\S+|".*")$' |
| |
| @dataclass(frozen=True) |
| class KconfigEntry: |
| name: str |
| value: str |
| |
| def __str__(self) -> str: |
| if self.value == 'n': |
| return f'# CONFIG_{self.name} is not set' |
| return f'CONFIG_{self.name}={self.value}' |
| |
| |
| class KconfigParseError(Exception): |
| """Error parsing Kconfig defconfig or .config.""" |
| |
| |
| class Kconfig: |
| """Represents defconfig or .config specified using the Kconfig language.""" |
| |
| def __init__(self) -> None: |
| self._entries = {} # type: Dict[str, str] |
| |
| def __eq__(self, other: Any) -> bool: |
| if not isinstance(other, self.__class__): |
| return False |
| return self._entries == other._entries |
| |
| def __repr__(self) -> str: |
| return ','.join(str(e) for e in self.as_entries()) |
| |
| def as_entries(self) -> Iterable[KconfigEntry]: |
| for name, value in self._entries.items(): |
| yield KconfigEntry(name, value) |
| |
| def add_entry(self, name: str, value: str) -> None: |
| self._entries[name] = value |
| |
| def is_subset_of(self, other: 'Kconfig') -> bool: |
| for name, value in self._entries.items(): |
| b = other._entries.get(name) |
| if b is None: |
| if value == 'n': |
| continue |
| return False |
| if value != b: |
| return False |
| return True |
| |
| def conflicting_options(self, other: 'Kconfig') -> List[Tuple[KconfigEntry, KconfigEntry]]: |
| diff = [] # type: List[Tuple[KconfigEntry, KconfigEntry]] |
| for name, value in self._entries.items(): |
| b = other._entries.get(name) |
| if b and value != b: |
| pair = (KconfigEntry(name, value), KconfigEntry(name, b)) |
| diff.append(pair) |
| return diff |
| |
| def merge_in_entries(self, other: 'Kconfig') -> None: |
| for name, value in other._entries.items(): |
| self._entries[name] = value |
| |
| def write_to_file(self, path: str) -> None: |
| with open(path, 'a+') as f: |
| for e in self.as_entries(): |
| f.write(str(e) + '\n') |
| |
| def parse_file(path: str) -> Kconfig: |
| with open(path, 'r') as f: |
| return parse_from_string(f.read()) |
| |
| def parse_from_string(blob: str) -> Kconfig: |
| """Parses a string containing Kconfig entries.""" |
| kconfig = Kconfig() |
| is_not_set_matcher = re.compile(CONFIG_IS_NOT_SET_PATTERN) |
| config_matcher = re.compile(CONFIG_PATTERN) |
| for line in blob.split('\n'): |
| line = line.strip() |
| if not line: |
| continue |
| |
| match = config_matcher.match(line) |
| if match: |
| kconfig.add_entry(match.group(1), match.group(2)) |
| continue |
| |
| empty_match = is_not_set_matcher.match(line) |
| if empty_match: |
| kconfig.add_entry(empty_match.group(1), 'n') |
| continue |
| |
| if line[0] == '#': |
| continue |
| raise KconfigParseError('Failed to parse: ' + line) |
| return kconfig |