2019-09-23 17:02:43 +08:00
|
|
|
# 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>
|
|
|
|
|
|
|
|
import collections
|
|
|
|
import re
|
2021-01-15 08:39:11 +08:00
|
|
|
from typing import List, Set
|
2019-09-23 17:02:43 +08:00
|
|
|
|
2020-03-24 10:43:33 +08:00
|
|
|
CONFIG_IS_NOT_SET_PATTERN = r'^# CONFIG_(\w+) is not set$'
|
2020-06-08 05:57:15 +08:00
|
|
|
CONFIG_PATTERN = r'^CONFIG_(\w+)=(\S+|".*")$'
|
2019-09-23 17:02:43 +08:00
|
|
|
|
2021-02-23 13:49:30 +08:00
|
|
|
KconfigEntryBase = collections.namedtuple('KconfigEntryBase', ['name', 'value'])
|
2019-09-23 17:02:43 +08:00
|
|
|
|
|
|
|
class KconfigEntry(KconfigEntryBase):
|
|
|
|
|
|
|
|
def __str__(self) -> str:
|
2020-03-24 10:43:33 +08:00
|
|
|
if self.value == 'n':
|
|
|
|
return r'# CONFIG_%s is not set' % (self.name)
|
|
|
|
else:
|
|
|
|
return r'CONFIG_%s=%s' % (self.name, self.value)
|
2019-09-23 17:02:43 +08:00
|
|
|
|
|
|
|
|
|
|
|
class KconfigParseError(Exception):
|
|
|
|
"""Error parsing Kconfig defconfig or .config."""
|
|
|
|
|
|
|
|
|
|
|
|
class Kconfig(object):
|
|
|
|
"""Represents defconfig or .config specified using the Kconfig language."""
|
|
|
|
|
2021-01-15 08:39:11 +08:00
|
|
|
def __init__(self) -> None:
|
|
|
|
self._entries = [] # type: List[KconfigEntry]
|
2019-09-23 17:02:43 +08:00
|
|
|
|
2021-01-15 08:39:11 +08:00
|
|
|
def entries(self) -> Set[KconfigEntry]:
|
2019-09-23 17:02:43 +08:00
|
|
|
return set(self._entries)
|
|
|
|
|
|
|
|
def add_entry(self, entry: KconfigEntry) -> None:
|
|
|
|
self._entries.append(entry)
|
|
|
|
|
|
|
|
def is_subset_of(self, other: 'Kconfig') -> bool:
|
2020-12-09 07:21:02 +08:00
|
|
|
other_dict = {e.name: e.value for e in other.entries()}
|
2020-03-24 10:43:33 +08:00
|
|
|
for a in self.entries():
|
2020-12-09 07:21:02 +08:00
|
|
|
b = other_dict.get(a.name)
|
|
|
|
if b is None:
|
|
|
|
if a.value == 'n':
|
2020-03-24 10:43:33 +08:00
|
|
|
continue
|
2020-12-09 07:21:02 +08:00
|
|
|
return False
|
|
|
|
elif a.value != b:
|
2020-03-24 10:43:33 +08:00
|
|
|
return False
|
|
|
|
return True
|
2019-09-23 17:02:43 +08:00
|
|
|
|
2021-05-27 05:24:06 +08:00
|
|
|
def merge_in_entries(self, other: 'Kconfig') -> None:
|
|
|
|
if other.is_subset_of(self):
|
|
|
|
return
|
|
|
|
self._entries = list(self.entries().union(other.entries()))
|
|
|
|
|
2019-09-23 17:02:43 +08:00
|
|
|
def write_to_file(self, path: str) -> None:
|
2021-05-27 05:24:06 +08:00
|
|
|
with open(path, 'a+') as f:
|
2019-09-23 17:02:43 +08:00
|
|
|
for entry in self.entries():
|
|
|
|
f.write(str(entry) + '\n')
|
|
|
|
|
|
|
|
def parse_from_string(self, blob: str) -> None:
|
|
|
|
"""Parses a string containing KconfigEntrys and populates this Kconfig."""
|
|
|
|
self._entries = []
|
|
|
|
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
|
2020-03-24 10:43:33 +08:00
|
|
|
|
|
|
|
match = config_matcher.match(line)
|
|
|
|
if match:
|
|
|
|
entry = KconfigEntry(match.group(1), match.group(2))
|
|
|
|
self.add_entry(entry)
|
|
|
|
continue
|
|
|
|
|
|
|
|
empty_match = is_not_set_matcher.match(line)
|
|
|
|
if empty_match:
|
|
|
|
entry = KconfigEntry(empty_match.group(1), 'n')
|
|
|
|
self.add_entry(entry)
|
|
|
|
continue
|
|
|
|
|
|
|
|
if line[0] == '#':
|
2019-09-23 17:02:43 +08:00
|
|
|
continue
|
|
|
|
else:
|
|
|
|
raise KconfigParseError('Failed to parse: ' + line)
|
|
|
|
|
|
|
|
def read_from_file(self, path: str) -> None:
|
|
|
|
with open(path, 'r') as f:
|
|
|
|
self.parse_from_string(f.read())
|