| # Copyright 2014 The Chromium OS Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| import gzip, logging, os, re |
| from autotest_lib.client.bin import utils |
| from autotest_lib.client.common_lib import error |
| |
| class KernelConfig(): |
| """ |
| Parse the kernel config and enable us to query it. |
| Used to verify the kernel config (see kernel_ConfigVerify). |
| """ |
| |
| def _passed(self, msg): |
| logging.info('ok: %s', msg) |
| |
| def _failed(self, msg): |
| logging.error('FAIL: %s', msg) |
| self._failures.append(msg) |
| |
| def failures(self): |
| """Return the list of failures that occured during the test. |
| |
| @return a list of string describing errors that occured since |
| initialization. |
| """ |
| return self._failures |
| |
| def _fatal(self, msg): |
| logging.error('FATAL: %s', msg) |
| raise error.TestError(msg) |
| |
| def get(self, key, default): |
| """Get the value associated to key or default if it does not exist |
| |
| @param key: key to look for. |
| @param default: value returned if key is not set in self._config |
| """ |
| return self._config.get(key, default) |
| |
| def _config_required(self, name, wanted): |
| value = self._config.get(name, None) |
| if value in wanted: |
| self._passed('"%s" was "%s" in kernel config' % (name, value)) |
| else: |
| states = [] |
| for state in wanted: |
| if state == None: |
| states.append("unset") |
| else: |
| states.append(state) |
| self._failed('"%s" was "%s" (wanted one of "%s") in kernel config' % |
| (name, value, '|'.join(states))) |
| |
| def has_value(self, name, value): |
| """Determine if the name config item has a specific value. |
| |
| @param name: name of config item to test |
| @param value: value expected for the given config name |
| """ |
| self._config_required('CONFIG_%s' % (name), value) |
| |
| def has_builtin(self, name): |
| """Check if the specific config item is built-in (present but not |
| built as a module). |
| |
| @param name: name of config item to test |
| """ |
| wanted = ['y'] |
| if name in self._missing_ok: |
| wanted.append(None) |
| self.has_value(name, wanted) |
| |
| def has_module(self, name): |
| """Check if the specific config item is a module (present but not |
| built-in). |
| |
| @param name: name of config item to test |
| """ |
| wanted = ['m'] |
| if name in self._missing_ok: |
| wanted.append(None) |
| self.has_value(name, wanted) |
| |
| def is_enabled(self, name): |
| """Check if the specific config item is present (either built-in or |
| a module). |
| |
| @param name: name of config item to test |
| """ |
| wanted = ['y', 'm'] |
| if name in self._missing_ok: |
| wanted.append(None) |
| self.has_value(name, wanted) |
| |
| def is_missing(self, name): |
| """Check if the specific config item is not present (neither built-in |
| nor a module). |
| |
| @param name: name of config item to test |
| """ |
| self.has_value(name, [None]) |
| |
| def is_exclusive(self, exclusive): |
| """Given a config item regex, make sure only the expected items |
| are present in the kernel configs. |
| |
| @param exclusive: hash containing "missing", "builtin", "module", |
| "enabled" each to be checked with the corresponding |
| has_* function based on config items matching the |
| "regex" value. |
| """ |
| expected = set() |
| for name in exclusive['missing']: |
| self.is_missing(name) |
| for name in exclusive['builtin']: |
| self.has_builtin(name) |
| expected.add('CONFIG_%s' % (name)) |
| for name in exclusive['module']: |
| self.has_module(name) |
| expected.add('CONFIG_%s' % (name)) |
| for name in exclusive['enabled']: |
| self.is_enabled(name) |
| expected.add('CONFIG_%s' % (name)) |
| |
| # Now make sure nothing else with the specified regex exists. |
| regex = r'CONFIG_%s' % (exclusive['regex']) |
| for name in self._config: |
| if not re.match(regex, name): |
| continue |
| if not name in expected: |
| self._failed('"%s" found for "%s" when only "%s" allowed' % |
| (name, regex, "|".join(expected))) |
| |
| def _open_config(self): |
| """Open the kernel's build config file. Attempt to use the built-in |
| symbols from /proc first, then fall back to looking for a text file |
| in /boot. |
| |
| @return fileobj for open config file |
| """ |
| filename = '/proc/config.gz' |
| if not os.path.exists(filename): |
| utils.system("modprobe configs", ignore_status=True) |
| if os.path.exists(filename): |
| return gzip.open(filename, "r") |
| |
| filename = '/boot/config-%s' % utils.system_output('uname -r') |
| if os.path.exists(filename): |
| logging.info('Falling back to reading %s', filename) |
| return file(filename, "r") |
| |
| self._fatal("Cannot locate suitable kernel config file") |
| |
| def initialize(self, missing_ok=None): |
| """Load the kernel configuration and parse it. |
| """ |
| fileobj = self._open_config() |
| # Import kernel config variables into a dictionary for each searching. |
| config = dict() |
| for item in fileobj.readlines(): |
| item = item.strip() |
| if not '=' in item: |
| continue |
| key, value = item.split('=', 1) |
| config[key] = value |
| |
| # Make sure we actually loaded something sensible. |
| if len(config) == 0: |
| self._fatal('No CONFIG variables found!') |
| |
| self._config = config |
| self._failures = [] |
| self._missing_ok = set() |
| if missing_ok: |
| self._missing_ok |= set(missing_ok) |