|  | #!/usr/bin/env python3 | 
|  | # SPDX-License-Identifier: GPL-2.0+ | 
|  | # | 
|  | # Copyright 2024 Google LLC | 
|  | # Written by Simon Glass <sjg@chromium.org> | 
|  | # | 
|  |  | 
|  | """Build a FIT containing a lot of devicetree files | 
|  |  | 
|  | Usage: | 
|  | make_fit.py -A arm64 -n 'Linux-6.6' -O linux | 
|  | -o arch/arm64/boot/image.fit -k /tmp/kern/arch/arm64/boot/image.itk | 
|  | @arch/arm64/boot/dts/dtbs-list -E -c gzip | 
|  |  | 
|  | Creates a FIT containing the supplied kernel and a set of devicetree files, | 
|  | either specified individually or listed in a file (with an '@' prefix). | 
|  |  | 
|  | Use -E to generate an external FIT (where the data is placed after the | 
|  | FIT data structure). This allows parsing of the data without loading | 
|  | the entire FIT. | 
|  |  | 
|  | Use -c to compress the data, using bzip2, gzip, lz4, lzma, lzo and | 
|  | zstd algorithms. | 
|  |  | 
|  | Use -D to decompose "composite" DTBs into their base components and | 
|  | deduplicate the resulting base DTBs and DTB overlays. This requires the | 
|  | DTBs to be sourced from the kernel build directory, as the implementation | 
|  | looks at the .cmd files produced by the kernel build. | 
|  |  | 
|  | The resulting FIT can be booted by bootloaders which support FIT, such | 
|  | as U-Boot, Linuxboot, Tianocore, etc. | 
|  |  | 
|  | Note that this tool does not yet support adding a ramdisk / initrd. | 
|  | """ | 
|  |  | 
|  | import argparse | 
|  | import collections | 
|  | import os | 
|  | import subprocess | 
|  | import sys | 
|  | import tempfile | 
|  | import time | 
|  |  | 
|  | import libfdt | 
|  |  | 
|  |  | 
|  | # Tool extension and the name of the command-line tools | 
|  | CompTool = collections.namedtuple('CompTool', 'ext,tools') | 
|  |  | 
|  | COMP_TOOLS = { | 
|  | 'bzip2': CompTool('.bz2', 'bzip2'), | 
|  | 'gzip': CompTool('.gz', 'pigz,gzip'), | 
|  | 'lz4': CompTool('.lz4', 'lz4'), | 
|  | 'lzma': CompTool('.lzma', 'lzma'), | 
|  | 'lzo': CompTool('.lzo', 'lzop'), | 
|  | 'zstd': CompTool('.zstd', 'zstd'), | 
|  | } | 
|  |  | 
|  |  | 
|  | def parse_args(): | 
|  | """Parse the program ArgumentParser | 
|  |  | 
|  | Returns: | 
|  | Namespace object containing the arguments | 
|  | """ | 
|  | epilog = 'Build a FIT from a directory tree containing .dtb files' | 
|  | parser = argparse.ArgumentParser(epilog=epilog, fromfile_prefix_chars='@') | 
|  | parser.add_argument('-A', '--arch', type=str, required=True, | 
|  | help='Specifies the architecture') | 
|  | parser.add_argument('-c', '--compress', type=str, default='none', | 
|  | help='Specifies the compression') | 
|  | parser.add_argument('-D', '--decompose-dtbs', action='store_true', | 
|  | help='Decompose composite DTBs into base DTB and overlays') | 
|  | parser.add_argument('-E', '--external', action='store_true', | 
|  | help='Convert the FIT to use external data') | 
|  | parser.add_argument('-n', '--name', type=str, required=True, | 
|  | help='Specifies the name') | 
|  | parser.add_argument('-o', '--output', type=str, required=True, | 
|  | help='Specifies the output file (.fit)') | 
|  | parser.add_argument('-O', '--os', type=str, required=True, | 
|  | help='Specifies the operating system') | 
|  | parser.add_argument('-k', '--kernel', type=str, required=True, | 
|  | help='Specifies the (uncompressed) kernel input file (.itk)') | 
|  | parser.add_argument('-v', '--verbose', action='store_true', | 
|  | help='Enable verbose output') | 
|  | parser.add_argument('dtbs', type=str, nargs='*', | 
|  | help='Specifies the devicetree files to process') | 
|  |  | 
|  | return parser.parse_args() | 
|  |  | 
|  |  | 
|  | def setup_fit(fsw, name): | 
|  | """Make a start on writing the FIT | 
|  |  | 
|  | Outputs the root properties and the 'images' node | 
|  |  | 
|  | Args: | 
|  | fsw (libfdt.FdtSw): Object to use for writing | 
|  | name (str): Name of kernel image | 
|  | """ | 
|  | fsw.INC_SIZE = 65536 | 
|  | fsw.finish_reservemap() | 
|  | fsw.begin_node('') | 
|  | fsw.property_string('description', f'{name} with devicetree set') | 
|  | fsw.property_u32('#address-cells', 1) | 
|  |  | 
|  | fsw.property_u32('timestamp', int(time.time())) | 
|  | fsw.begin_node('images') | 
|  |  | 
|  |  | 
|  | def write_kernel(fsw, data, args): | 
|  | """Write out the kernel image | 
|  |  | 
|  | Writes a kernel node along with the required properties | 
|  |  | 
|  | Args: | 
|  | fsw (libfdt.FdtSw): Object to use for writing | 
|  | data (bytes): Data to write (possibly compressed) | 
|  | args (Namespace): Contains necessary strings: | 
|  | arch: FIT architecture, e.g. 'arm64' | 
|  | fit_os: Operating Systems, e.g. 'linux' | 
|  | name: Name of OS, e.g. 'Linux-6.6.0-rc7' | 
|  | compress: Compression algorithm to use, e.g. 'gzip' | 
|  | """ | 
|  | with fsw.add_node('kernel'): | 
|  | fsw.property_string('description', args.name) | 
|  | fsw.property_string('type', 'kernel_noload') | 
|  | fsw.property_string('arch', args.arch) | 
|  | fsw.property_string('os', args.os) | 
|  | fsw.property_string('compression', args.compress) | 
|  | fsw.property('data', data) | 
|  | fsw.property_u32('load', 0) | 
|  | fsw.property_u32('entry', 0) | 
|  |  | 
|  |  | 
|  | def finish_fit(fsw, entries): | 
|  | """Finish the FIT ready for use | 
|  |  | 
|  | Writes the /configurations node and subnodes | 
|  |  | 
|  | Args: | 
|  | fsw (libfdt.FdtSw): Object to use for writing | 
|  | entries (list of tuple): List of configurations: | 
|  | str: Description of model | 
|  | str: Compatible stringlist | 
|  | """ | 
|  | fsw.end_node() | 
|  | seq = 0 | 
|  | with fsw.add_node('configurations'): | 
|  | for model, compat, files in entries: | 
|  | seq += 1 | 
|  | with fsw.add_node(f'conf-{seq}'): | 
|  | fsw.property('compatible', bytes(compat)) | 
|  | fsw.property_string('description', model) | 
|  | fsw.property('fdt', bytes(''.join(f'fdt-{x}\x00' for x in files), "ascii")) | 
|  | fsw.property_string('kernel', 'kernel') | 
|  | fsw.end_node() | 
|  |  | 
|  |  | 
|  | def compress_data(inf, compress): | 
|  | """Compress data using a selected algorithm | 
|  |  | 
|  | Args: | 
|  | inf (IOBase): Filename containing the data to compress | 
|  | compress (str): Compression algorithm, e.g. 'gzip' | 
|  |  | 
|  | Return: | 
|  | bytes: Compressed data | 
|  | """ | 
|  | if compress == 'none': | 
|  | return inf.read() | 
|  |  | 
|  | comp = COMP_TOOLS.get(compress) | 
|  | if not comp: | 
|  | raise ValueError(f"Unknown compression algorithm '{compress}'") | 
|  |  | 
|  | with tempfile.NamedTemporaryFile() as comp_fname: | 
|  | with open(comp_fname.name, 'wb') as outf: | 
|  | done = False | 
|  | for tool in comp.tools.split(','): | 
|  | try: | 
|  | subprocess.call([tool, '-c'], stdin=inf, stdout=outf) | 
|  | done = True | 
|  | break | 
|  | except FileNotFoundError: | 
|  | pass | 
|  | if not done: | 
|  | raise ValueError(f'Missing tool(s): {comp.tools}\n') | 
|  | with open(comp_fname.name, 'rb') as compf: | 
|  | comp_data = compf.read() | 
|  | return comp_data | 
|  |  | 
|  |  | 
|  | def output_dtb(fsw, seq, fname, arch, compress): | 
|  | """Write out a single devicetree to the FIT | 
|  |  | 
|  | Args: | 
|  | fsw (libfdt.FdtSw): Object to use for writing | 
|  | seq (int): Sequence number (1 for first) | 
|  | fname (str): Filename containing the DTB | 
|  | arch: FIT architecture, e.g. 'arm64' | 
|  | compress (str): Compressed algorithm, e.g. 'gzip' | 
|  | """ | 
|  | with fsw.add_node(f'fdt-{seq}'): | 
|  | fsw.property_string('description', os.path.basename(fname)) | 
|  | fsw.property_string('type', 'flat_dt') | 
|  | fsw.property_string('arch', arch) | 
|  | fsw.property_string('compression', compress) | 
|  |  | 
|  | with open(fname, 'rb') as inf: | 
|  | compressed = compress_data(inf, compress) | 
|  | fsw.property('data', compressed) | 
|  |  | 
|  |  | 
|  | def process_dtb(fname, args): | 
|  | """Process an input DTB, decomposing it if requested and is possible | 
|  |  | 
|  | Args: | 
|  | fname (str): Filename containing the DTB | 
|  | args (Namespace): Program arguments | 
|  | Returns: | 
|  | tuple: | 
|  | str: Model name string | 
|  | str: Root compatible string | 
|  | files: list of filenames corresponding to the DTB | 
|  | """ | 
|  | # Get the compatible / model information | 
|  | with open(fname, 'rb') as inf: | 
|  | data = inf.read() | 
|  | fdt = libfdt.FdtRo(data) | 
|  | model = fdt.getprop(0, 'model').as_str() | 
|  | compat = fdt.getprop(0, 'compatible') | 
|  |  | 
|  | if args.decompose_dtbs: | 
|  | # Check if the DTB needs to be decomposed | 
|  | path, basename = os.path.split(fname) | 
|  | cmd_fname = os.path.join(path, f'.{basename}.cmd') | 
|  | with open(cmd_fname, 'r', encoding='ascii') as inf: | 
|  | cmd = inf.read() | 
|  |  | 
|  | if 'scripts/dtc/fdtoverlay' in cmd: | 
|  | # This depends on the structure of the composite DTB command | 
|  | files = cmd.split() | 
|  | files = files[files.index('-i') + 1:] | 
|  | else: | 
|  | files = [fname] | 
|  | else: | 
|  | files = [fname] | 
|  |  | 
|  | return (model, compat, files) | 
|  |  | 
|  | def build_fit(args): | 
|  | """Build the FIT from the provided files and arguments | 
|  |  | 
|  | Args: | 
|  | args (Namespace): Program arguments | 
|  |  | 
|  | Returns: | 
|  | tuple: | 
|  | bytes: FIT data | 
|  | int: Number of configurations generated | 
|  | size: Total uncompressed size of data | 
|  | """ | 
|  | seq = 0 | 
|  | size = 0 | 
|  | fsw = libfdt.FdtSw() | 
|  | setup_fit(fsw, args.name) | 
|  | entries = [] | 
|  | fdts = {} | 
|  |  | 
|  | # Handle the kernel | 
|  | with open(args.kernel, 'rb') as inf: | 
|  | comp_data = compress_data(inf, args.compress) | 
|  | size += os.path.getsize(args.kernel) | 
|  | write_kernel(fsw, comp_data, args) | 
|  |  | 
|  | for fname in args.dtbs: | 
|  | # Ignore non-DTB (*.dtb) files | 
|  | if os.path.splitext(fname)[1] != '.dtb': | 
|  | continue | 
|  |  | 
|  | try: | 
|  | (model, compat, files) = process_dtb(fname, args) | 
|  | except Exception as e: | 
|  | sys.stderr.write(f"Error processing {fname}:\n") | 
|  | raise e | 
|  |  | 
|  | for fn in files: | 
|  | if fn not in fdts: | 
|  | seq += 1 | 
|  | size += os.path.getsize(fn) | 
|  | output_dtb(fsw, seq, fn, args.arch, args.compress) | 
|  | fdts[fn] = seq | 
|  |  | 
|  | files_seq = [fdts[fn] for fn in files] | 
|  |  | 
|  | entries.append([model, compat, files_seq]) | 
|  |  | 
|  | finish_fit(fsw, entries) | 
|  |  | 
|  | # Include the kernel itself in the returned file count | 
|  | return fsw.as_fdt().as_bytearray(), seq + 1, size | 
|  |  | 
|  |  | 
|  | def run_make_fit(): | 
|  | """Run the tool's main logic""" | 
|  | args = parse_args() | 
|  |  | 
|  | out_data, count, size = build_fit(args) | 
|  | with open(args.output, 'wb') as outf: | 
|  | outf.write(out_data) | 
|  |  | 
|  | ext_fit_size = None | 
|  | if args.external: | 
|  | mkimage = os.environ.get('MKIMAGE', 'mkimage') | 
|  | subprocess.check_call([mkimage, '-E', '-F', args.output], | 
|  | stdout=subprocess.DEVNULL) | 
|  |  | 
|  | with open(args.output, 'rb') as inf: | 
|  | data = inf.read() | 
|  | ext_fit = libfdt.FdtRo(data) | 
|  | ext_fit_size = ext_fit.totalsize() | 
|  |  | 
|  | if args.verbose: | 
|  | comp_size = len(out_data) | 
|  | print(f'FIT size {comp_size:#x}/{comp_size / 1024 / 1024:.1f} MB', | 
|  | end='') | 
|  | if ext_fit_size: | 
|  | print(f', header {ext_fit_size:#x}/{ext_fit_size / 1024:.1f} KB', | 
|  | end='') | 
|  | print(f', {count} files, uncompressed {size / 1024 / 1024:.1f} MB') | 
|  |  | 
|  |  | 
|  | if __name__ == "__main__": | 
|  | sys.exit(run_make_fit()) |