Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 1 | # SPDX-License-Identifier: GPL-2.0+ |
| 2 | # |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 3 | # Copyright (c) 2018, Linaro Limited |
| 4 | # Author: Takahiro Akashi <takahiro.akashi@linaro.org> |
| 5 | |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 6 | """Helper functions for dealing with filesystems""" |
| 7 | |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 8 | import re |
| 9 | import os |
| 10 | from subprocess import call, check_call, check_output, CalledProcessError |
| 11 | |
Richard Weinberger | 41eca32 | 2024-11-21 15:32:07 -0700 | [diff] [blame] | 12 | def mk_fs(config, fs_type, size, prefix, src_dir=None, size_gran = 0x100000): |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 13 | """Create a file system volume |
| 14 | |
| 15 | Args: |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 16 | config (u_boot_config): U-Boot configuration |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 17 | fs_type (str): File system type, e.g. 'ext4' |
| 18 | size (int): Size of file system in bytes |
| 19 | prefix (str): Prefix string of volume's file name |
Richard Weinberger | 41eca32 | 2024-11-21 15:32:07 -0700 | [diff] [blame] | 20 | src_dir (str): Root directory to use, or None for none |
Christian Taedcke | d604880 | 2023-11-15 13:44:23 +0100 | [diff] [blame] | 21 | size_gran (int): Size granularity of file system image in bytes |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 22 | |
| 23 | Raises: |
| 24 | CalledProcessError: if any error occurs when creating the filesystem |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 25 | """ |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 26 | fs_img = f'{prefix}.{fs_type}.img' |
Simon Glass | 0870921 | 2023-08-24 13:55:38 -0600 | [diff] [blame] | 27 | fs_img = os.path.join(config.persistent_data_dir, fs_img) |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 28 | |
Christian Taedcke | ff0ceca | 2023-11-15 13:44:21 +0100 | [diff] [blame] | 29 | if fs_type == 'fat12': |
| 30 | mkfs_opt = '-F 12' |
| 31 | elif fs_type == 'fat16': |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 32 | mkfs_opt = '-F 16' |
| 33 | elif fs_type == 'fat32': |
| 34 | mkfs_opt = '-F 32' |
| 35 | else: |
| 36 | mkfs_opt = '' |
| 37 | |
| 38 | if re.match('fat', fs_type): |
| 39 | fs_lnxtype = 'vfat' |
| 40 | else: |
| 41 | fs_lnxtype = fs_type |
| 42 | |
Richard Weinberger | 41eca32 | 2024-11-21 15:32:07 -0700 | [diff] [blame] | 43 | if src_dir: |
| 44 | if fs_lnxtype == 'ext4': |
| 45 | mkfs_opt = mkfs_opt + ' -d ' + src_dir |
| 46 | elif fs_lnxtype != 'vfat': |
| 47 | raise ValueError(f'src_dir not implemented for fs {fs_lnxtype}') |
| 48 | |
Christian Taedcke | d604880 | 2023-11-15 13:44:23 +0100 | [diff] [blame] | 49 | count = (size + size_gran - 1) // size_gran |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 50 | |
| 51 | # Some distributions do not add /sbin to the default PATH, where mkfs lives |
| 52 | if '/sbin' not in os.environ["PATH"].split(os.pathsep): |
| 53 | os.environ["PATH"] += os.pathsep + '/sbin' |
| 54 | |
| 55 | try: |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 56 | check_call(f'rm -f {fs_img}', shell=True) |
Christian Taedcke | d604880 | 2023-11-15 13:44:23 +0100 | [diff] [blame] | 57 | check_call(f'dd if=/dev/zero of={fs_img} bs={size_gran} count={count}', |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 58 | shell=True) |
| 59 | check_call(f'mkfs.{fs_lnxtype} {mkfs_opt} {fs_img}', shell=True) |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 60 | if fs_type == 'ext4': |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 61 | sb_content = check_output(f'tune2fs -l {fs_img}', |
| 62 | shell=True).decode() |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 63 | if 'metadata_csum' in sb_content: |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 64 | check_call(f'tune2fs -O ^metadata_csum {fs_img}', shell=True) |
Richard Weinberger | 41eca32 | 2024-11-21 15:32:07 -0700 | [diff] [blame] | 65 | elif fs_lnxtype == 'vfat' and src_dir: |
| 66 | check_call(f'mcopy -i {fs_img} -vsmpQ {src_dir}/* ::/', shell=True) |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 67 | return fs_img |
| 68 | except CalledProcessError: |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 69 | call(f'rm -f {fs_img}', shell=True) |
Simon Glass | 1d5006c | 2022-10-29 19:47:05 -0600 | [diff] [blame] | 70 | raise |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 71 | |
Tom Rini | 20b0919 | 2025-03-20 07:59:24 -0600 | [diff] [blame^] | 72 | def setup_image(ubman, devnum, part_type, img_size=20, second_part=False, |
| 73 | basename='mmc'): |
| 74 | """Create a disk image with a single partition |
| 75 | |
| 76 | Args: |
| 77 | ubman (ConsoleBase): Console to use |
| 78 | devnum (int): Device number to use, e.g. 1 |
| 79 | part_type (int): Partition type, e.g. 0xc for FAT32 |
| 80 | img_size (int): Image size in MiB |
| 81 | second_part (bool): True to contain a small second partition |
| 82 | basename (str): Base name to use in the filename, e.g. 'mmc' |
| 83 | |
| 84 | Returns: |
| 85 | tuple: |
| 86 | str: Filename of MMC image |
| 87 | str: Directory name of scratch directory |
| 88 | """ |
| 89 | fname = os.path.join(ubman.config.source_dir, f'{basename}{devnum}.img') |
| 90 | mnt = os.path.join(ubman.config.persistent_data_dir, 'scratch') |
| 91 | |
| 92 | spec = f'type={part_type:x}, size={img_size - 2}M, start=1M, bootable' |
| 93 | if second_part: |
| 94 | spec += '\ntype=c' |
| 95 | |
| 96 | try: |
| 97 | check_call(f'mkdir -p {mnt}', shell=True) |
| 98 | check_call(f'qemu-img create {fname} 20M', shell=True) |
| 99 | check_call(f'printf "{spec}" | sfdisk {fname}', shell=True) |
| 100 | except CalledProcessError: |
| 101 | call(f'rm -f {fname}', shell=True) |
| 102 | raise |
| 103 | |
| 104 | return fname, mnt |
| 105 | |
Simon Glass | fbd9550 | 2022-10-29 19:47:06 -0600 | [diff] [blame] | 106 | # Just for trying out |
| 107 | if __name__ == "__main__": |
| 108 | import collections |
| 109 | |
| 110 | CNF= collections.namedtuple('config', 'persistent_data_dir') |
| 111 | |
| 112 | mk_fs(CNF('.'), 'ext4', 0x1000000, 'pref') |