mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	This makes it easier to pinpoint where the error happened. For example:
  FIT     arch/powerpc/boot/image.fit
Error processing arch/powerpc/boot/dts/microwatt.dtb:
Traceback (most recent call last):
  File "/home/jn/dev/linux/linux-git/build-mpc83xx/../scripts/make_fit.py", line 335, in <module>
    sys.exit(run_make_fit())
             ^^^^^^^^^^^^^^
  File "/home/jn/dev/linux/linux-git/build-mpc83xx/../scripts/make_fit.py", line 309, in run_make_fit
    out_data, count, size = build_fit(args)
                            ^^^^^^^^^^^^^^^
  File "/home/jn/dev/linux/linux-git/build-mpc83xx/../scripts/make_fit.py", line 286, in build_fit
    raise e
  File "/home/jn/dev/linux/linux-git/build-mpc83xx/../scripts/make_fit.py", line 283, in build_fit
    (model, compat, files) = process_dtb(fname, args)
                             ^^^^^^^^^^^^^^^^^^^^^^^^
  File "/home/jn/dev/linux/linux-git/build-mpc83xx/../scripts/make_fit.py", line 231, in process_dtb
    model = fdt.getprop(0, 'model').as_str()
            ^^^^^^^^^^^^^^^^^^^^^^^
  File "/usr/lib/python3/dist-packages/libfdt.py", line 448, in getprop
    pdata = check_err_null(fdt_getprop(self._fdt, nodeoffset, prop_name),
            ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "/usr/lib/python3/dist-packages/libfdt.py", line 153, in check_err_null
    raise FdtException(val)
libfdt.FdtException: pylibfdt error -1: FDT_ERR_NOTFOUND
Signed-off-by: J. Neuschäfer <j.ne@posteo.net>
Link: https://lore.kernel.org/r/20250209-makefit-v1-1-bfe6151e8f0a@posteo.net
Signed-off-by: Rob Herring (Arm) <robh@kernel.org>
		
	
			
		
			
				
	
	
		
			335 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			335 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
#!/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())
 |