xref: /OK3568_Linux_fs/yocto/scripts/lib/wic/engine.py (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun#
2*4882a593Smuzhiyun# Copyright (c) 2013, Intel Corporation.
3*4882a593Smuzhiyun#
4*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0-only
5*4882a593Smuzhiyun#
6*4882a593Smuzhiyun# DESCRIPTION
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun# This module implements the image creation engine used by 'wic' to
9*4882a593Smuzhiyun# create images.  The engine parses through the OpenEmbedded kickstart
10*4882a593Smuzhiyun# (wks) file specified and generates images that can then be directly
11*4882a593Smuzhiyun# written onto media.
12*4882a593Smuzhiyun#
13*4882a593Smuzhiyun# AUTHORS
14*4882a593Smuzhiyun# Tom Zanussi <tom.zanussi (at] linux.intel.com>
15*4882a593Smuzhiyun#
16*4882a593Smuzhiyun
17*4882a593Smuzhiyunimport logging
18*4882a593Smuzhiyunimport os
19*4882a593Smuzhiyunimport tempfile
20*4882a593Smuzhiyunimport json
21*4882a593Smuzhiyunimport subprocess
22*4882a593Smuzhiyunimport shutil
23*4882a593Smuzhiyunimport re
24*4882a593Smuzhiyun
25*4882a593Smuzhiyunfrom collections import namedtuple, OrderedDict
26*4882a593Smuzhiyun
27*4882a593Smuzhiyunfrom wic import WicError
28*4882a593Smuzhiyunfrom wic.filemap import sparse_copy
29*4882a593Smuzhiyunfrom wic.pluginbase import PluginMgr
30*4882a593Smuzhiyunfrom wic.misc import get_bitbake_var, exec_cmd
31*4882a593Smuzhiyun
32*4882a593Smuzhiyunlogger = logging.getLogger('wic')
33*4882a593Smuzhiyun
34*4882a593Smuzhiyundef verify_build_env():
35*4882a593Smuzhiyun    """
36*4882a593Smuzhiyun    Verify that the build environment is sane.
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun    Returns True if it is, false otherwise
39*4882a593Smuzhiyun    """
40*4882a593Smuzhiyun    if not os.environ.get("BUILDDIR"):
41*4882a593Smuzhiyun        raise WicError("BUILDDIR not found, exiting. (Did you forget to source oe-init-build-env?)")
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun    return True
44*4882a593Smuzhiyun
45*4882a593Smuzhiyun
46*4882a593SmuzhiyunCANNED_IMAGE_DIR = "lib/wic/canned-wks" # relative to scripts
47*4882a593SmuzhiyunSCRIPTS_CANNED_IMAGE_DIR = "scripts/" + CANNED_IMAGE_DIR
48*4882a593SmuzhiyunWIC_DIR = "wic"
49*4882a593Smuzhiyun
50*4882a593Smuzhiyundef build_canned_image_list(path):
51*4882a593Smuzhiyun    layers_path = get_bitbake_var("BBLAYERS")
52*4882a593Smuzhiyun    canned_wks_layer_dirs = []
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun    if layers_path is not None:
55*4882a593Smuzhiyun        for layer_path in layers_path.split():
56*4882a593Smuzhiyun            for wks_path in (WIC_DIR, SCRIPTS_CANNED_IMAGE_DIR):
57*4882a593Smuzhiyun                cpath = os.path.join(layer_path, wks_path)
58*4882a593Smuzhiyun                if os.path.isdir(cpath):
59*4882a593Smuzhiyun                    canned_wks_layer_dirs.append(cpath)
60*4882a593Smuzhiyun
61*4882a593Smuzhiyun    cpath = os.path.join(path, CANNED_IMAGE_DIR)
62*4882a593Smuzhiyun    canned_wks_layer_dirs.append(cpath)
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun    return canned_wks_layer_dirs
65*4882a593Smuzhiyun
66*4882a593Smuzhiyundef find_canned_image(scripts_path, wks_file):
67*4882a593Smuzhiyun    """
68*4882a593Smuzhiyun    Find a .wks file with the given name in the canned files dir.
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun    Return False if not found
71*4882a593Smuzhiyun    """
72*4882a593Smuzhiyun    layers_canned_wks_dir = build_canned_image_list(scripts_path)
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun    for canned_wks_dir in layers_canned_wks_dir:
75*4882a593Smuzhiyun        for root, dirs, files in os.walk(canned_wks_dir):
76*4882a593Smuzhiyun            for fname in files:
77*4882a593Smuzhiyun                if fname.endswith("~") or fname.endswith("#"):
78*4882a593Smuzhiyun                    continue
79*4882a593Smuzhiyun                if ((fname.endswith(".wks") and wks_file + ".wks" == fname) or \
80*4882a593Smuzhiyun                   (fname.endswith(".wks.in") and wks_file + ".wks.in" == fname)):
81*4882a593Smuzhiyun                    fullpath = os.path.join(canned_wks_dir, fname)
82*4882a593Smuzhiyun                    return fullpath
83*4882a593Smuzhiyun    return None
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun
86*4882a593Smuzhiyundef list_canned_images(scripts_path):
87*4882a593Smuzhiyun    """
88*4882a593Smuzhiyun    List the .wks files in the canned image dir, minus the extension.
89*4882a593Smuzhiyun    """
90*4882a593Smuzhiyun    layers_canned_wks_dir = build_canned_image_list(scripts_path)
91*4882a593Smuzhiyun
92*4882a593Smuzhiyun    for canned_wks_dir in layers_canned_wks_dir:
93*4882a593Smuzhiyun        for root, dirs, files in os.walk(canned_wks_dir):
94*4882a593Smuzhiyun            for fname in files:
95*4882a593Smuzhiyun                if fname.endswith("~") or fname.endswith("#"):
96*4882a593Smuzhiyun                    continue
97*4882a593Smuzhiyun                if fname.endswith(".wks") or fname.endswith(".wks.in"):
98*4882a593Smuzhiyun                    fullpath = os.path.join(canned_wks_dir, fname)
99*4882a593Smuzhiyun                    with open(fullpath) as wks:
100*4882a593Smuzhiyun                        for line in wks:
101*4882a593Smuzhiyun                            desc = ""
102*4882a593Smuzhiyun                            idx = line.find("short-description:")
103*4882a593Smuzhiyun                            if idx != -1:
104*4882a593Smuzhiyun                                desc = line[idx + len("short-description:"):].strip()
105*4882a593Smuzhiyun                                break
106*4882a593Smuzhiyun                    basename = fname.split('.')[0]
107*4882a593Smuzhiyun                    print("  %s\t\t%s" % (basename.ljust(30), desc))
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun
110*4882a593Smuzhiyundef list_canned_image_help(scripts_path, fullpath):
111*4882a593Smuzhiyun    """
112*4882a593Smuzhiyun    List the help and params in the specified canned image.
113*4882a593Smuzhiyun    """
114*4882a593Smuzhiyun    found = False
115*4882a593Smuzhiyun    with open(fullpath) as wks:
116*4882a593Smuzhiyun        for line in wks:
117*4882a593Smuzhiyun            if not found:
118*4882a593Smuzhiyun                idx = line.find("long-description:")
119*4882a593Smuzhiyun                if idx != -1:
120*4882a593Smuzhiyun                    print()
121*4882a593Smuzhiyun                    print(line[idx + len("long-description:"):].strip())
122*4882a593Smuzhiyun                    found = True
123*4882a593Smuzhiyun                continue
124*4882a593Smuzhiyun            if not line.strip():
125*4882a593Smuzhiyun                break
126*4882a593Smuzhiyun            idx = line.find("#")
127*4882a593Smuzhiyun            if idx != -1:
128*4882a593Smuzhiyun                print(line[idx + len("#:"):].rstrip())
129*4882a593Smuzhiyun            else:
130*4882a593Smuzhiyun                break
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun
133*4882a593Smuzhiyundef list_source_plugins():
134*4882a593Smuzhiyun    """
135*4882a593Smuzhiyun    List the available source plugins i.e. plugins available for --source.
136*4882a593Smuzhiyun    """
137*4882a593Smuzhiyun    plugins = PluginMgr.get_plugins('source')
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun    for plugin in plugins:
140*4882a593Smuzhiyun        print("  %s" % plugin)
141*4882a593Smuzhiyun
142*4882a593Smuzhiyun
143*4882a593Smuzhiyundef wic_create(wks_file, rootfs_dir, bootimg_dir, kernel_dir,
144*4882a593Smuzhiyun               native_sysroot, options):
145*4882a593Smuzhiyun    """
146*4882a593Smuzhiyun    Create image
147*4882a593Smuzhiyun
148*4882a593Smuzhiyun    wks_file - user-defined OE kickstart file
149*4882a593Smuzhiyun    rootfs_dir - absolute path to the build's /rootfs dir
150*4882a593Smuzhiyun    bootimg_dir - absolute path to the build's boot artifacts directory
151*4882a593Smuzhiyun    kernel_dir - absolute path to the build's kernel directory
152*4882a593Smuzhiyun    native_sysroot - absolute path to the build's native sysroots dir
153*4882a593Smuzhiyun    image_output_dir - dirname to create for image
154*4882a593Smuzhiyun    options - wic command line options (debug, bmap, etc)
155*4882a593Smuzhiyun
156*4882a593Smuzhiyun    Normally, the values for the build artifacts values are determined
157*4882a593Smuzhiyun    by 'wic -e' from the output of the 'bitbake -e' command given an
158*4882a593Smuzhiyun    image name e.g. 'core-image-minimal' and a given machine set in
159*4882a593Smuzhiyun    local.conf.  If that's the case, the variables get the following
160*4882a593Smuzhiyun    values from the output of 'bitbake -e':
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun    rootfs_dir:        IMAGE_ROOTFS
163*4882a593Smuzhiyun    kernel_dir:        DEPLOY_DIR_IMAGE
164*4882a593Smuzhiyun    native_sysroot:    STAGING_DIR_NATIVE
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun    In the above case, bootimg_dir remains unset and the
167*4882a593Smuzhiyun    plugin-specific image creation code is responsible for finding the
168*4882a593Smuzhiyun    bootimg artifacts.
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun    In the case where the values are passed in explicitly i.e 'wic -e'
171*4882a593Smuzhiyun    is not used but rather the individual 'wic' options are used to
172*4882a593Smuzhiyun    explicitly specify these values.
173*4882a593Smuzhiyun    """
174*4882a593Smuzhiyun    try:
175*4882a593Smuzhiyun        oe_builddir = os.environ["BUILDDIR"]
176*4882a593Smuzhiyun    except KeyError:
177*4882a593Smuzhiyun        raise WicError("BUILDDIR not found, exiting. (Did you forget to source oe-init-build-env?)")
178*4882a593Smuzhiyun
179*4882a593Smuzhiyun    if not os.path.exists(options.outdir):
180*4882a593Smuzhiyun        os.makedirs(options.outdir)
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun    pname = options.imager
183*4882a593Smuzhiyun    plugin_class = PluginMgr.get_plugins('imager').get(pname)
184*4882a593Smuzhiyun    if not plugin_class:
185*4882a593Smuzhiyun        raise WicError('Unknown plugin: %s' % pname)
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun    plugin = plugin_class(wks_file, rootfs_dir, bootimg_dir, kernel_dir,
188*4882a593Smuzhiyun                          native_sysroot, oe_builddir, options)
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun    plugin.do_create()
191*4882a593Smuzhiyun
192*4882a593Smuzhiyun    logger.info("The image(s) were created using OE kickstart file:\n  %s", wks_file)
193*4882a593Smuzhiyun
194*4882a593Smuzhiyun
195*4882a593Smuzhiyundef wic_list(args, scripts_path):
196*4882a593Smuzhiyun    """
197*4882a593Smuzhiyun    Print the list of images or source plugins.
198*4882a593Smuzhiyun    """
199*4882a593Smuzhiyun    if args.list_type is None:
200*4882a593Smuzhiyun        return False
201*4882a593Smuzhiyun
202*4882a593Smuzhiyun    if args.list_type == "images":
203*4882a593Smuzhiyun
204*4882a593Smuzhiyun        list_canned_images(scripts_path)
205*4882a593Smuzhiyun        return True
206*4882a593Smuzhiyun    elif args.list_type == "source-plugins":
207*4882a593Smuzhiyun        list_source_plugins()
208*4882a593Smuzhiyun        return True
209*4882a593Smuzhiyun    elif len(args.help_for) == 1 and args.help_for[0] == 'help':
210*4882a593Smuzhiyun        wks_file = args.list_type
211*4882a593Smuzhiyun        fullpath = find_canned_image(scripts_path, wks_file)
212*4882a593Smuzhiyun        if not fullpath:
213*4882a593Smuzhiyun            raise WicError("No image named %s found, exiting. "
214*4882a593Smuzhiyun                           "(Use 'wic list images' to list available images, "
215*4882a593Smuzhiyun                           "or specify a fully-qualified OE kickstart (.wks) "
216*4882a593Smuzhiyun                           "filename)" % wks_file)
217*4882a593Smuzhiyun
218*4882a593Smuzhiyun        list_canned_image_help(scripts_path, fullpath)
219*4882a593Smuzhiyun        return True
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun    return False
222*4882a593Smuzhiyun
223*4882a593Smuzhiyun
224*4882a593Smuzhiyunclass Disk:
225*4882a593Smuzhiyun    def __init__(self, imagepath, native_sysroot, fstypes=('fat', 'ext')):
226*4882a593Smuzhiyun        self.imagepath = imagepath
227*4882a593Smuzhiyun        self.native_sysroot = native_sysroot
228*4882a593Smuzhiyun        self.fstypes = fstypes
229*4882a593Smuzhiyun        self._partitions = None
230*4882a593Smuzhiyun        self._partimages = {}
231*4882a593Smuzhiyun        self._lsector_size = None
232*4882a593Smuzhiyun        self._psector_size = None
233*4882a593Smuzhiyun        self._ptable_format = None
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun        # find parted
236*4882a593Smuzhiyun        # read paths from $PATH environment variable
237*4882a593Smuzhiyun        # if it fails, use hardcoded paths
238*4882a593Smuzhiyun        pathlist = "/bin:/usr/bin:/usr/sbin:/sbin/"
239*4882a593Smuzhiyun        try:
240*4882a593Smuzhiyun            self.paths = os.environ['PATH'] + ":" + pathlist
241*4882a593Smuzhiyun        except KeyError:
242*4882a593Smuzhiyun            self.paths = pathlist
243*4882a593Smuzhiyun
244*4882a593Smuzhiyun        if native_sysroot:
245*4882a593Smuzhiyun            for path in pathlist.split(':'):
246*4882a593Smuzhiyun                self.paths = "%s%s:%s" % (native_sysroot, path, self.paths)
247*4882a593Smuzhiyun
248*4882a593Smuzhiyun        self.parted = shutil.which("parted", path=self.paths)
249*4882a593Smuzhiyun        if not self.parted:
250*4882a593Smuzhiyun            raise WicError("Can't find executable parted")
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun        self.partitions = self.get_partitions()
253*4882a593Smuzhiyun
254*4882a593Smuzhiyun    def __del__(self):
255*4882a593Smuzhiyun        for path in self._partimages.values():
256*4882a593Smuzhiyun            os.unlink(path)
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun    def get_partitions(self):
259*4882a593Smuzhiyun        if self._partitions is None:
260*4882a593Smuzhiyun            self._partitions = OrderedDict()
261*4882a593Smuzhiyun            out = exec_cmd("%s -sm %s unit B print" % (self.parted, self.imagepath))
262*4882a593Smuzhiyun            parttype = namedtuple("Part", "pnum start end size fstype")
263*4882a593Smuzhiyun            splitted = out.splitlines()
264*4882a593Smuzhiyun            # skip over possible errors in exec_cmd output
265*4882a593Smuzhiyun            try:
266*4882a593Smuzhiyun                idx =splitted.index("BYT;")
267*4882a593Smuzhiyun            except ValueError:
268*4882a593Smuzhiyun                raise WicError("Error getting partition information from %s" % (self.parted))
269*4882a593Smuzhiyun            lsector_size, psector_size, self._ptable_format = splitted[idx + 1].split(":")[3:6]
270*4882a593Smuzhiyun            self._lsector_size = int(lsector_size)
271*4882a593Smuzhiyun            self._psector_size = int(psector_size)
272*4882a593Smuzhiyun            for line in splitted[idx + 2:]:
273*4882a593Smuzhiyun                pnum, start, end, size, fstype = line.split(':')[:5]
274*4882a593Smuzhiyun                partition = parttype(int(pnum), int(start[:-1]), int(end[:-1]),
275*4882a593Smuzhiyun                                     int(size[:-1]), fstype)
276*4882a593Smuzhiyun                self._partitions[pnum] = partition
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun        return self._partitions
279*4882a593Smuzhiyun
280*4882a593Smuzhiyun    def __getattr__(self, name):
281*4882a593Smuzhiyun        """Get path to the executable in a lazy way."""
282*4882a593Smuzhiyun        if name in ("mdir", "mcopy", "mdel", "mdeltree", "sfdisk", "e2fsck",
283*4882a593Smuzhiyun                    "resize2fs", "mkswap", "mkdosfs", "debugfs","blkid"):
284*4882a593Smuzhiyun            aname = "_%s" % name
285*4882a593Smuzhiyun            if aname not in self.__dict__:
286*4882a593Smuzhiyun                setattr(self, aname, shutil.which(name, path=self.paths))
287*4882a593Smuzhiyun                if aname not in self.__dict__ or self.__dict__[aname] is None:
288*4882a593Smuzhiyun                    raise WicError("Can't find executable '{}'".format(name))
289*4882a593Smuzhiyun            return self.__dict__[aname]
290*4882a593Smuzhiyun        return self.__dict__[name]
291*4882a593Smuzhiyun
292*4882a593Smuzhiyun    def _get_part_image(self, pnum):
293*4882a593Smuzhiyun        if pnum not in self.partitions:
294*4882a593Smuzhiyun            raise WicError("Partition %s is not in the image" % pnum)
295*4882a593Smuzhiyun        part = self.partitions[pnum]
296*4882a593Smuzhiyun        # check if fstype is supported
297*4882a593Smuzhiyun        for fstype in self.fstypes:
298*4882a593Smuzhiyun            if part.fstype.startswith(fstype):
299*4882a593Smuzhiyun                break
300*4882a593Smuzhiyun        else:
301*4882a593Smuzhiyun            raise WicError("Not supported fstype: {}".format(part.fstype))
302*4882a593Smuzhiyun        if pnum not in self._partimages:
303*4882a593Smuzhiyun            tmpf = tempfile.NamedTemporaryFile(prefix="wic-part")
304*4882a593Smuzhiyun            dst_fname = tmpf.name
305*4882a593Smuzhiyun            tmpf.close()
306*4882a593Smuzhiyun            sparse_copy(self.imagepath, dst_fname, skip=part.start, length=part.size)
307*4882a593Smuzhiyun            self._partimages[pnum] = dst_fname
308*4882a593Smuzhiyun
309*4882a593Smuzhiyun        return self._partimages[pnum]
310*4882a593Smuzhiyun
311*4882a593Smuzhiyun    def _put_part_image(self, pnum):
312*4882a593Smuzhiyun        """Put partition image into partitioned image."""
313*4882a593Smuzhiyun        sparse_copy(self._partimages[pnum], self.imagepath,
314*4882a593Smuzhiyun                    seek=self.partitions[pnum].start)
315*4882a593Smuzhiyun
316*4882a593Smuzhiyun    def dir(self, pnum, path):
317*4882a593Smuzhiyun        if pnum not in self.partitions:
318*4882a593Smuzhiyun            raise WicError("Partition %s is not in the image" % pnum)
319*4882a593Smuzhiyun
320*4882a593Smuzhiyun        if self.partitions[pnum].fstype.startswith('ext'):
321*4882a593Smuzhiyun            return exec_cmd("{} {} -R 'ls -l {}'".format(self.debugfs,
322*4882a593Smuzhiyun                                                         self._get_part_image(pnum),
323*4882a593Smuzhiyun                                                         path), as_shell=True)
324*4882a593Smuzhiyun        else: # fat
325*4882a593Smuzhiyun            return exec_cmd("{} -i {} ::{}".format(self.mdir,
326*4882a593Smuzhiyun                                                   self._get_part_image(pnum),
327*4882a593Smuzhiyun                                                   path))
328*4882a593Smuzhiyun
329*4882a593Smuzhiyun    def copy(self, src, dest):
330*4882a593Smuzhiyun        """Copy partition image into wic image."""
331*4882a593Smuzhiyun        pnum =  dest.part if isinstance(src, str) else src.part
332*4882a593Smuzhiyun
333*4882a593Smuzhiyun        if self.partitions[pnum].fstype.startswith('ext'):
334*4882a593Smuzhiyun            if isinstance(src, str):
335*4882a593Smuzhiyun                cmd = "printf 'cd {}\nwrite {} {}\n' | {} -w {}".\
336*4882a593Smuzhiyun                      format(os.path.dirname(dest.path), src, os.path.basename(src),
337*4882a593Smuzhiyun                             self.debugfs, self._get_part_image(pnum))
338*4882a593Smuzhiyun            else: # copy from wic
339*4882a593Smuzhiyun                # run both dump and rdump to support both files and directory
340*4882a593Smuzhiyun                cmd = "printf 'cd {}\ndump /{} {}\nrdump /{} {}\n' | {} {}".\
341*4882a593Smuzhiyun                      format(os.path.dirname(src.path), src.path,
342*4882a593Smuzhiyun                             dest, src.path, dest, self.debugfs,
343*4882a593Smuzhiyun                             self._get_part_image(pnum))
344*4882a593Smuzhiyun        else: # fat
345*4882a593Smuzhiyun            if isinstance(src, str):
346*4882a593Smuzhiyun                cmd = "{} -i {} -snop {} ::{}".format(self.mcopy,
347*4882a593Smuzhiyun                                                  self._get_part_image(pnum),
348*4882a593Smuzhiyun                                                  src, dest.path)
349*4882a593Smuzhiyun            else:
350*4882a593Smuzhiyun                cmd = "{} -i {} -snop ::{} {}".format(self.mcopy,
351*4882a593Smuzhiyun                                                  self._get_part_image(pnum),
352*4882a593Smuzhiyun                                                  src.path, dest)
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun        exec_cmd(cmd, as_shell=True)
355*4882a593Smuzhiyun        self._put_part_image(pnum)
356*4882a593Smuzhiyun
357*4882a593Smuzhiyun    def remove_ext(self, pnum, path, recursive):
358*4882a593Smuzhiyun        """
359*4882a593Smuzhiyun        Remove files/dirs and their contents from the partition.
360*4882a593Smuzhiyun        This only applies to ext* partition.
361*4882a593Smuzhiyun        """
362*4882a593Smuzhiyun        abs_path = re.sub('\/\/+', '/', path)
363*4882a593Smuzhiyun        cmd = "{} {} -wR 'rm \"{}\"'".format(self.debugfs,
364*4882a593Smuzhiyun                                            self._get_part_image(pnum),
365*4882a593Smuzhiyun                                            abs_path)
366*4882a593Smuzhiyun        out = exec_cmd(cmd , as_shell=True)
367*4882a593Smuzhiyun        for line in out.splitlines():
368*4882a593Smuzhiyun            if line.startswith("rm:"):
369*4882a593Smuzhiyun                if "file is a directory" in line:
370*4882a593Smuzhiyun                    if recursive:
371*4882a593Smuzhiyun                        # loop through content and delete them one by one if
372*4882a593Smuzhiyun                        # flaged with -r
373*4882a593Smuzhiyun                        subdirs = iter(self.dir(pnum, abs_path).splitlines())
374*4882a593Smuzhiyun                        next(subdirs)
375*4882a593Smuzhiyun                        for subdir in subdirs:
376*4882a593Smuzhiyun                            dir = subdir.split(':')[1].split(" ", 1)[1]
377*4882a593Smuzhiyun                            if not dir == "." and not dir == "..":
378*4882a593Smuzhiyun                                self.remove_ext(pnum, "%s/%s" % (abs_path, dir), recursive)
379*4882a593Smuzhiyun
380*4882a593Smuzhiyun                    rmdir_out = exec_cmd("{} {} -wR 'rmdir \"{}\"'".format(self.debugfs,
381*4882a593Smuzhiyun                                                    self._get_part_image(pnum),
382*4882a593Smuzhiyun                                                    abs_path.rstrip('/'))
383*4882a593Smuzhiyun                                                    , as_shell=True)
384*4882a593Smuzhiyun
385*4882a593Smuzhiyun                    for rmdir_line in rmdir_out.splitlines():
386*4882a593Smuzhiyun                        if "directory not empty" in rmdir_line:
387*4882a593Smuzhiyun                            raise WicError("Could not complete operation: \n%s \n"
388*4882a593Smuzhiyun                                            "use -r to remove non-empty directory" % rmdir_line)
389*4882a593Smuzhiyun                        if rmdir_line.startswith("rmdir:"):
390*4882a593Smuzhiyun                            raise WicError("Could not complete operation: \n%s "
391*4882a593Smuzhiyun                                            "\n%s" % (str(line), rmdir_line))
392*4882a593Smuzhiyun
393*4882a593Smuzhiyun                else:
394*4882a593Smuzhiyun                    raise WicError("Could not complete operation: \n%s "
395*4882a593Smuzhiyun                                    "\nUnable to remove %s" % (str(line), abs_path))
396*4882a593Smuzhiyun
397*4882a593Smuzhiyun    def remove(self, pnum, path, recursive):
398*4882a593Smuzhiyun        """Remove files/dirs from the partition."""
399*4882a593Smuzhiyun        partimg = self._get_part_image(pnum)
400*4882a593Smuzhiyun        if self.partitions[pnum].fstype.startswith('ext'):
401*4882a593Smuzhiyun            self.remove_ext(pnum, path, recursive)
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun        else: # fat
404*4882a593Smuzhiyun            cmd = "{} -i {} ::{}".format(self.mdel, partimg, path)
405*4882a593Smuzhiyun            try:
406*4882a593Smuzhiyun                exec_cmd(cmd)
407*4882a593Smuzhiyun            except WicError as err:
408*4882a593Smuzhiyun                if "not found" in str(err) or "non empty" in str(err):
409*4882a593Smuzhiyun                    # mdel outputs 'File ... not found' or 'directory .. non empty"
410*4882a593Smuzhiyun                    # try to use mdeltree as path could be a directory
411*4882a593Smuzhiyun                    cmd = "{} -i {} ::{}".format(self.mdeltree,
412*4882a593Smuzhiyun                                                 partimg, path)
413*4882a593Smuzhiyun                    exec_cmd(cmd)
414*4882a593Smuzhiyun                else:
415*4882a593Smuzhiyun                    raise err
416*4882a593Smuzhiyun        self._put_part_image(pnum)
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun    def write(self, target, expand):
419*4882a593Smuzhiyun        """Write disk image to the media or file."""
420*4882a593Smuzhiyun        def write_sfdisk_script(outf, parts):
421*4882a593Smuzhiyun            for key, val in parts['partitiontable'].items():
422*4882a593Smuzhiyun                if key in ("partitions", "device", "firstlba", "lastlba"):
423*4882a593Smuzhiyun                    continue
424*4882a593Smuzhiyun                if key == "id":
425*4882a593Smuzhiyun                    key = "label-id"
426*4882a593Smuzhiyun                outf.write("{}: {}\n".format(key, val))
427*4882a593Smuzhiyun            outf.write("\n")
428*4882a593Smuzhiyun            for part in parts['partitiontable']['partitions']:
429*4882a593Smuzhiyun                line = ''
430*4882a593Smuzhiyun                for name in ('attrs', 'name', 'size', 'type', 'uuid'):
431*4882a593Smuzhiyun                    if name == 'size' and part['type'] == 'f':
432*4882a593Smuzhiyun                        # don't write size for extended partition
433*4882a593Smuzhiyun                        continue
434*4882a593Smuzhiyun                    val = part.get(name)
435*4882a593Smuzhiyun                    if val:
436*4882a593Smuzhiyun                        line += '{}={}, '.format(name, val)
437*4882a593Smuzhiyun                if line:
438*4882a593Smuzhiyun                    line = line[:-2] # strip ', '
439*4882a593Smuzhiyun                if part.get('bootable'):
440*4882a593Smuzhiyun                    line += ' ,bootable'
441*4882a593Smuzhiyun                outf.write("{}\n".format(line))
442*4882a593Smuzhiyun            outf.flush()
443*4882a593Smuzhiyun
444*4882a593Smuzhiyun        def read_ptable(path):
445*4882a593Smuzhiyun            out = exec_cmd("{} -J {}".format(self.sfdisk, path))
446*4882a593Smuzhiyun            return json.loads(out)
447*4882a593Smuzhiyun
448*4882a593Smuzhiyun        def write_ptable(parts, target):
449*4882a593Smuzhiyun            with tempfile.NamedTemporaryFile(prefix="wic-sfdisk-", mode='w') as outf:
450*4882a593Smuzhiyun                write_sfdisk_script(outf, parts)
451*4882a593Smuzhiyun                cmd = "{} --no-reread {} < {} ".format(self.sfdisk, target, outf.name)
452*4882a593Smuzhiyun                exec_cmd(cmd, as_shell=True)
453*4882a593Smuzhiyun
454*4882a593Smuzhiyun        if expand is None:
455*4882a593Smuzhiyun            sparse_copy(self.imagepath, target)
456*4882a593Smuzhiyun        else:
457*4882a593Smuzhiyun            # copy first sectors that may contain bootloader
458*4882a593Smuzhiyun            sparse_copy(self.imagepath, target, length=2048 * self._lsector_size)
459*4882a593Smuzhiyun
460*4882a593Smuzhiyun            # copy source partition table to the target
461*4882a593Smuzhiyun            parts = read_ptable(self.imagepath)
462*4882a593Smuzhiyun            write_ptable(parts, target)
463*4882a593Smuzhiyun
464*4882a593Smuzhiyun            # get size of unpartitioned space
465*4882a593Smuzhiyun            free = None
466*4882a593Smuzhiyun            for line in exec_cmd("{} -F {}".format(self.sfdisk, target)).splitlines():
467*4882a593Smuzhiyun                if line.startswith("Unpartitioned space ") and line.endswith("sectors"):
468*4882a593Smuzhiyun                    free = int(line.split()[-2])
469*4882a593Smuzhiyun                    # Align free space to a 2048 sector boundary. YOCTO #12840.
470*4882a593Smuzhiyun                    free = free - (free % 2048)
471*4882a593Smuzhiyun            if free is None:
472*4882a593Smuzhiyun                raise WicError("Can't get size of unpartitioned space")
473*4882a593Smuzhiyun
474*4882a593Smuzhiyun            # calculate expanded partitions sizes
475*4882a593Smuzhiyun            sizes = {}
476*4882a593Smuzhiyun            num_auto_resize = 0
477*4882a593Smuzhiyun            for num, part in enumerate(parts['partitiontable']['partitions'], 1):
478*4882a593Smuzhiyun                if num in expand:
479*4882a593Smuzhiyun                    if expand[num] != 0: # don't resize partition if size is set to 0
480*4882a593Smuzhiyun                        sectors = expand[num] // self._lsector_size
481*4882a593Smuzhiyun                        free -= sectors - part['size']
482*4882a593Smuzhiyun                        part['size'] = sectors
483*4882a593Smuzhiyun                        sizes[num] = sectors
484*4882a593Smuzhiyun                elif part['type'] != 'f':
485*4882a593Smuzhiyun                    sizes[num] = -1
486*4882a593Smuzhiyun                    num_auto_resize += 1
487*4882a593Smuzhiyun
488*4882a593Smuzhiyun            for num, part in enumerate(parts['partitiontable']['partitions'], 1):
489*4882a593Smuzhiyun                if sizes.get(num) == -1:
490*4882a593Smuzhiyun                    part['size'] += free // num_auto_resize
491*4882a593Smuzhiyun
492*4882a593Smuzhiyun            # write resized partition table to the target
493*4882a593Smuzhiyun            write_ptable(parts, target)
494*4882a593Smuzhiyun
495*4882a593Smuzhiyun            # read resized partition table
496*4882a593Smuzhiyun            parts = read_ptable(target)
497*4882a593Smuzhiyun
498*4882a593Smuzhiyun            # copy partitions content
499*4882a593Smuzhiyun            for num, part in enumerate(parts['partitiontable']['partitions'], 1):
500*4882a593Smuzhiyun                pnum = str(num)
501*4882a593Smuzhiyun                fstype = self.partitions[pnum].fstype
502*4882a593Smuzhiyun
503*4882a593Smuzhiyun                # copy unchanged partition
504*4882a593Smuzhiyun                if part['size'] == self.partitions[pnum].size // self._lsector_size:
505*4882a593Smuzhiyun                    logger.info("copying unchanged partition {}".format(pnum))
506*4882a593Smuzhiyun                    sparse_copy(self._get_part_image(pnum), target, seek=part['start'] * self._lsector_size)
507*4882a593Smuzhiyun                    continue
508*4882a593Smuzhiyun
509*4882a593Smuzhiyun                # resize or re-create partitions
510*4882a593Smuzhiyun                if fstype.startswith('ext') or fstype.startswith('fat') or \
511*4882a593Smuzhiyun                   fstype.startswith('linux-swap'):
512*4882a593Smuzhiyun
513*4882a593Smuzhiyun                    partfname = None
514*4882a593Smuzhiyun                    with tempfile.NamedTemporaryFile(prefix="wic-part{}-".format(pnum)) as partf:
515*4882a593Smuzhiyun                        partfname = partf.name
516*4882a593Smuzhiyun
517*4882a593Smuzhiyun                    if fstype.startswith('ext'):
518*4882a593Smuzhiyun                        logger.info("resizing ext partition {}".format(pnum))
519*4882a593Smuzhiyun                        partimg = self._get_part_image(pnum)
520*4882a593Smuzhiyun                        sparse_copy(partimg, partfname)
521*4882a593Smuzhiyun                        exec_cmd("{} -pf {}".format(self.e2fsck, partfname))
522*4882a593Smuzhiyun                        exec_cmd("{} {} {}s".format(\
523*4882a593Smuzhiyun                                 self.resize2fs, partfname, part['size']))
524*4882a593Smuzhiyun                    elif fstype.startswith('fat'):
525*4882a593Smuzhiyun                        logger.info("copying content of the fat partition {}".format(pnum))
526*4882a593Smuzhiyun                        with tempfile.TemporaryDirectory(prefix='wic-fatdir-') as tmpdir:
527*4882a593Smuzhiyun                            # copy content to the temporary directory
528*4882a593Smuzhiyun                            cmd = "{} -snompi {} :: {}".format(self.mcopy,
529*4882a593Smuzhiyun                                                               self._get_part_image(pnum),
530*4882a593Smuzhiyun                                                               tmpdir)
531*4882a593Smuzhiyun                            exec_cmd(cmd)
532*4882a593Smuzhiyun                            # create new msdos partition
533*4882a593Smuzhiyun                            label = part.get("name")
534*4882a593Smuzhiyun                            label_str = "-n {}".format(label) if label else ''
535*4882a593Smuzhiyun
536*4882a593Smuzhiyun                            cmd = "{} {} -C {} {}".format(self.mkdosfs, label_str, partfname,
537*4882a593Smuzhiyun                                                          part['size'])
538*4882a593Smuzhiyun                            exec_cmd(cmd)
539*4882a593Smuzhiyun                            # copy content from the temporary directory to the new partition
540*4882a593Smuzhiyun                            cmd = "{} -snompi {} {}/* ::".format(self.mcopy, partfname, tmpdir)
541*4882a593Smuzhiyun                            exec_cmd(cmd, as_shell=True)
542*4882a593Smuzhiyun                    elif fstype.startswith('linux-swap'):
543*4882a593Smuzhiyun                        logger.info("creating swap partition {}".format(pnum))
544*4882a593Smuzhiyun                        label = part.get("name")
545*4882a593Smuzhiyun                        label_str = "-L {}".format(label) if label else ''
546*4882a593Smuzhiyun                        out = exec_cmd("{} --probe {}".format(self.blkid, self._get_part_image(pnum)))
547*4882a593Smuzhiyun                        uuid = out[out.index("UUID=\"")+6:out.index("UUID=\"")+42]
548*4882a593Smuzhiyun                        uuid_str = "-U {}".format(uuid) if uuid else ''
549*4882a593Smuzhiyun                        with open(partfname, 'w') as sparse:
550*4882a593Smuzhiyun                            os.ftruncate(sparse.fileno(), part['size'] * self._lsector_size)
551*4882a593Smuzhiyun                        exec_cmd("{} {} {} {}".format(self.mkswap, label_str, uuid_str, partfname))
552*4882a593Smuzhiyun                    sparse_copy(partfname, target, seek=part['start'] * self._lsector_size)
553*4882a593Smuzhiyun                    os.unlink(partfname)
554*4882a593Smuzhiyun                elif part['type'] != 'f':
555*4882a593Smuzhiyun                    logger.warning("skipping partition {}: unsupported fstype {}".format(pnum, fstype))
556*4882a593Smuzhiyun
557*4882a593Smuzhiyundef wic_ls(args, native_sysroot):
558*4882a593Smuzhiyun    """List contents of partitioned image or vfat partition."""
559*4882a593Smuzhiyun    disk = Disk(args.path.image, native_sysroot)
560*4882a593Smuzhiyun    if not args.path.part:
561*4882a593Smuzhiyun        if disk.partitions:
562*4882a593Smuzhiyun            print('Num     Start        End          Size      Fstype')
563*4882a593Smuzhiyun            for part in disk.partitions.values():
564*4882a593Smuzhiyun                print("{:2d}  {:12d} {:12d} {:12d}  {}".format(\
565*4882a593Smuzhiyun                          part.pnum, part.start, part.end,
566*4882a593Smuzhiyun                          part.size, part.fstype))
567*4882a593Smuzhiyun    else:
568*4882a593Smuzhiyun        path = args.path.path or '/'
569*4882a593Smuzhiyun        print(disk.dir(args.path.part, path))
570*4882a593Smuzhiyun
571*4882a593Smuzhiyundef wic_cp(args, native_sysroot):
572*4882a593Smuzhiyun    """
573*4882a593Smuzhiyun    Copy file or directory to/from the vfat/ext partition of
574*4882a593Smuzhiyun    partitioned image.
575*4882a593Smuzhiyun    """
576*4882a593Smuzhiyun    if isinstance(args.dest, str):
577*4882a593Smuzhiyun        disk = Disk(args.src.image, native_sysroot)
578*4882a593Smuzhiyun    else:
579*4882a593Smuzhiyun        disk = Disk(args.dest.image, native_sysroot)
580*4882a593Smuzhiyun    disk.copy(args.src, args.dest)
581*4882a593Smuzhiyun
582*4882a593Smuzhiyun
583*4882a593Smuzhiyundef wic_rm(args, native_sysroot):
584*4882a593Smuzhiyun    """
585*4882a593Smuzhiyun    Remove files or directories from the vfat partition of
586*4882a593Smuzhiyun    partitioned image.
587*4882a593Smuzhiyun    """
588*4882a593Smuzhiyun    disk = Disk(args.path.image, native_sysroot)
589*4882a593Smuzhiyun    disk.remove(args.path.part, args.path.path, args.recursive_delete)
590*4882a593Smuzhiyun
591*4882a593Smuzhiyundef wic_write(args, native_sysroot):
592*4882a593Smuzhiyun    """
593*4882a593Smuzhiyun    Write image to a target device.
594*4882a593Smuzhiyun    """
595*4882a593Smuzhiyun    disk = Disk(args.image, native_sysroot, ('fat', 'ext', 'linux-swap'))
596*4882a593Smuzhiyun    disk.write(args.target, args.expand)
597*4882a593Smuzhiyun
598*4882a593Smuzhiyundef find_canned(scripts_path, file_name):
599*4882a593Smuzhiyun    """
600*4882a593Smuzhiyun    Find a file either by its path or by name in the canned files dir.
601*4882a593Smuzhiyun
602*4882a593Smuzhiyun    Return None if not found
603*4882a593Smuzhiyun    """
604*4882a593Smuzhiyun    if os.path.exists(file_name):
605*4882a593Smuzhiyun        return file_name
606*4882a593Smuzhiyun
607*4882a593Smuzhiyun    layers_canned_wks_dir = build_canned_image_list(scripts_path)
608*4882a593Smuzhiyun    for canned_wks_dir in layers_canned_wks_dir:
609*4882a593Smuzhiyun        for root, dirs, files in os.walk(canned_wks_dir):
610*4882a593Smuzhiyun            for fname in files:
611*4882a593Smuzhiyun                if fname == file_name:
612*4882a593Smuzhiyun                    fullpath = os.path.join(canned_wks_dir, fname)
613*4882a593Smuzhiyun                    return fullpath
614*4882a593Smuzhiyun
615*4882a593Smuzhiyundef get_custom_config(boot_file):
616*4882a593Smuzhiyun    """
617*4882a593Smuzhiyun    Get the custom configuration to be used for the bootloader.
618*4882a593Smuzhiyun
619*4882a593Smuzhiyun    Return None if the file can't be found.
620*4882a593Smuzhiyun    """
621*4882a593Smuzhiyun    # Get the scripts path of poky
622*4882a593Smuzhiyun    scripts_path = os.path.abspath("%s/../.." % os.path.dirname(__file__))
623*4882a593Smuzhiyun
624*4882a593Smuzhiyun    cfg_file = find_canned(scripts_path, boot_file)
625*4882a593Smuzhiyun    if cfg_file:
626*4882a593Smuzhiyun        with open(cfg_file, "r") as f:
627*4882a593Smuzhiyun            config = f.read()
628*4882a593Smuzhiyun        return config
629