Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 1 | # |
| 2 | # Copyright (c) 2013-2016 Intel Corporation. |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 3 | # |
Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 4 | # SPDX-License-Identifier: GPL-2.0-only |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 5 | # |
| 6 | # DESCRIPTION |
| 7 | # This module provides the OpenEmbedded partition object definitions. |
| 8 | # |
| 9 | # AUTHORS |
| 10 | # Tom Zanussi <tom.zanussi (at] linux.intel.com> |
| 11 | # Ed Bartosh <ed.bartosh> (at] linux.intel.com> |
| 12 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 13 | import logging |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 14 | import os |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 15 | import uuid |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 16 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 17 | from wic import WicError |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 18 | from wic.misc import exec_cmd, exec_native_cmd, get_bitbake_var |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 19 | from wic.pluginbase import PluginMgr |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 20 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 21 | logger = logging.getLogger('wic') |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 22 | |
Patrick Williams | c0f7c04 | 2017-02-23 20:41:17 -0600 | [diff] [blame] | 23 | class Partition(): |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 24 | |
| 25 | def __init__(self, args, lineno): |
| 26 | self.args = args |
| 27 | self.active = args.active |
| 28 | self.align = args.align |
| 29 | self.disk = args.disk |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 30 | self.device = None |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 31 | self.extra_space = args.extra_space |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 32 | self.exclude_path = args.exclude_path |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 33 | self.include_path = args.include_path |
| 34 | self.change_directory = args.change_directory |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 35 | self.fsopts = args.fsopts |
Andrew Geissler | d583833 | 2022-05-27 11:33:10 -0500 | [diff] [blame] | 36 | self.fspassno = args.fspassno |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 37 | self.fstype = args.fstype |
| 38 | self.label = args.label |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 39 | self.use_label = args.use_label |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 40 | self.mkfs_extraopts = args.mkfs_extraopts |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 41 | self.mountpoint = args.mountpoint |
| 42 | self.no_table = args.no_table |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 43 | self.num = None |
Andrew Geissler | 4ed12e1 | 2020-06-05 18:00:41 -0500 | [diff] [blame] | 44 | self.offset = args.offset |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 45 | self.overhead_factor = args.overhead_factor |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 46 | self.part_name = args.part_name |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 47 | self.part_type = args.part_type |
| 48 | self.rootfs_dir = args.rootfs_dir |
| 49 | self.size = args.size |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 50 | self.fixed_size = args.fixed_size |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 51 | self.source = args.source |
| 52 | self.sourceparams = args.sourceparams |
Patrick Williams | c0f7c04 | 2017-02-23 20:41:17 -0600 | [diff] [blame] | 53 | self.system_id = args.system_id |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 54 | self.use_uuid = args.use_uuid |
| 55 | self.uuid = args.uuid |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 56 | self.fsuuid = args.fsuuid |
Brad Bishop | 08902b0 | 2019-08-20 09:16:51 -0400 | [diff] [blame] | 57 | self.type = args.type |
Andrew Geissler | d159c7f | 2021-09-02 21:05:58 -0500 | [diff] [blame] | 58 | self.no_fstab_update = args.no_fstab_update |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 59 | self.updated_fstab_path = None |
| 60 | self.has_fstab = False |
| 61 | self.update_fstab_in_rootfs = False |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 62 | |
| 63 | self.lineno = lineno |
| 64 | self.source_file = "" |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 65 | |
| 66 | def get_extra_block_count(self, current_blocks): |
| 67 | """ |
| 68 | The --size param is reflected in self.size (in kB), and we already |
| 69 | have current_blocks (1k) blocks, calculate and return the |
| 70 | number of (1k) blocks we need to add to get to --size, 0 if |
| 71 | we're already there or beyond. |
| 72 | """ |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 73 | logger.debug("Requested partition size for %s: %d", |
| 74 | self.mountpoint, self.size) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 75 | |
| 76 | if not self.size: |
| 77 | return 0 |
| 78 | |
| 79 | requested_blocks = self.size |
| 80 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 81 | logger.debug("Requested blocks %d, current_blocks %d", |
| 82 | requested_blocks, current_blocks) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 83 | |
| 84 | if requested_blocks > current_blocks: |
| 85 | return requested_blocks - current_blocks |
| 86 | else: |
| 87 | return 0 |
| 88 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 89 | def get_rootfs_size(self, actual_rootfs_size=0): |
| 90 | """ |
| 91 | Calculate the required size of rootfs taking into consideration |
| 92 | --size/--fixed-size flags as well as overhead and extra space, as |
| 93 | specified in kickstart file. Raises an error if the |
| 94 | `actual_rootfs_size` is larger than fixed-size rootfs. |
| 95 | |
| 96 | """ |
| 97 | if self.fixed_size: |
| 98 | rootfs_size = self.fixed_size |
| 99 | if actual_rootfs_size > rootfs_size: |
| 100 | raise WicError("Actual rootfs size (%d kB) is larger than " |
| 101 | "allowed size %d kB" % |
| 102 | (actual_rootfs_size, rootfs_size)) |
| 103 | else: |
| 104 | extra_blocks = self.get_extra_block_count(actual_rootfs_size) |
| 105 | if extra_blocks < self.extra_space: |
| 106 | extra_blocks = self.extra_space |
| 107 | |
| 108 | rootfs_size = actual_rootfs_size + extra_blocks |
Andrew Geissler | 5199d83 | 2021-09-24 16:47:35 -0500 | [diff] [blame] | 109 | rootfs_size = int(rootfs_size * self.overhead_factor) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 110 | |
| 111 | logger.debug("Added %d extra blocks to %s to get to %d total blocks", |
| 112 | extra_blocks, self.mountpoint, rootfs_size) |
| 113 | |
| 114 | return rootfs_size |
| 115 | |
| 116 | @property |
| 117 | def disk_size(self): |
| 118 | """ |
| 119 | Obtain on-disk size of partition taking into consideration |
| 120 | --size/--fixed-size options. |
| 121 | |
| 122 | """ |
| 123 | return self.fixed_size if self.fixed_size else self.size |
| 124 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 125 | def prepare(self, creator, cr_workdir, oe_builddir, rootfs_dir, |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 126 | bootimg_dir, kernel_dir, native_sysroot, updated_fstab_path): |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 127 | """ |
| 128 | Prepare content for individual partitions, depending on |
| 129 | partition command parameters. |
| 130 | """ |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 131 | self.updated_fstab_path = updated_fstab_path |
| 132 | if self.updated_fstab_path and not (self.fstype.startswith("ext") or self.fstype == "msdos"): |
| 133 | self.update_fstab_in_rootfs = True |
| 134 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 135 | if not self.source: |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 136 | if not self.size and not self.fixed_size: |
| 137 | raise WicError("The %s partition has a size of zero. Please " |
| 138 | "specify a non-zero --size/--fixed-size for that " |
| 139 | "partition." % self.mountpoint) |
| 140 | |
| 141 | if self.fstype == "swap": |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 142 | self.prepare_swap_partition(cr_workdir, oe_builddir, |
| 143 | native_sysroot) |
| 144 | self.source_file = "%s/fs.%s" % (cr_workdir, self.fstype) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 145 | else: |
William A. Kennington III | ac69b48 | 2021-06-02 12:28:27 -0700 | [diff] [blame] | 146 | if self.fstype in ('squashfs', 'erofs'): |
| 147 | raise WicError("It's not possible to create empty %s " |
| 148 | "partition '%s'" % (self.fstype, self.mountpoint)) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 149 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 150 | rootfs = "%s/fs_%s.%s.%s" % (cr_workdir, self.label, |
| 151 | self.lineno, self.fstype) |
| 152 | if os.path.isfile(rootfs): |
| 153 | os.remove(rootfs) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 154 | |
| 155 | prefix = "ext" if self.fstype.startswith("ext") else self.fstype |
| 156 | method = getattr(self, "prepare_empty_partition_" + prefix) |
| 157 | method(rootfs, oe_builddir, native_sysroot) |
| 158 | self.source_file = rootfs |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 159 | return |
| 160 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 161 | plugins = PluginMgr.get_plugins('source') |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 162 | |
| 163 | if self.source not in plugins: |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 164 | raise WicError("The '%s' --source specified for %s doesn't exist.\n\t" |
| 165 | "See 'wic list source-plugins' for a list of available" |
| 166 | " --sources.\n\tSee 'wic help source-plugins' for " |
| 167 | "details on adding a new source plugin." % |
| 168 | (self.source, self.mountpoint)) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 169 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 170 | srcparams_dict = {} |
| 171 | if self.sourceparams: |
| 172 | # Split sourceparams string of the form key1=val1[,key2=val2,...] |
| 173 | # into a dict. Also accepts valueless keys i.e. without = |
| 174 | splitted = self.sourceparams.split(',') |
Andrew Geissler | 7e0e3c0 | 2022-02-25 20:34:39 +0000 | [diff] [blame] | 175 | srcparams_dict = dict((par.split('=', 1) + [None])[:2] for par in splitted if par) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 176 | |
| 177 | plugin = PluginMgr.get_plugins('source')[self.source] |
| 178 | plugin.do_configure_partition(self, srcparams_dict, creator, |
| 179 | cr_workdir, oe_builddir, bootimg_dir, |
| 180 | kernel_dir, native_sysroot) |
| 181 | plugin.do_stage_partition(self, srcparams_dict, creator, |
| 182 | cr_workdir, oe_builddir, bootimg_dir, |
| 183 | kernel_dir, native_sysroot) |
| 184 | plugin.do_prepare_partition(self, srcparams_dict, creator, |
| 185 | cr_workdir, oe_builddir, bootimg_dir, |
| 186 | kernel_dir, rootfs_dir, native_sysroot) |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 187 | plugin.do_post_partition(self, srcparams_dict, creator, |
| 188 | cr_workdir, oe_builddir, bootimg_dir, |
| 189 | kernel_dir, rootfs_dir, native_sysroot) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 190 | |
Brad Bishop | 37a0e4d | 2017-12-04 01:01:44 -0500 | [diff] [blame] | 191 | # further processing required Partition.size to be an integer, make |
| 192 | # sure that it is one |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 193 | if not isinstance(self.size, int): |
| 194 | raise WicError("Partition %s internal size is not an integer. " |
| 195 | "This a bug in source plugin %s and needs to be fixed." % |
| 196 | (self.mountpoint, self.source)) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 197 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 198 | if self.fixed_size and self.size > self.fixed_size: |
| 199 | raise WicError("File system image of partition %s is " |
| 200 | "larger (%d kB) than its allowed size %d kB" % |
| 201 | (self.mountpoint, self.size, self.fixed_size)) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 202 | |
| 203 | def prepare_rootfs(self, cr_workdir, oe_builddir, rootfs_dir, |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 204 | native_sysroot, real_rootfs = True, pseudo_dir = None): |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 205 | """ |
| 206 | Prepare content for a rootfs partition i.e. create a partition |
| 207 | and fill it from a /rootfs dir. |
| 208 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 209 | Currently handles ext2/3/4, btrfs, vfat and squashfs. |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 210 | """ |
Andrew Geissler | f034379 | 2020-11-18 10:42:21 -0600 | [diff] [blame] | 211 | |
| 212 | rootfs = "%s/rootfs_%s.%s.%s" % (cr_workdir, self.label, |
| 213 | self.lineno, self.fstype) |
| 214 | if os.path.isfile(rootfs): |
| 215 | os.remove(rootfs) |
| 216 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 217 | p_prefix = os.environ.get("PSEUDO_PREFIX", "%s/usr" % native_sysroot) |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 218 | if (pseudo_dir): |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 219 | # Canonicalize the ignore paths. This corresponds to |
| 220 | # calling oe.path.canonicalize(), which is used in bitbake.conf. |
| 221 | ignore_paths = [rootfs] + (get_bitbake_var("PSEUDO_IGNORE_PATHS") or "").split(",") |
| 222 | canonical_paths = [] |
| 223 | for path in ignore_paths: |
| 224 | if "$" not in path: |
| 225 | trailing_slash = path.endswith("/") and "/" or "" |
| 226 | canonical_paths.append(os.path.realpath(path) + trailing_slash) |
| 227 | ignore_paths = ",".join(canonical_paths) |
| 228 | |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 229 | pseudo = "export PSEUDO_PREFIX=%s;" % p_prefix |
| 230 | pseudo += "export PSEUDO_LOCALSTATEDIR=%s;" % pseudo_dir |
| 231 | pseudo += "export PSEUDO_PASSWD=%s;" % rootfs_dir |
| 232 | pseudo += "export PSEUDO_NOSYMLINKEXP=1;" |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 233 | pseudo += "export PSEUDO_IGNORE_PATHS=%s;" % ignore_paths |
Andrew Geissler | 82c905d | 2020-04-13 13:39:40 -0500 | [diff] [blame] | 234 | pseudo += "%s " % get_bitbake_var("FAKEROOTCMD") |
| 235 | else: |
| 236 | pseudo = None |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 237 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 238 | if not self.size and real_rootfs: |
Brad Bishop | 00e122a | 2019-10-05 11:10:57 -0400 | [diff] [blame] | 239 | # The rootfs size is not set in .ks file so try to get it |
| 240 | # from bitbake variable |
| 241 | rsize_bb = get_bitbake_var('ROOTFS_SIZE') |
| 242 | rdir = get_bitbake_var('IMAGE_ROOTFS') |
| 243 | if rsize_bb and rdir == rootfs_dir: |
| 244 | # Bitbake variable ROOTFS_SIZE is calculated in |
| 245 | # Image._get_rootfs_size method from meta/lib/oe/image.py |
| 246 | # using IMAGE_ROOTFS_SIZE, IMAGE_ROOTFS_ALIGNMENT, |
| 247 | # IMAGE_OVERHEAD_FACTOR and IMAGE_ROOTFS_EXTRA_SPACE |
| 248 | self.size = int(round(float(rsize_bb))) |
| 249 | else: |
| 250 | # Bitbake variable ROOTFS_SIZE is not defined so compute it |
| 251 | # from the rootfs_dir size using the same logic found in |
| 252 | # get_rootfs_size() from meta/classes/image.bbclass |
| 253 | du_cmd = "du -ks %s" % rootfs_dir |
| 254 | out = exec_cmd(du_cmd) |
| 255 | self.size = int(out.split()[0]) |
Brad Bishop | 37a0e4d | 2017-12-04 01:01:44 -0500 | [diff] [blame] | 256 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 257 | prefix = "ext" if self.fstype.startswith("ext") else self.fstype |
| 258 | method = getattr(self, "prepare_rootfs_" + prefix) |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 259 | method(rootfs, cr_workdir, oe_builddir, rootfs_dir, native_sysroot, pseudo) |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 260 | self.source_file = rootfs |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 261 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 262 | # get the rootfs size in the right units for kickstart (kB) |
| 263 | du_cmd = "du -Lbks %s" % rootfs |
| 264 | out = exec_cmd(du_cmd) |
| 265 | self.size = int(out.split()[0]) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 266 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 267 | def prepare_rootfs_ext(self, rootfs, cr_workdir, oe_builddir, rootfs_dir, |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 268 | native_sysroot, pseudo): |
| 269 | """ |
| 270 | Prepare content for an ext2/3/4 rootfs partition. |
| 271 | """ |
| 272 | du_cmd = "du -ks %s" % rootfs_dir |
| 273 | out = exec_cmd(du_cmd) |
| 274 | actual_rootfs_size = int(out.split()[0]) |
| 275 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 276 | rootfs_size = self.get_rootfs_size(actual_rootfs_size) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 277 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 278 | with open(rootfs, 'w') as sparse: |
| 279 | os.ftruncate(sparse.fileno(), rootfs_size * 1024) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 280 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 281 | extraopts = self.mkfs_extraopts or "-F -i 8192" |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 282 | |
| 283 | label_str = "" |
| 284 | if self.label: |
| 285 | label_str = "-L %s" % self.label |
| 286 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 287 | mkfs_cmd = "mkfs.%s %s %s %s -U %s -d %s" % \ |
| 288 | (self.fstype, extraopts, rootfs, label_str, self.fsuuid, rootfs_dir) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 289 | exec_native_cmd(mkfs_cmd, native_sysroot, pseudo=pseudo) |
| 290 | |
Andrew Geissler | d159c7f | 2021-09-02 21:05:58 -0500 | [diff] [blame] | 291 | if self.updated_fstab_path and self.has_fstab and not self.no_fstab_update: |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 292 | debugfs_script_path = os.path.join(cr_workdir, "debugfs_script") |
| 293 | with open(debugfs_script_path, "w") as f: |
| 294 | f.write("cd etc\n") |
| 295 | f.write("rm fstab\n") |
| 296 | f.write("write %s fstab\n" % (self.updated_fstab_path)) |
| 297 | debugfs_cmd = "debugfs -w -f %s %s" % (debugfs_script_path, rootfs) |
| 298 | exec_native_cmd(debugfs_cmd, native_sysroot) |
| 299 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 300 | mkfs_cmd = "fsck.%s -pvfD %s" % (self.fstype, rootfs) |
Brad Bishop | 37a0e4d | 2017-12-04 01:01:44 -0500 | [diff] [blame] | 301 | exec_native_cmd(mkfs_cmd, native_sysroot, pseudo=pseudo) |
| 302 | |
Patrick Williams | 7784c42 | 2022-11-17 07:29:11 -0600 | [diff] [blame] | 303 | if os.getenv('SOURCE_DATE_EPOCH'): |
| 304 | sde_time = hex(int(os.getenv('SOURCE_DATE_EPOCH'))) |
| 305 | debugfs_script_path = os.path.join(cr_workdir, "debugfs_script") |
| 306 | files = [] |
| 307 | for root, dirs, others in os.walk(rootfs_dir): |
| 308 | base = root.replace(rootfs_dir, "").rstrip(os.sep) |
| 309 | files += [ "/" if base == "" else base ] |
| 310 | files += [ base + "/" + n for n in dirs + others ] |
| 311 | with open(debugfs_script_path, "w") as f: |
| 312 | f.write("set_current_time %s\n" % (sde_time)) |
| 313 | if self.updated_fstab_path and self.has_fstab and not self.no_fstab_update: |
| 314 | f.write("set_inode_field /etc/fstab mtime %s\n" % (sde_time)) |
| 315 | f.write("set_inode_field /etc/fstab mtime_extra 0\n") |
| 316 | for file in set(files): |
| 317 | for time in ["atime", "ctime", "crtime"]: |
| 318 | f.write("set_inode_field \"%s\" %s %s\n" % (file, time, sde_time)) |
| 319 | f.write("set_inode_field \"%s\" %s_extra 0\n" % (file, time)) |
| 320 | for time in ["wtime", "mkfs_time", "lastcheck"]: |
| 321 | f.write("set_super_value %s %s\n" % (time, sde_time)) |
| 322 | for time in ["mtime", "first_error_time", "last_error_time"]: |
| 323 | f.write("set_super_value %s 0\n" % (time)) |
| 324 | debugfs_cmd = "debugfs -w -f %s %s" % (debugfs_script_path, rootfs) |
| 325 | exec_native_cmd(debugfs_cmd, native_sysroot) |
| 326 | |
Andrew Geissler | 90fd73c | 2021-03-05 15:25:55 -0600 | [diff] [blame] | 327 | self.check_for_Y2038_problem(rootfs, native_sysroot) |
| 328 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 329 | def prepare_rootfs_btrfs(self, rootfs, cr_workdir, oe_builddir, rootfs_dir, |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 330 | native_sysroot, pseudo): |
| 331 | """ |
| 332 | Prepare content for a btrfs rootfs partition. |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 333 | """ |
| 334 | du_cmd = "du -ks %s" % rootfs_dir |
| 335 | out = exec_cmd(du_cmd) |
| 336 | actual_rootfs_size = int(out.split()[0]) |
| 337 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 338 | rootfs_size = self.get_rootfs_size(actual_rootfs_size) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 339 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 340 | with open(rootfs, 'w') as sparse: |
| 341 | os.ftruncate(sparse.fileno(), rootfs_size * 1024) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 342 | |
| 343 | label_str = "" |
| 344 | if self.label: |
| 345 | label_str = "-L %s" % self.label |
| 346 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 347 | mkfs_cmd = "mkfs.%s -b %d -r %s %s %s -U %s %s" % \ |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 348 | (self.fstype, rootfs_size * 1024, rootfs_dir, label_str, |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 349 | self.mkfs_extraopts, self.fsuuid, rootfs) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 350 | exec_native_cmd(mkfs_cmd, native_sysroot, pseudo=pseudo) |
| 351 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 352 | def prepare_rootfs_msdos(self, rootfs, cr_workdir, oe_builddir, rootfs_dir, |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 353 | native_sysroot, pseudo): |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 354 | """ |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 355 | Prepare content for a msdos/vfat rootfs partition. |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 356 | """ |
| 357 | du_cmd = "du -bks %s" % rootfs_dir |
| 358 | out = exec_cmd(du_cmd) |
| 359 | blocks = int(out.split()[0]) |
| 360 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 361 | rootfs_size = self.get_rootfs_size(blocks) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 362 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 363 | label_str = "-n boot" |
| 364 | if self.label: |
| 365 | label_str = "-n %s" % self.label |
| 366 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 367 | size_str = "" |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 368 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 369 | extraopts = self.mkfs_extraopts or '-S 512' |
| 370 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 371 | dosfs_cmd = "mkdosfs %s -i %s %s %s -C %s %d" % \ |
| 372 | (label_str, self.fsuuid, size_str, extraopts, rootfs, |
Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 373 | rootfs_size) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 374 | exec_native_cmd(dosfs_cmd, native_sysroot) |
| 375 | |
| 376 | mcopy_cmd = "mcopy -i %s -s %s/* ::/" % (rootfs, rootfs_dir) |
| 377 | exec_native_cmd(mcopy_cmd, native_sysroot) |
| 378 | |
Andrew Geissler | d159c7f | 2021-09-02 21:05:58 -0500 | [diff] [blame] | 379 | if self.updated_fstab_path and self.has_fstab and not self.no_fstab_update: |
Patrick Williams | 2390b1b | 2022-11-03 13:47:49 -0500 | [diff] [blame] | 380 | mcopy_cmd = "mcopy -m -i %s %s ::/etc/fstab" % (rootfs, self.updated_fstab_path) |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 381 | exec_native_cmd(mcopy_cmd, native_sysroot) |
| 382 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 383 | chmod_cmd = "chmod 644 %s" % rootfs |
| 384 | exec_cmd(chmod_cmd) |
| 385 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 386 | prepare_rootfs_vfat = prepare_rootfs_msdos |
| 387 | |
Andrew Geissler | d1e8949 | 2021-02-12 15:35:20 -0600 | [diff] [blame] | 388 | def prepare_rootfs_squashfs(self, rootfs, cr_workdir, oe_builddir, rootfs_dir, |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 389 | native_sysroot, pseudo): |
| 390 | """ |
| 391 | Prepare content for a squashfs rootfs partition. |
| 392 | """ |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 393 | extraopts = self.mkfs_extraopts or '-noappend' |
| 394 | squashfs_cmd = "mksquashfs %s %s %s" % \ |
| 395 | (rootfs_dir, rootfs, extraopts) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 396 | exec_native_cmd(squashfs_cmd, native_sysroot, pseudo=pseudo) |
| 397 | |
William A. Kennington III | ac69b48 | 2021-06-02 12:28:27 -0700 | [diff] [blame] | 398 | def prepare_rootfs_erofs(self, rootfs, cr_workdir, oe_builddir, rootfs_dir, |
| 399 | native_sysroot, pseudo): |
| 400 | """ |
| 401 | Prepare content for a erofs rootfs partition. |
| 402 | """ |
| 403 | extraopts = self.mkfs_extraopts or '' |
| 404 | erofs_cmd = "mkfs.erofs %s -U %s %s %s" % \ |
| 405 | (extraopts, self.fsuuid, rootfs, rootfs_dir) |
| 406 | exec_native_cmd(erofs_cmd, native_sysroot, pseudo=pseudo) |
| 407 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 408 | def prepare_empty_partition_ext(self, rootfs, oe_builddir, |
| 409 | native_sysroot): |
| 410 | """ |
| 411 | Prepare an empty ext2/3/4 partition. |
| 412 | """ |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 413 | size = self.disk_size |
| 414 | with open(rootfs, 'w') as sparse: |
| 415 | os.ftruncate(sparse.fileno(), size * 1024) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 416 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 417 | extraopts = self.mkfs_extraopts or "-i 8192" |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 418 | |
| 419 | label_str = "" |
| 420 | if self.label: |
| 421 | label_str = "-L %s" % self.label |
| 422 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 423 | mkfs_cmd = "mkfs.%s -F %s %s -U %s %s" % \ |
| 424 | (self.fstype, extraopts, label_str, self.fsuuid, rootfs) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 425 | exec_native_cmd(mkfs_cmd, native_sysroot) |
| 426 | |
Andrew Geissler | 90fd73c | 2021-03-05 15:25:55 -0600 | [diff] [blame] | 427 | self.check_for_Y2038_problem(rootfs, native_sysroot) |
| 428 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 429 | def prepare_empty_partition_btrfs(self, rootfs, oe_builddir, |
| 430 | native_sysroot): |
| 431 | """ |
| 432 | Prepare an empty btrfs partition. |
| 433 | """ |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 434 | size = self.disk_size |
| 435 | with open(rootfs, 'w') as sparse: |
| 436 | os.ftruncate(sparse.fileno(), size * 1024) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 437 | |
| 438 | label_str = "" |
| 439 | if self.label: |
| 440 | label_str = "-L %s" % self.label |
| 441 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 442 | mkfs_cmd = "mkfs.%s -b %d %s -U %s %s %s" % \ |
| 443 | (self.fstype, self.size * 1024, label_str, self.fsuuid, |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 444 | self.mkfs_extraopts, rootfs) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 445 | exec_native_cmd(mkfs_cmd, native_sysroot) |
| 446 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 447 | def prepare_empty_partition_msdos(self, rootfs, oe_builddir, |
| 448 | native_sysroot): |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 449 | """ |
| 450 | Prepare an empty vfat partition. |
| 451 | """ |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 452 | blocks = self.disk_size |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 453 | |
| 454 | label_str = "-n boot" |
| 455 | if self.label: |
| 456 | label_str = "-n %s" % self.label |
| 457 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 458 | size_str = "" |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 459 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 460 | extraopts = self.mkfs_extraopts or '-S 512' |
| 461 | |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 462 | dosfs_cmd = "mkdosfs %s -i %s %s %s -C %s %d" % \ |
| 463 | (label_str, self.fsuuid, extraopts, size_str, rootfs, |
| 464 | blocks) |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 465 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 466 | exec_native_cmd(dosfs_cmd, native_sysroot) |
| 467 | |
| 468 | chmod_cmd = "chmod 644 %s" % rootfs |
| 469 | exec_cmd(chmod_cmd) |
| 470 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 471 | prepare_empty_partition_vfat = prepare_empty_partition_msdos |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 472 | |
| 473 | def prepare_swap_partition(self, cr_workdir, oe_builddir, native_sysroot): |
| 474 | """ |
| 475 | Prepare a swap partition. |
| 476 | """ |
| 477 | path = "%s/fs.%s" % (cr_workdir, self.fstype) |
| 478 | |
Brad Bishop | 6e60e8b | 2018-02-01 10:27:11 -0500 | [diff] [blame] | 479 | with open(path, 'w') as sparse: |
| 480 | os.ftruncate(sparse.fileno(), self.size * 1024) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 481 | |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 482 | label_str = "" |
| 483 | if self.label: |
| 484 | label_str = "-L %s" % self.label |
Brad Bishop | 316dfdd | 2018-06-25 12:45:53 -0400 | [diff] [blame] | 485 | |
| 486 | mkswap_cmd = "mkswap %s -U %s %s" % (label_str, self.fsuuid, path) |
Patrick Williams | d8c66bc | 2016-06-20 12:57:21 -0500 | [diff] [blame] | 487 | exec_native_cmd(mkswap_cmd, native_sysroot) |
Andrew Geissler | 90fd73c | 2021-03-05 15:25:55 -0600 | [diff] [blame] | 488 | |
| 489 | def check_for_Y2038_problem(self, rootfs, native_sysroot): |
| 490 | """ |
| 491 | Check if the filesystem is affected by the Y2038 problem |
| 492 | (Y2038 problem = 32 bit time_t overflow in January 2038) |
| 493 | """ |
| 494 | def get_err_str(part): |
| 495 | err = "The {} filesystem {} has no Y2038 support." |
| 496 | if part.mountpoint: |
| 497 | args = [part.fstype, "mounted at %s" % part.mountpoint] |
| 498 | elif part.label: |
| 499 | args = [part.fstype, "labeled '%s'" % part.label] |
| 500 | elif part.part_name: |
| 501 | args = [part.fstype, "in partition '%s'" % part.part_name] |
| 502 | else: |
| 503 | args = [part.fstype, "in partition %s" % part.num] |
| 504 | return err.format(*args) |
| 505 | |
| 506 | # ext2 and ext3 are always affected by the Y2038 problem |
| 507 | if self.fstype in ["ext2", "ext3"]: |
| 508 | logger.warn(get_err_str(self)) |
| 509 | return |
| 510 | |
| 511 | ret, out = exec_native_cmd("dumpe2fs %s" % rootfs, native_sysroot) |
| 512 | |
| 513 | # if ext4 is affected by the Y2038 problem depends on the inode size |
| 514 | for line in out.splitlines(): |
| 515 | if line.startswith("Inode size:"): |
| 516 | size = int(line.split(":")[1].strip()) |
| 517 | if size < 256: |
| 518 | logger.warn("%s Inodes (of size %d) are too small." % |
| 519 | (get_err_str(self), size)) |
| 520 | break |
| 521 | |