mirror of
https://github.com/vyos/vyos-build.git
synced 2025-10-01 20:28:40 +02:00
The logic of generating the required firmware file name has been moved from build-linux-firmware.sh directly into list-required-firmware.py which makes much more sense. That move was made by Daniil Baturin - I only did the integration part. Thanks!
134 lines
4.8 KiB
Python
Executable File
134 lines
4.8 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
#
|
|
# Copyright (C) 2020 Daniil Baturin
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License version 2 or later as
|
|
# published by the Free Software Foundation.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
import re
|
|
import os
|
|
import sys
|
|
import glob
|
|
import argparse
|
|
import subprocess
|
|
|
|
# Loads the kernel config -- only options set to y or m
|
|
def load_config(path):
|
|
with open(path, 'r') as f:
|
|
config = f.read()
|
|
targets = re.findall(r'(.*)=(?:y|m)', config)
|
|
return targets
|
|
|
|
# Finds subdir targets from the Makefile
|
|
# that are enabled by the kernel build config
|
|
def find_enabled_subdirs(config, makefile_path):
|
|
try:
|
|
with open(makefile_path, 'r') as f:
|
|
makefile = f.read()
|
|
except OSError:
|
|
# Shouldn't happen due to the way collect_source_files()
|
|
# calls this function.
|
|
return []
|
|
|
|
dir_stmts = re.findall(r'obj-\$\((.*)\)\s+\+=\s+(.*)/(?:\n|$)', makefile)
|
|
subdirs = []
|
|
|
|
for ds in dir_stmts:
|
|
config_key, src_dir = ds
|
|
|
|
if args.debug:
|
|
print("Processing make targets from {0} ({1})".format(ds[1], ds[0]), file=sys.stderr)
|
|
if config_key in config:
|
|
subdirs.append(src_dir)
|
|
elif args.debug:
|
|
print("{0} is disabled in the config, ignoring {1}".format(ds[0], ds[1]), file=sys.stderr)
|
|
|
|
return subdirs
|
|
|
|
# For filtering
|
|
def file_loads_firmware(file):
|
|
with open(file, 'r') as f:
|
|
source = f.read()
|
|
if re.search(r'MODULE_FIRMWARE\((.*)\)', source):
|
|
return True
|
|
|
|
# Find all source files that reference firmware
|
|
def collect_source_files(config, path):
|
|
files = []
|
|
|
|
makefile = os.path.join(path, "Makefile")
|
|
|
|
# Find and process all C files in this directory
|
|
# This is a compromise: sometimes there are single-file modules,
|
|
# that in fact may be disabled in the config,
|
|
# so this approach can create occasional false positives.
|
|
c_files = glob.glob("{0}/*.c".format(path))
|
|
files = list(filter(file_loads_firmware, c_files))
|
|
|
|
# Now walk the subdirectories
|
|
enabled_subdirs = find_enabled_subdirs(config, makefile)
|
|
subdirs = glob.glob("{0}/*/".format(path))
|
|
for d in subdirs:
|
|
dir_name = d.rstrip("/")
|
|
|
|
if os.path.exists(os.path.join(d, "Makefile")):
|
|
# If there's a makefile, it's an independent module
|
|
# or a high level dir
|
|
if os.path.basename(dir_name) in enabled_subdirs:
|
|
files = files + collect_source_files(config, d)
|
|
else:
|
|
# It's simply a subdirectory of the current module
|
|
# Some modules, like iwlwifi, keep their firmware-loading files
|
|
# in subdirs, so we have to handle this case
|
|
c_files = glob.iglob("{0}/**/*.c".format(d), recursive=True)
|
|
files += list(filter(file_loads_firmware, c_files))
|
|
|
|
return files
|
|
|
|
if __name__ == '__main__':
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("-s", "--source-dir", action="append", help="Kernel source directory to process", required=True)
|
|
parser.add_argument("-c", "--kernel-config", action="store", help="Kernel configuration")
|
|
parser.add_argument("-d", "--debug", action="store_true", help="Enable Debug output")
|
|
parser.add_argument("-f", "--list-source-files", action="store_true", help="List source files that reference firmware and exit")
|
|
args = parser.parse_args()
|
|
|
|
if not args.kernel_config:
|
|
args.kernel_config = ".config"
|
|
|
|
config = load_config(args.kernel_config)
|
|
|
|
# Collect source files that reference firmware
|
|
for directory in args.source_dir:
|
|
source_files = collect_source_files(config, directory)
|
|
|
|
if args.list_source_files:
|
|
for sf in source_files:
|
|
print(sf)
|
|
else:
|
|
fw_files = []
|
|
for sf in source_files:
|
|
i_file = re.sub(r'\.c', r'.i', sf)
|
|
res = subprocess.run(["make {0} 2>&1".format(i_file)], shell=True, capture_output=True)
|
|
if res.returncode != 0:
|
|
print("Failed to preprocess file {0}".format(sf), file=sys.stderr)
|
|
print(res.stdout.decode(), file=sys.stderr)
|
|
else:
|
|
with open(i_file, 'r') as f:
|
|
source = f.read()
|
|
fw_statements = re.findall(r'__UNIQUE_ID_firmware.*"firmware"\s+"="\s+(.*);', source)
|
|
fw_files += list(map(lambda s: re.sub(r'(\s|")', r'', s), fw_statements))
|
|
|
|
for fw in fw_files:
|
|
print(fw)
|