1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909 |
- #
- # Copyright (c) 2015, Intel Corporation.
- #
- # SPDX-License-Identifier: GPL-2.0-only
- #
- # AUTHORS
- # Ed Bartosh <ed.bartosh@linux.intel.com>
- """Test cases for wic."""
- import os
- import sys
- import unittest
- import hashlib
- import subprocess
- from glob import glob
- from shutil import rmtree, copy
- from tempfile import NamedTemporaryFile
- from tempfile import TemporaryDirectory
- from oeqa.selftest.case import OESelftestTestCase
- from oeqa.core.decorator import OETestTag
- from oeqa.core.decorator.data import skipIfNotArch
- from oeqa.utils.commands import runCmd, bitbake, get_bb_var, get_bb_vars, runqemu
- def extract_files(debugfs_output):
- """
- extract file names from the output of debugfs -R 'ls -p',
- which looks like this:
- /2/040755/0/0/.//\n
- /2/040755/0/0/..//\n
- /11/040700/0/0/lost+found^M//\n
- /12/040755/1002/1002/run//\n
- /13/040755/1002/1002/sys//\n
- /14/040755/1002/1002/bin//\n
- /80/040755/1002/1002/var//\n
- /92/040755/1002/1002/tmp//\n
- """
- # NOTE the occasional ^M in file names
- return [line.split('/')[5].strip() for line in \
- debugfs_output.strip().split('/\n')]
- def files_own_by_root(debugfs_output):
- for line in debugfs_output.strip().split('/\n'):
- if line.split('/')[3:5] != ['0', '0']:
- print(debugfs_output)
- return False
- return True
- class WicTestCase(OESelftestTestCase):
- """Wic test class."""
- image_is_ready = False
- wicenv_cache = {}
- def setUpLocal(self):
- """This code is executed before each test method."""
- self.resultdir = os.path.join(self.builddir, "wic-tmp")
- super(WicTestCase, self).setUpLocal()
- # Do this here instead of in setUpClass as the base setUp does some
- # clean up which can result in the native tools built earlier in
- # setUpClass being unavailable.
- if not WicTestCase.image_is_ready:
- if self.td['USE_NLS'] != 'yes':
- self.skipTest('wic-tools needs USE_NLS=yes')
- bitbake('wic-tools core-image-minimal core-image-minimal-mtdutils')
- WicTestCase.image_is_ready = True
- rmtree(self.resultdir, ignore_errors=True)
- def tearDownLocal(self):
- """Remove resultdir as it may contain images."""
- rmtree(self.resultdir, ignore_errors=True)
- super(WicTestCase, self).tearDownLocal()
- def _get_image_env_path(self, image):
- """Generate and obtain the path to <image>.env"""
- if image not in WicTestCase.wicenv_cache:
- bitbake('%s -c do_rootfs_wicenv' % image)
- stdir = get_bb_var('STAGING_DIR', image)
- machine = self.td["MACHINE"]
- WicTestCase.wicenv_cache[image] = os.path.join(stdir, machine, 'imgdata')
- return WicTestCase.wicenv_cache[image]
- class CLITests(OESelftestTestCase):
- def test_version(self):
- """Test wic --version"""
- runCmd('wic --version')
- def test_help(self):
- """Test wic --help and wic -h"""
- runCmd('wic --help')
- runCmd('wic -h')
- def test_createhelp(self):
- """Test wic create --help"""
- runCmd('wic create --help')
- def test_listhelp(self):
- """Test wic list --help"""
- runCmd('wic list --help')
- def test_help_create(self):
- """Test wic help create"""
- runCmd('wic help create')
- def test_help_list(self):
- """Test wic help list"""
- runCmd('wic help list')
- def test_help_overview(self):
- """Test wic help overview"""
- runCmd('wic help overview')
- def test_help_plugins(self):
- """Test wic help plugins"""
- runCmd('wic help plugins')
- def test_help_kickstart(self):
- """Test wic help kickstart"""
- runCmd('wic help kickstart')
- def test_list_images(self):
- """Test wic list images"""
- runCmd('wic list images')
- def test_list_source_plugins(self):
- """Test wic list source-plugins"""
- runCmd('wic list source-plugins')
- def test_listed_images_help(self):
- """Test wic listed images help"""
- output = runCmd('wic list images').output
- imagelist = [line.split()[0] for line in output.splitlines()]
- for image in imagelist:
- runCmd('wic list %s help' % image)
- def test_unsupported_subcommand(self):
- """Test unsupported subcommand"""
- self.assertNotEqual(0, runCmd('wic unsupported', ignore_status=True).status)
- def test_no_command(self):
- """Test wic without command"""
- self.assertEqual(1, runCmd('wic', ignore_status=True).status)
- class Wic(WicTestCase):
- def test_skip_kernel_install(self):
- """Test the functionality of not installing the kernel in the boot directory using the wic plugin"""
- # create a temporary file for the WKS content
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.write(
- 'part --source bootimg-efi '
- '--sourceparams="loader=grub-efi,install-kernel-into-boot-dir=false" '
- '--label boot --active\n'
- )
- wks.flush()
- # create a temporary directory to extract the disk image to
- with TemporaryDirectory() as tmpdir:
- img = 'core-image-minimal'
- # build the image using the WKS file
- cmd = "wic create %s -e %s -o %s" % (
- wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(
- self.resultdir, "%s-*.direct" % wksname))
- self.assertEqual(1, len(out))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # extract the content of the disk image to the temporary directory
- cmd = "wic cp %s:1 %s -n %s" % (out[0], tmpdir, sysroot)
- runCmd(cmd)
- # check if the kernel is installed or not
- kimgtype = get_bb_var('KERNEL_IMAGETYPE', img)
- for file in os.listdir(tmpdir):
- if file == kimgtype:
- raise AssertionError(
- "The kernel image '{}' was found in the partition".format(kimgtype)
- )
- def test_kernel_install(self):
- """Test the installation of the kernel to the boot directory in the wic plugin"""
- # create a temporary file for the WKS content
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.write(
- 'part --source bootimg-efi '
- '--sourceparams="loader=grub-efi,install-kernel-into-boot-dir=true" '
- '--label boot --active\n'
- )
- wks.flush()
- # create a temporary directory to extract the disk image to
- with TemporaryDirectory() as tmpdir:
- img = 'core-image-minimal'
- # build the image using the WKS file
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*.direct" % wksname))
- self.assertEqual(1, len(out))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # extract the content of the disk image to the temporary directory
- cmd = "wic cp %s:1 %s -n %s" % (out[0], tmpdir, sysroot)
- runCmd(cmd)
- # check if the kernel is installed or not
- kimgtype = get_bb_var('KERNEL_IMAGETYPE', img)
- found = False
- for file in os.listdir(tmpdir):
- if file == kimgtype:
- found = True
- break
- self.assertTrue(
- found, "The kernel image '{}' was not found in the boot partition".format(kimgtype)
- )
- def test_build_image_name(self):
- """Test wic create wictestdisk --image-name=core-image-minimal"""
- cmd = "wic create wictestdisk --image-name=core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join (self.resultdir, "wictestdisk-*.direct"))))
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_gpt_image(self):
- """Test creation of core-image-minimal with gpt table and UUID boot"""
- cmd = "wic create directdisk-gpt --image-name core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "directdisk-*.direct"))))
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_iso_image(self):
- """Test creation of hybrid iso image with legacy and EFI boot"""
- config = 'INITRAMFS_IMAGE = "core-image-minimal-initramfs"\n'\
- 'MACHINE_FEATURES:append = " efi"\n'\
- 'DEPENDS:pn-core-image-minimal += "syslinux"\n'
- self.append_config(config)
- bitbake('core-image-minimal core-image-minimal-initramfs')
- self.remove_config(config)
- cmd = "wic create mkhybridiso --image-name core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "HYBRID_ISO_IMG-*.direct"))))
- self.assertEqual(1, len(glob(os.path.join (self.resultdir, "HYBRID_ISO_IMG-*.iso"))))
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_qemux86_directdisk(self):
- """Test creation of qemux-86-directdisk image"""
- cmd = "wic create qemux86-directdisk -e core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "qemux86-directdisk-*direct"))))
- @skipIfNotArch(['i586', 'i686', 'x86_64', 'aarch64'])
- def test_mkefidisk(self):
- """Test creation of mkefidisk image"""
- cmd = "wic create mkefidisk -e core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "mkefidisk-*direct"))))
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_bootloader_config(self):
- """Test creation of directdisk-bootloader-config image"""
- config = 'DEPENDS:pn-core-image-minimal += "syslinux"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- self.remove_config(config)
- cmd = "wic create directdisk-bootloader-config -e core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "directdisk-bootloader-config-*direct"))))
- @skipIfNotArch(['i586', 'i686', 'x86_64', 'aarch64'])
- def test_systemd_bootdisk(self):
- """Test creation of systemd-bootdisk image"""
- config = 'MACHINE_FEATURES:append = " efi"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- self.remove_config(config)
- cmd = "wic create systemd-bootdisk -e core-image-minimal -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "systemd-bootdisk-*direct"))))
- def test_efi_bootpart(self):
- """Test creation of efi-bootpart image"""
- cmd = "wic create mkefidisk -e core-image-minimal -o %s" % self.resultdir
- kimgtype = get_bb_var('KERNEL_IMAGETYPE', 'core-image-minimal')
- self.append_config('IMAGE_EFI_BOOT_FILES = "%s;kernel"\n' % kimgtype)
- runCmd(cmd)
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- images = glob(os.path.join(self.resultdir, "mkefidisk-*.direct"))
- result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot))
- self.assertIn("kernel",result.output)
- def test_sdimage_bootpart(self):
- """Test creation of sdimage-bootpart image"""
- cmd = "wic create sdimage-bootpart -e core-image-minimal -o %s" % self.resultdir
- kimgtype = get_bb_var('KERNEL_IMAGETYPE', 'core-image-minimal')
- self.write_config('IMAGE_BOOT_FILES = "%s"\n' % kimgtype)
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "sdimage-bootpart-*direct"))))
- # TODO this doesn't have to be x86-specific
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_default_output_dir(self):
- """Test default output location"""
- for fname in glob("directdisk-*.direct"):
- os.remove(fname)
- config = 'DEPENDS:pn-core-image-minimal += "syslinux"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- self.remove_config(config)
- cmd = "wic create directdisk -e core-image-minimal"
- runCmd(cmd)
- self.assertEqual(1, len(glob("directdisk-*.direct")))
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_build_artifacts(self):
- """Test wic create directdisk providing all artifacts."""
- bb_vars = get_bb_vars(['STAGING_DATADIR', 'RECIPE_SYSROOT_NATIVE'],
- 'wic-tools')
- bb_vars.update(get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_ROOTFS'],
- 'core-image-minimal'))
- bbvars = {key.lower(): value for key, value in bb_vars.items()}
- bbvars['resultdir'] = self.resultdir
- runCmd("wic create directdisk "
- "-b %(staging_datadir)s "
- "-k %(deploy_dir_image)s "
- "-n %(recipe_sysroot_native)s "
- "-r %(image_rootfs)s "
- "-o %(resultdir)s" % bbvars)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "directdisk-*.direct"))))
- def test_compress_gzip(self):
- """Test compressing an image with gzip"""
- runCmd("wic create wictestdisk "
- "--image-name core-image-minimal "
- "-c gzip -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct.gz"))))
- def test_compress_bzip2(self):
- """Test compressing an image with bzip2"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-c bzip2 -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct.bz2"))))
- def test_compress_xz(self):
- """Test compressing an image with xz"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "--compress-with=xz -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct.xz"))))
- def test_wrong_compressor(self):
- """Test how wic breaks if wrong compressor is provided"""
- self.assertEqual(2, runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-c wrong -o %s" % self.resultdir,
- ignore_status=True).status)
- def test_debug_short(self):
- """Test -D option"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))))
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "tmp.wic*"))))
- def test_debug_long(self):
- """Test --debug option"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "--debug -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))))
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "tmp.wic*"))))
- def test_skip_build_check_short(self):
- """Test -s option"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-s -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))))
- def test_skip_build_check_long(self):
- """Test --skip-build-check option"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "--skip-build-check "
- "--outdir %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))))
- def test_build_rootfs_short(self):
- """Test -f option"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-f -o %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))))
- def test_build_rootfs_long(self):
- """Test --build-rootfs option"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "--build-rootfs "
- "--outdir %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))))
- # TODO this doesn't have to be x86-specific
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_rootfs_indirect_recipes(self):
- """Test usage of rootfs plugin with rootfs recipes"""
- runCmd("wic create directdisk-multi-rootfs "
- "--image-name=core-image-minimal "
- "--rootfs rootfs1=core-image-minimal "
- "--rootfs rootfs2=core-image-minimal "
- "--outdir %s" % self.resultdir)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "directdisk-multi-rootfs*.direct"))))
- # TODO this doesn't have to be x86-specific
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_rootfs_artifacts(self):
- """Test usage of rootfs plugin with rootfs paths"""
- bb_vars = get_bb_vars(['STAGING_DATADIR', 'RECIPE_SYSROOT_NATIVE'],
- 'wic-tools')
- bb_vars.update(get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_ROOTFS'],
- 'core-image-minimal'))
- bbvars = {key.lower(): value for key, value in bb_vars.items()}
- bbvars['wks'] = "directdisk-multi-rootfs"
- bbvars['resultdir'] = self.resultdir
- runCmd("wic create %(wks)s "
- "--bootimg-dir=%(staging_datadir)s "
- "--kernel-dir=%(deploy_dir_image)s "
- "--native-sysroot=%(recipe_sysroot_native)s "
- "--rootfs-dir rootfs1=%(image_rootfs)s "
- "--rootfs-dir rootfs2=%(image_rootfs)s "
- "--outdir %(resultdir)s" % bbvars)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "%(wks)s-*.direct" % bbvars))))
- def test_exclude_path(self):
- """Test --exclude-path wks option."""
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- try:
- wks_file = 'temp.wks'
- with open(wks_file, 'w') as wks:
- rootfs_dir = get_bb_var('IMAGE_ROOTFS', 'core-image-minimal')
- wks.write("""
- part / --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path usr
- part /usr --source rootfs --ondisk mmcblk0 --fstype=ext4 --rootfs-dir %s/usr
- part /etc --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path bin/ --rootfs-dir %s/usr
- part /mnt --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path bin/whoami --rootfs-dir %s/usr"""
- % (rootfs_dir, rootfs_dir, rootfs_dir))
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- os.remove(wks_file)
- wicout = glob(os.path.join(self.resultdir, "%s-*direct" % 'temp'))
- self.assertEqual(1, len(wicout))
- wicimg = wicout[0]
- # verify partition size with wic
- res = runCmd("parted -m %s unit b p" % wicimg, stderr=subprocess.PIPE)
- # parse parted output which looks like this:
- # BYT;\n
- # /var/tmp/wic/build/tmpfwvjjkf_-201611101222-hda.direct:200MiB:file:512:512:msdos::;\n
- # 1:0.00MiB:200MiB:200MiB:ext4::;\n
- partlns = res.output.splitlines()[2:]
- self.assertEqual(4, len(partlns))
- for part in [1, 2, 3, 4]:
- part_file = os.path.join(self.resultdir, "selftest_img.part%d" % part)
- partln = partlns[part-1].split(":")
- self.assertEqual(7, len(partln))
- start = int(partln[1].rstrip("B")) / 512
- length = int(partln[3].rstrip("B")) / 512
- runCmd("dd if=%s of=%s skip=%d count=%d" %
- (wicimg, part_file, start, length))
- # Test partition 1, should contain the normal root directories, except
- # /usr.
- res = runCmd("debugfs -R 'ls -p' %s" % \
- os.path.join(self.resultdir, "selftest_img.part1"), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn("etc", files)
- self.assertNotIn("usr", files)
- # Partition 2, should contain common directories for /usr, not root
- # directories.
- res = runCmd("debugfs -R 'ls -p' %s" % \
- os.path.join(self.resultdir, "selftest_img.part2"), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertNotIn("etc", files)
- self.assertNotIn("usr", files)
- self.assertIn("share", files)
- # Partition 3, should contain the same as partition 2, including the bin
- # directory, but not the files inside it.
- res = runCmd("debugfs -R 'ls -p' %s" % \
- os.path.join(self.resultdir, "selftest_img.part3"), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertNotIn("etc", files)
- self.assertNotIn("usr", files)
- self.assertIn("share", files)
- self.assertIn("bin", files)
- res = runCmd("debugfs -R 'ls -p bin' %s" % \
- os.path.join(self.resultdir, "selftest_img.part3"), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn(".", files)
- self.assertIn("..", files)
- self.assertEqual(2, len(files))
- # Partition 4, should contain the same as partition 2, including the bin
- # directory, but not whoami (a symlink to busybox.nosuid) inside it.
- res = runCmd("debugfs -R 'ls -p' %s" % \
- os.path.join(self.resultdir, "selftest_img.part4"), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertNotIn("etc", files)
- self.assertNotIn("usr", files)
- self.assertIn("share", files)
- self.assertIn("bin", files)
- res = runCmd("debugfs -R 'ls -p bin' %s" % \
- os.path.join(self.resultdir, "selftest_img.part4"), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn(".", files)
- self.assertIn("..", files)
- self.assertIn("who", files)
- self.assertNotIn("whoami", files)
- for part in [1, 2, 3, 4]:
- part_file = os.path.join(self.resultdir, "selftest_img.part%d" % part)
- os.remove(part_file)
- finally:
- os.environ['PATH'] = oldpath
- def test_exclude_path_with_extra_space(self):
- """Test having --exclude-path with IMAGE_ROOTFS_EXTRA_SPACE. [Yocto #15555]"""
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- try:
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(
- ['bootloader --ptable gpt\n',
- 'part /boot --size=100M --active --fstype=ext4 --label boot\n',
- 'part / --source rootfs --fstype=ext4 --label root --exclude-path boot/\n'])
- wks.flush()
- config = 'IMAGE_ROOTFS_EXTRA_SPACE = "500000"\n'\
- 'DEPENDS:pn-core-image-minimal += "wic-tools"\n'\
- 'IMAGE_FSTYPES += "wic"\n'\
- 'WKS_FILE = "%s"\n' % wks.name
- self.append_config(config)
- bitbake('core-image-minimal')
- """
- the output of "wic ls <image>.wic" will look something like:
- Num Start End Size Fstype
- 1 17408 136332287 136314880 ext4
- 2 136332288 171464703 35132416 ext4
- we are looking for the size of partition 2
- i.e. in this case the number 35,132,416
- without the fix the size will be around 85,403,648
- with the fix the size should be around 799,960,064
- """
- bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'MACHINE'], 'core-image-minimal')
- deploy_dir = bb_vars['DEPLOY_DIR_IMAGE']
- machine = bb_vars['MACHINE']
- wicout = glob(os.path.join(deploy_dir, "core-image-minimal-%s.rootfs-*.wic" % machine))[0]
- size_of_root_partition = int(runCmd("wic ls %s" % wicout).output.split('\n')[2].split()[3])
- self.assertGreater(size_of_root_partition, 500000000)
- finally:
- os.environ['PATH'] = oldpath
- def test_include_path(self):
- """Test --include-path wks option."""
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- try:
- include_path = os.path.join(self.resultdir, 'test-include')
- os.makedirs(include_path)
- with open(os.path.join(include_path, 'test-file'), 'w') as t:
- t.write("test\n")
- wks_file = os.path.join(include_path, 'temp.wks')
- with open(wks_file, 'w') as wks:
- rootfs_dir = get_bb_var('IMAGE_ROOTFS', 'core-image-minimal')
- wks.write("""
- part /part1 --source rootfs --ondisk mmcblk0 --fstype=ext4
- part /part2 --source rootfs --ondisk mmcblk0 --fstype=ext4 --include-path %s"""
- % (include_path))
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- part1 = glob(os.path.join(self.resultdir, 'temp-*.direct.p1'))[0]
- part2 = glob(os.path.join(self.resultdir, 'temp-*.direct.p2'))[0]
- # Test partition 1, should not contain 'test-file'
- res = runCmd("debugfs -R 'ls -p' %s" % (part1), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertNotIn('test-file', files)
- self.assertEqual(True, files_own_by_root(res.output))
- # Test partition 2, should contain 'test-file'
- res = runCmd("debugfs -R 'ls -p' %s" % (part2), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn('test-file', files)
- self.assertEqual(True, files_own_by_root(res.output))
- finally:
- os.environ['PATH'] = oldpath
- def test_include_path_embeded(self):
- """Test --include-path wks option."""
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- try:
- include_path = os.path.join(self.resultdir, 'test-include')
- os.makedirs(include_path)
- with open(os.path.join(include_path, 'test-file'), 'w') as t:
- t.write("test\n")
- wks_file = os.path.join(include_path, 'temp.wks')
- with open(wks_file, 'w') as wks:
- wks.write("""
- part / --source rootfs --fstype=ext4 --include-path %s --include-path core-image-minimal-mtdutils export/"""
- % (include_path))
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- part1 = glob(os.path.join(self.resultdir, 'temp-*.direct.p1'))[0]
- res = runCmd("debugfs -R 'ls -p' %s" % (part1), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn('test-file', files)
- self.assertEqual(True, files_own_by_root(res.output))
- res = runCmd("debugfs -R 'ls -p /export/etc/' %s" % (part1), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn('passwd', files)
- self.assertEqual(True, files_own_by_root(res.output))
- finally:
- os.environ['PATH'] = oldpath
- def test_include_path_errors(self):
- """Test --include-path wks option error handling."""
- wks_file = 'temp.wks'
- # Absolute argument.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --include-path core-image-minimal-mtdutils /export")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- # Argument pointing to parent directory.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --include-path core-image-minimal-mtdutils ././..")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- # 3 Argument pointing to parent directory.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --include-path core-image-minimal-mtdutils export/ dummy")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- def test_exclude_path_errors(self):
- """Test --exclude-path wks option error handling."""
- wks_file = 'temp.wks'
- # Absolute argument.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path /usr")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- # Argument pointing to parent directory.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --ondisk mmcblk0 --fstype=ext4 --exclude-path ././..")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- def test_permissions(self):
- """Test permissions are respected"""
- # prepare wicenv and rootfs
- bitbake('core-image-minimal core-image-minimal-mtdutils -c do_rootfs_wicenv')
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- t_normal = """
- part / --source rootfs --fstype=ext4
- """
- t_exclude = """
- part / --source rootfs --fstype=ext4 --exclude-path=home
- """
- t_multi = """
- part / --source rootfs --ondisk sda --fstype=ext4
- part /export --source rootfs --rootfs=core-image-minimal-mtdutils --fstype=ext4
- """
- t_change = """
- part / --source rootfs --ondisk sda --fstype=ext4 --exclude-path=etc/
- part /etc --source rootfs --fstype=ext4 --change-directory=etc
- """
- tests = [t_normal, t_exclude, t_multi, t_change]
- try:
- for test in tests:
- include_path = os.path.join(self.resultdir, 'test-include')
- os.makedirs(include_path)
- wks_file = os.path.join(include_path, 'temp.wks')
- with open(wks_file, 'w') as wks:
- wks.write(test)
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- for part in glob(os.path.join(self.resultdir, 'temp-*.direct.p*')):
- res = runCmd("debugfs -R 'ls -p' %s" % (part), stderr=subprocess.PIPE)
- self.assertEqual(True, files_own_by_root(res.output))
- config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "%s"\n' % wks_file
- self.append_config(config)
- bitbake('core-image-minimal')
- tmpdir = os.path.join(get_bb_var('WORKDIR', 'core-image-minimal'),'build-wic')
- # check each partition for permission
- for part in glob(os.path.join(tmpdir, 'temp-*.direct.p*')):
- res = runCmd("debugfs -R 'ls -p' %s" % (part), stderr=subprocess.PIPE)
- self.assertTrue(files_own_by_root(res.output)
- ,msg='Files permission incorrect using wks set "%s"' % test)
- # clean config and result directory for next cases
- self.remove_config(config)
- rmtree(self.resultdir, ignore_errors=True)
- finally:
- os.environ['PATH'] = oldpath
- def test_change_directory(self):
- """Test --change-directory wks option."""
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- try:
- include_path = os.path.join(self.resultdir, 'test-include')
- os.makedirs(include_path)
- wks_file = os.path.join(include_path, 'temp.wks')
- with open(wks_file, 'w') as wks:
- wks.write("part /etc --source rootfs --fstype=ext4 --change-directory=etc")
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- part1 = glob(os.path.join(self.resultdir, 'temp-*.direct.p1'))[0]
- res = runCmd("debugfs -R 'ls -p' %s" % (part1), stderr=subprocess.PIPE)
- files = extract_files(res.output)
- self.assertIn('passwd', files)
- finally:
- os.environ['PATH'] = oldpath
- def test_change_directory_errors(self):
- """Test --change-directory wks option error handling."""
- wks_file = 'temp.wks'
- # Absolute argument.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --change-directory /usr")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- # Argument pointing to parent directory.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --change-directory ././..")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- def test_no_fstab_update(self):
- """Test --no-fstab-update wks option."""
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- # Get stock fstab from base-files recipe
- bitbake('base-files -c do_install')
- bf_fstab = os.path.join(get_bb_var('D', 'base-files'), 'etc', 'fstab')
- self.assertEqual(True, os.path.exists(bf_fstab))
- bf_fstab_md5sum = runCmd('md5sum %s ' % bf_fstab).output.split(" ")[0]
- try:
- no_fstab_update_path = os.path.join(self.resultdir, 'test-no-fstab-update')
- os.makedirs(no_fstab_update_path)
- wks_file = os.path.join(no_fstab_update_path, 'temp.wks')
- with open(wks_file, 'w') as wks:
- wks.writelines(['part / --source rootfs --fstype=ext4 --label rootfs\n',
- 'part /mnt/p2 --source rootfs --rootfs-dir=core-image-minimal ',
- '--fstype=ext4 --label p2 --no-fstab-update\n'])
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- part_fstab_md5sum = []
- for i in range(1, 3):
- part = glob(os.path.join(self.resultdir, 'temp-*.direct.p') + str(i))[0]
- part_fstab = runCmd("debugfs -R 'cat etc/fstab' %s" % (part), stderr=subprocess.PIPE)
- part_fstab_md5sum.append(hashlib.md5((part_fstab.output + "\n\n").encode('utf-8')).hexdigest())
- # '/etc/fstab' in partition 2 should contain the same stock fstab file
- # as the one installed by the base-file recipe.
- self.assertEqual(bf_fstab_md5sum, part_fstab_md5sum[1])
- # '/etc/fstab' in partition 1 should contain an updated fstab file.
- self.assertNotEqual(bf_fstab_md5sum, part_fstab_md5sum[0])
- finally:
- os.environ['PATH'] = oldpath
- def test_no_fstab_update_errors(self):
- """Test --no-fstab-update wks option error handling."""
- wks_file = 'temp.wks'
- # Absolute argument.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --no-fstab-update /etc")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- # Argument pointing to parent directory.
- with open(wks_file, 'w') as wks:
- wks.write("part / --source rootfs --fstype=ext4 --no-fstab-update ././..")
- self.assertNotEqual(0, runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir), ignore_status=True).status)
- os.remove(wks_file)
- def test_extra_space(self):
- """Test --extra-space wks option."""
- extraspace = 1024**3
- runCmd("wic create wictestdisk "
- "--image-name core-image-minimal "
- "--extra-space %i -o %s" % (extraspace ,self.resultdir))
- wicout = glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))
- self.assertEqual(1, len(wicout))
- size = os.path.getsize(wicout[0])
- self.assertTrue(size > extraspace, msg="Extra space not present (%s vs %s)" % (size, extraspace))
- def test_no_table(self):
- """Test --no-table wks option."""
- wks_file = 'temp.wks'
- # Absolute argument.
- with open(wks_file, 'w') as wks:
- wks.write("part testspace --no-table --fixed-size 16k --offset 4080k")
- runCmd("wic create %s --image-name core-image-minimal -o %s" % (wks_file, self.resultdir))
- wicout = glob(os.path.join(self.resultdir, "*.*"))
- self.assertEqual(1, len(wicout))
- size = os.path.getsize(wicout[0])
- self.assertEqual(size, 4 * 1024 * 1024)
- os.remove(wks_file)
- def test_partition_hidden_attributes(self):
- """Test --hidden wks option."""
- wks_file = 'temp.wks'
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- try:
- with open(wks_file, 'w') as wks:
- wks.write("""
- part / --source rootfs --fstype=ext4
- part / --source rootfs --fstype=ext4 --hidden
- bootloader --ptable gpt""")
- runCmd("wic create %s -e core-image-minimal -o %s" \
- % (wks_file, self.resultdir))
- wicout = os.path.join(self.resultdir, "*.direct")
- result = runCmd("%s/usr/sbin/sfdisk --part-attrs %s 1" % (sysroot, wicout))
- self.assertEqual('', result.output)
- result = runCmd("%s/usr/sbin/sfdisk --part-attrs %s 2" % (sysroot, wicout))
- self.assertEqual('RequiredPartition', result.output)
- finally:
- os.remove(wks_file)
- def test_wic_sector_size(self):
- """Test generation image sector size"""
-
- oldpath = os.environ['PATH']
- os.environ['PATH'] = get_bb_var("PATH", "wic-tools")
- try:
- # Add WIC_SECTOR_SIZE into config
- config = 'WIC_SECTOR_SIZE = "4096"\n'\
- 'WICVARS:append = " WIC_SECTOR_SIZE"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- # Check WIC_SECTOR_SIZE apply to bitbake variable
- wic_sector_size_str = get_bb_var('WIC_SECTOR_SIZE', 'core-image-minimal')
- wic_sector_size = int(wic_sector_size_str)
- self.assertEqual(4096, wic_sector_size)
- self.logger.info("Test wic_sector_size: %d \n" % wic_sector_size)
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(
- ['bootloader --ptable gpt\n',
- 'part --fstype ext4 --source rootfs --label rofs-a --mkfs-extraopts "-b 4096"\n',
- 'part --fstype ext4 --source rootfs --use-uuid --mkfs-extraopts "-b 4096"\n'])
- wks.flush()
- cmd = "wic create %s -e core-image-minimal -o %s" % (wks.name, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- images = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # list partitions
- result = runCmd("wic ls %s -n %s" % (images[0], sysroot))
- self.assertEqual(3, len(result.output.split('\n')))
- # verify partition size with wic
- res = runCmd("export PARTED_SECTOR_SIZE=%d; parted -m %s unit b p" % (wic_sector_size, images[0]),
- stderr=subprocess.PIPE)
- # parse parted output which looks like this:
- # BYT;\n
- # /var/tmp/wic/build/tmpgjzzefdd-202410281021-sda.direct:78569472B:file:4096:4096:gpt::;\n
- # 1:139264B:39284735B:39145472B:ext4:rofs-a:;\n
- # 2:39284736B:78430207B:39145472B:ext4:primary:;\n
- disk_info = res.output.splitlines()[1]
- # Check sector sizes
- sector_size_logical = int(disk_info.split(":")[3])
- sector_size_physical = int(disk_info.split(":")[4])
- self.assertEqual(wic_sector_size, sector_size_logical, "Logical sector size is not %d." % wic_sector_size)
- self.assertEqual(wic_sector_size, sector_size_physical, "Physical sector size is not %d." % wic_sector_size)
- finally:
- os.environ['PATH'] = oldpath
- class Wic2(WicTestCase):
- def test_bmap_short(self):
- """Test generation of .bmap file -m option"""
- cmd = "wic create wictestdisk -e core-image-minimal -m -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*direct"))))
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*direct.bmap"))))
- def test_bmap_long(self):
- """Test generation of .bmap file --bmap option"""
- cmd = "wic create wictestdisk -e core-image-minimal --bmap -o %s" % self.resultdir
- runCmd(cmd)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*direct"))))
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*direct.bmap"))))
- def test_image_env(self):
- """Test generation of <image>.env files."""
- image = 'core-image-minimal'
- imgdatadir = self._get_image_env_path(image)
- bb_vars = get_bb_vars(['IMAGE_BASENAME', 'WICVARS'], image)
- basename = bb_vars['IMAGE_BASENAME']
- self.assertEqual(basename, image)
- path = os.path.join(imgdatadir, basename) + '.env'
- self.assertTrue(os.path.isfile(path), msg="File %s wasn't generated as expected" % path)
- wicvars = set(bb_vars['WICVARS'].split())
- # filter out optional variables
- wicvars = wicvars.difference(('DEPLOY_DIR_IMAGE', 'IMAGE_BOOT_FILES',
- 'INITRD', 'INITRD_LIVE', 'ISODIR','INITRAMFS_IMAGE',
- 'INITRAMFS_IMAGE_BUNDLE', 'INITRAMFS_LINK_NAME',
- 'APPEND', 'IMAGE_EFI_BOOT_FILES'))
- with open(path) as envfile:
- content = dict(line.split("=", 1) for line in envfile)
- # test if variables used by wic present in the .env file
- for var in wicvars:
- self.assertTrue(var in content, "%s is not in .env file" % var)
- self.assertTrue(content[var], "%s doesn't have a value (%s)" % (var, content[var]))
- def test_image_vars_dir_short(self):
- """Test image vars directory selection -v option"""
- image = 'core-image-minimal'
- imgenvdir = self._get_image_env_path(image)
- native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools")
- runCmd("wic create wictestdisk "
- "--image-name=%s -v %s -n %s -o %s"
- % (image, imgenvdir, native_sysroot,
- self.resultdir))
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*direct"))))
- def test_image_vars_dir_long(self):
- """Test image vars directory selection --vars option"""
- image = 'core-image-minimal'
- imgenvdir = self._get_image_env_path(image)
- native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools")
- runCmd("wic create wictestdisk "
- "--image-name=%s "
- "--vars %s "
- "--native-sysroot %s "
- "--outdir %s"
- % (image, imgenvdir, native_sysroot,
- self.resultdir))
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "wictestdisk-*direct"))))
- # TODO this test could also work on aarch64
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_wic_image_type(self):
- """Test building wic images by bitbake"""
- config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "wic-image-minimal"\n'\
- 'MACHINE_FEATURES:append = " efi"\n'
- self.append_config(config)
- image = 'wic-image-minimal'
- bitbake(image)
- self.remove_config(config)
- bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_LINK_NAME'], image)
- prefix = os.path.join(bb_vars['DEPLOY_DIR_IMAGE'], '%s.' % bb_vars['IMAGE_LINK_NAME'])
- # check if we have result image and manifests symlinks
- # pointing to existing files
- for suffix in ('wic', 'manifest'):
- path = prefix + suffix
- self.assertTrue(os.path.islink(path), msg="Link %s wasn't generated as expected" % path)
- self.assertTrue(os.path.isfile(os.path.realpath(path)), msg="File linked to by %s wasn't generated as expected" % path)
- # TODO this should work on aarch64
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_qemu(self):
- """Test wic-image-minimal under qemu"""
- config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "wic-image-minimal"\n'\
- 'MACHINE_FEATURES:append = " efi"\n'
- self.append_config(config)
- bitbake('wic-image-minimal')
- self.remove_config(config)
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'wic-image-minimal') or ""
- with runqemu('wic-image-minimal', ssh=False, runqemuparams='%s nographic' % (runqemu_params)) as qemu:
- cmd = "mount | grep '^/dev/' | cut -f1,3 -d ' ' | egrep -c -e '/dev/sda1 /boot' " \
- "-e '/dev/root /|/dev/sda2 /' -e '/dev/sda3 /media' -e '/dev/sda4 /mnt'"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '4')
- cmd = "grep UUID=2c71ef06-a81d-4735-9d3a-379b69c6bdba /etc/fstab"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, 'UUID=2c71ef06-a81d-4735-9d3a-379b69c6bdba\t/media\text4\tdefaults\t0\t0')
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_qemu_efi(self):
- """Test core-image-minimal efi image under qemu"""
- config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "mkefidisk.wks"\n'
- self.append_config(config)
- bitbake('core-image-minimal ovmf')
- self.remove_config(config)
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-minimal') or ""
- with runqemu('core-image-minimal', ssh=False,
- runqemuparams='%s nographic ovmf' % (runqemu_params), image_fstype='wic') as qemu:
- cmd = "grep sda. /proc/partitions |wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '3')
- @staticmethod
- def _make_fixed_size_wks(size):
- """
- Create a wks of an image with a single partition. Size of the partition is set
- using --fixed-size flag. Returns a tuple: (path to wks file, wks image name)
- """
- with NamedTemporaryFile("w", suffix=".wks", delete=False) as tempf:
- wkspath = tempf.name
- tempf.write("part " \
- "--source rootfs --ondisk hda --align 4 --fixed-size %d "
- "--fstype=ext4\n" % size)
- return wkspath
- def _get_wic_partitions(self, wkspath, native_sysroot=None, ignore_status=False):
- p = runCmd("wic create %s -e core-image-minimal -o %s" % (wkspath, self.resultdir),
- ignore_status=ignore_status)
- if p.status:
- return (p, None)
- wksname = os.path.splitext(os.path.basename(wkspath))[0]
- wicout = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- if not wicout:
- return (p, None)
- wicimg = wicout[0]
- if not native_sysroot:
- native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools")
- # verify partition size with wic
- res = runCmd("parted -m %s unit kib p" % wicimg,
- native_sysroot=native_sysroot, stderr=subprocess.PIPE)
- # parse parted output which looks like this:
- # BYT;\n
- # /var/tmp/wic/build/tmpfwvjjkf_-201611101222-hda.direct:200MiB:file:512:512:msdos::;\n
- # 1:0.00MiB:200MiB:200MiB:ext4::;\n
- return (p, res.output.splitlines()[2:])
- def test_fixed_size(self):
- """
- Test creation of a simple image with partition size controlled through
- --fixed-size flag
- """
- wkspath = Wic2._make_fixed_size_wks(200)
- _, partlns = self._get_wic_partitions(wkspath)
- os.remove(wkspath)
- self.assertEqual(partlns, [
- "1:4.00kiB:204804kiB:204800kiB:ext4::;",
- ])
- def test_fixed_size_error(self):
- """
- Test creation of a simple image with partition size controlled through
- --fixed-size flag. The size of partition is intentionally set to 1MiB
- in order to trigger an error in wic.
- """
- wkspath = Wic2._make_fixed_size_wks(1)
- p, _ = self._get_wic_partitions(wkspath, ignore_status=True)
- os.remove(wkspath)
- self.assertNotEqual(p.status, 0, "wic exited successfully when an error was expected:\n%s" % p.output)
- def test_offset(self):
- native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools")
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that partitions are placed at the correct offsets, default KB
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --offset 32 --fixed-size 100M --fstype=ext4\n" \
- "part /bar --ondisk hda --offset 102432 --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- _, partlns = self._get_wic_partitions(tempf.name, native_sysroot)
- self.assertEqual(partlns, [
- "1:32.0kiB:102432kiB:102400kiB:ext4:primary:;",
- "2:102432kiB:204832kiB:102400kiB:ext4:primary:;",
- ])
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that partitions are placed at the correct offsets, same with explicit KB
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --offset 32K --fixed-size 100M --fstype=ext4\n" \
- "part /bar --ondisk hda --offset 102432K --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- _, partlns = self._get_wic_partitions(tempf.name, native_sysroot)
- self.assertEqual(partlns, [
- "1:32.0kiB:102432kiB:102400kiB:ext4:primary:;",
- "2:102432kiB:204832kiB:102400kiB:ext4:primary:;",
- ])
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that partitions are placed at the correct offsets using MB
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --offset 32K --fixed-size 100M --fstype=ext4\n" \
- "part /bar --ondisk hda --offset 101M --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- _, partlns = self._get_wic_partitions(tempf.name, native_sysroot)
- self.assertEqual(partlns, [
- "1:32.0kiB:102432kiB:102400kiB:ext4:primary:;",
- "2:103424kiB:205824kiB:102400kiB:ext4:primary:;",
- ])
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that partitions can be placed on a 512 byte sector boundary
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --offset 65s --fixed-size 99M --fstype=ext4\n" \
- "part /bar --ondisk hda --offset 102432 --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- _, partlns = self._get_wic_partitions(tempf.name, native_sysroot)
- self.assertEqual(partlns, [
- "1:32.5kiB:101408kiB:101376kiB:ext4:primary:;",
- "2:102432kiB:204832kiB:102400kiB:ext4:primary:;",
- ])
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that a partition can be placed immediately after a MSDOS partition table
- tempf.write("bootloader --ptable msdos\n" \
- "part / --source rootfs --ondisk hda --offset 1s --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- _, partlns = self._get_wic_partitions(tempf.name, native_sysroot)
- self.assertEqual(partlns, [
- "1:0.50kiB:102400kiB:102400kiB:ext4::;",
- ])
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that image creation fails if the partitions would overlap
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --offset 32 --fixed-size 100M --fstype=ext4\n" \
- "part /bar --ondisk hda --offset 102431 --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- p, _ = self._get_wic_partitions(tempf.name, ignore_status=True)
- self.assertNotEqual(p.status, 0, "wic exited successfully when an error was expected:\n%s" % p.output)
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- # Test that partitions are not allowed to overlap with the booloader
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --offset 8 --fixed-size 100M --fstype=ext4\n")
- tempf.flush()
- p, _ = self._get_wic_partitions(tempf.name, ignore_status=True)
- self.assertNotEqual(p.status, 0, "wic exited successfully when an error was expected:\n%s" % p.output)
- def test_extra_space(self):
- native_sysroot = get_bb_var("RECIPE_SYSROOT_NATIVE", "wic-tools")
- with NamedTemporaryFile("w", suffix=".wks") as tempf:
- tempf.write("bootloader --ptable gpt\n" \
- "part / --source rootfs --ondisk hda --extra-space 200M --fstype=ext4\n")
- tempf.flush()
- _, partlns = self._get_wic_partitions(tempf.name, native_sysroot)
- self.assertEqual(len(partlns), 1)
- size = partlns[0].split(':')[3]
- self.assertRegex(size, r'^[0-9]+kiB$')
- size = int(size[:-3])
- self.assertGreaterEqual(size, 204800)
- # TODO this test could also work on aarch64
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_rawcopy_plugin_qemu(self):
- """Test rawcopy plugin in qemu"""
- # build ext4 and then use it for a wic image
- config = 'IMAGE_FSTYPES = "ext4"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- image_link_name = get_bb_var('IMAGE_LINK_NAME', 'core-image-minimal')
- self.remove_config(config)
- config = 'IMAGE_FSTYPES = "wic"\n' \
- 'IMAGE_LINK_NAME_CORE_IMAGE_MINIMAL = "%s"\n'\
- 'WKS_FILE = "test_rawcopy_plugin.wks.in"\n'\
- % image_link_name
- self.append_config(config)
- bitbake('core-image-minimal-mtdutils')
- self.remove_config(config)
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-minimal-mtdutils') or ""
- with runqemu('core-image-minimal-mtdutils', ssh=False,
- runqemuparams='%s nographic' % (runqemu_params), image_fstype='wic') as qemu:
- cmd = "grep sda. /proc/partitions |wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '2')
- def _rawcopy_plugin(self, fstype):
- """Test rawcopy plugin"""
- image = 'core-image-minimal'
- bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_LINK_NAME'], image)
- params = ',unpack' if fstype.endswith('.gz') else ''
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.write('part / --source rawcopy --sourceparams="file=%s.%s%s"\n'\
- % (bb_vars['IMAGE_LINK_NAME'], fstype, params))
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, image, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- self.assertEqual(1, len(out))
- def test_rawcopy_plugin(self):
- config = 'IMAGE_FSTYPES = "ext4"\n'
- self.append_config(config)
- self.assertEqual(0, bitbake('core-image-minimal').status)
- self.remove_config(config)
- self._rawcopy_plugin('ext4')
- def test_rawcopy_plugin_unpack(self):
- fstype = 'ext4.gz'
- config = 'IMAGE_FSTYPES = "%s"\n' % fstype
- self.append_config(config)
- self.assertEqual(0, bitbake('core-image-minimal').status)
- self.remove_config(config)
- self._rawcopy_plugin(fstype)
- def test_empty_plugin(self):
- """Test empty plugin"""
- config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_empty_plugin.wks"\n'
- self.append_config(config)
- image = 'core-image-minimal'
- bitbake(image)
- self.remove_config(config)
- bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_LINK_NAME'], image)
- image_path = os.path.join(bb_vars['DEPLOY_DIR_IMAGE'], '%s.wic' % bb_vars['IMAGE_LINK_NAME'])
- self.assertTrue(os.path.exists(image_path), msg="Image file %s wasn't generated as expected" % image_path)
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # Fstype column from 'wic ls' should be empty for the second partition
- # as listed in test_empty_plugin.wks
- result = runCmd("wic ls %s -n %s | awk -F ' ' '{print $1 \" \" $5}' | grep '^2' | wc -w" % (image_path, sysroot))
- self.assertEqual('1', result.output)
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_biosplusefi_plugin_qemu(self):
- """Test biosplusefi plugin in qemu"""
- config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_biosplusefi_plugin.wks"\nMACHINE_FEATURES:append = " efi"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- self.remove_config(config)
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-minimal') or ""
- with runqemu('core-image-minimal', ssh=False,
- runqemuparams='%s nographic' % (runqemu_params), image_fstype='wic') as qemu:
- # Check that we have ONLY two /dev/sda* partitions (/boot and /)
- cmd = "grep sda. /proc/partitions | wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '2')
- # Check that /dev/sda1 is /boot and that either /dev/root OR /dev/sda2 is /
- cmd = "mount | grep '^/dev/' | cut -f1,3 -d ' ' | egrep -c -e '/dev/sda1 /boot' -e '/dev/root /|/dev/sda2 /'"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '2')
- # Check that /boot has EFI bootx64.efi (required for EFI)
- cmd = "ls /boot/EFI/BOOT/bootx64.efi | wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '1')
- # Check that "BOOTABLE" flag is set on boot partition (required for PC-Bios)
- # Trailing "cat" seems to be required; otherwise run_serial() sends back echo of the input command
- cmd = "fdisk -l /dev/sda | grep /dev/sda1 | awk {print'$2'} | cat"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '*')
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- def test_biosplusefi_plugin(self):
- """Test biosplusefi plugin"""
- # Wic generation below may fail depending on the order of the unittests
- # This is because bootimg-pcbios (that bootimg-biosplusefi uses) generate its MBR inside STAGING_DATADIR directory
- # which may or may not exists depending on what was built already
- # If an image hasn't been built yet, directory ${STAGING_DATADIR}/syslinux won't exists and _get_bootimg_dir()
- # will raise with "Couldn't find correct bootimg_dir"
- # The easiest way to work-around this issue is to make sure we already built an image here, hence the bitbake call
- config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "test_biosplusefi_plugin.wks"\nMACHINE_FEATURES:append = " efi"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- self.remove_config(config)
- img = 'core-image-minimal'
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(['part /boot --active --source bootimg-biosplusefi --sourceparams="loader=grub-efi"\n',
- 'part / --source rootfs --fstype=ext4 --align 1024 --use-uuid\n'\
- 'bootloader --timeout=0 --append="console=ttyS0,115200n8"\n'])
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*.direct" % wksname))
- self.assertEqual(1, len(out))
- @skipIfNotArch(['i586', 'i686', 'x86_64', 'aarch64'])
- def test_uefi_kernel(self):
- """ Test uefi-kernel in wic """
- config = 'IMAGE_EFI_BOOT_FILES="/etc/fstab;testfile"\nIMAGE_FSTYPES = "wic"\nWKS_FILE = "test_uefikernel.wks"\nMACHINE_FEATURES:append = " efi"\n'
- self.append_config(config)
- bitbake('core-image-minimal')
- self.remove_config(config)
- img = 'core-image-minimal'
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(['part /boot --source bootimg-efi --sourceparams="loader=uefi-kernel"\n'
- 'part / --source rootfs --fstype=ext4 --align 1024 --use-uuid\n'\
- 'bootloader --timeout=0 --append="console=ttyS0,115200n8"\n'])
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*.direct" % wksname))
- self.assertEqual(1, len(out))
- # TODO this test could also work on aarch64
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_efi_plugin_unified_kernel_image_qemu(self):
- """Test Unified Kernel Image feature in qemu without systemd in initramfs or rootfs"""
- config = """
- # efi firmware must load systemd-boot, not grub
- EFI_PROVIDER = "systemd-boot"
- # image format must be wic, needs esp partition for firmware etc
- IMAGE_FSTYPES:pn-core-image-base:append = " wic"
- WKS_FILE = "test_efi_plugin.wks"
- # efi, uki and systemd features must be enabled
- MACHINE_FEATURES:append = " efi"
- IMAGE_CLASSES:append:pn-core-image-base = " uki"
- # uki embeds also an initrd, no systemd or udev
- INITRAMFS_IMAGE = "core-image-initramfs-boot"
- # runqemu must not load kernel separately, it's in the uki
- QB_KERNEL_ROOT = ""
- QB_DEFAULT_KERNEL = "none"
- # boot command line provided via uki, not via bootloader
- UKI_CMDLINE = "rootwait root=LABEL=root console=${KERNEL_CONSOLE}"
- """
- self.append_config(config)
- bitbake('core-image-base ovmf')
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-base') or ""
- uki_filename = get_bb_var('UKI_FILENAME', 'core-image-base')
- self.remove_config(config)
- with runqemu('core-image-base', ssh=False,
- runqemuparams='%s nographic ovmf' % (runqemu_params), image_fstype='wic') as qemu:
- # Check that /boot has EFI boot*.efi (required for EFI)
- cmd = "ls /boot/EFI/BOOT/boot*.efi | wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '1')
- # Check that /boot has EFI/Linux/${UKI_FILENAME} (required for Unified Kernel Images auto detection)
- cmd = "ls /boot/EFI/Linux/%s | wc -l" % (uki_filename)
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '1')
- # Check that /boot doesn't have loader/entries/boot.conf (Unified Kernel Images are auto detected by the bootloader)
- cmd = "ls /boot/loader/entries/boot.conf 2&>/dev/null | wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '0')
- @skipIfNotArch(['aarch64'])
- @OETestTag("runqemu")
- def test_efi_plugin_plain_systemd_boot_qemu_aarch64(self):
- """Test plain systemd-boot in qemu with systemd"""
- config = """
- INIT_MANAGER = "systemd"
- EFI_PROVIDER = "systemd-boot"
- # image format must be wic, needs esp partition for firmware etc
- IMAGE_FSTYPES:pn-core-image-base:append = " wic"
- WKS_FILE = "test_efi_plugin_plain_systemd-boot.wks"
- INITRAMFS_IMAGE = "core-image-initramfs-boot"
- # to configure runqemu
- IMAGE_CLASSES += "qemuboot"
- # u-boot efi firmware
- QB_DEFAULT_BIOS = "u-boot.bin"
- # need to use virtio, scsi not supported by u-boot by default
- QB_DRIVE_TYPE = "/dev/vd"
- # disable kvm, breaks boot
- QEMU_USE_KVM = ""
- IMAGE_CLASSES:remove = 'testimage'
- """
- self.append_config(config)
- bitbake('core-image-base u-boot')
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-base') or ""
- with runqemu('core-image-base', ssh=False,
- runqemuparams='%s nographic' % (runqemu_params), image_fstype='wic') as qemu:
- # Check that /boot has EFI boot*.efi (required for EFI)
- cmd = "ls /boot/EFI/BOOT/boot*.efi | wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '1')
- # Check that boot.conf exists
- cmd = "cat /boot/loader/entries/boot.conf"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.remove_config(config)
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_efi_plugin_plain_systemd_boot_qemu_x86(self):
- """Test plain systemd-boot to systemd in qemu"""
- config = """
- INIT_MANAGER = "systemd"
- EFI_PROVIDER = "systemd-boot"
- # image format must be wic, needs esp partition for firmware etc
- IMAGE_FSTYPES:pn-core-image-base:append = " wic"
- WKS_FILE = "test_efi_plugin_plain_systemd-boot.wks"
- INITRAMFS_IMAGE = "core-image-initramfs-boot"
- """
- self.append_config(config)
- bitbake('core-image-base ovmf')
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-base') or ""
- self.remove_config(config)
- with runqemu('core-image-base', ssh=False,
- runqemuparams='%s nographic ovmf' % (runqemu_params), image_fstype='wic') as qemu:
- # Check that /boot has EFI boot*.efi (required for EFI)
- cmd = "ls /boot/EFI/BOOT/boot*.efi | wc -l"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- self.assertEqual(output, '1')
- # Check that boot.conf exists
- cmd = "cat /boot/loader/entries/boot.conf"
- status, output = qemu.run_serial(cmd)
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- def test_fs_types(self):
- """Test filesystem types for empty and not empty partitions"""
- img = 'core-image-minimal'
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(['part ext2 --fstype ext2 --source rootfs\n',
- 'part btrfs --fstype btrfs --source rootfs --size 40M\n',
- 'part squash --fstype squashfs --source rootfs\n',
- 'part swap --fstype swap --size 1M\n',
- 'part emptyvfat --fstype vfat --size 1M\n',
- 'part emptymsdos --fstype msdos --size 1M\n',
- 'part emptyext2 --fstype ext2 --size 1M\n',
- 'part emptybtrfs --fstype btrfs --size 150M\n'])
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- self.assertEqual(1, len(out))
- def test_kickstart_parser(self):
- """Test wks parser options"""
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(['part / --fstype ext3 --source rootfs --system-id 0xFF '\
- '--overhead-factor 1.2 --size 100k\n'])
- wks.flush()
- cmd = "wic create %s -e core-image-minimal -o %s" % (wks.name, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- self.assertEqual(1, len(out))
- def test_image_bootpart_globbed(self):
- """Test globbed sources with image-bootpart plugin"""
- img = "core-image-minimal"
- cmd = "wic create sdimage-bootpart -e %s -o %s" % (img, self.resultdir)
- config = 'IMAGE_BOOT_FILES = "%s*"' % get_bb_var('KERNEL_IMAGETYPE', img)
- self.append_config(config)
- runCmd(cmd)
- self.remove_config(config)
- self.assertEqual(1, len(glob(os.path.join(self.resultdir, "sdimage-bootpart-*direct"))))
- def test_sparse_copy(self):
- """Test sparse_copy with FIEMAP and SEEK_HOLE filemap APIs"""
- libpath = os.path.join(self.td['COREBASE'], 'scripts', 'lib', 'wic')
- sys.path.insert(0, libpath)
- from filemap import FilemapFiemap, FilemapSeek, sparse_copy, ErrorNotSupp
- with NamedTemporaryFile("w", suffix=".wic-sparse") as sparse:
- src_name = sparse.name
- src_size = 1024 * 10
- sparse.truncate(src_size)
- # write one byte to the file
- with open(src_name, 'r+b') as sfile:
- sfile.seek(1024 * 4)
- sfile.write(b'\x00')
- dest = sparse.name + '.out'
- # copy src file to dest using different filemap APIs
- for api in (FilemapFiemap, FilemapSeek, None):
- if os.path.exists(dest):
- os.unlink(dest)
- try:
- sparse_copy(sparse.name, dest, api=api)
- except ErrorNotSupp:
- continue # skip unsupported API
- dest_stat = os.stat(dest)
- self.assertEqual(dest_stat.st_size, src_size)
- # 8 blocks is 4K (physical sector size)
- self.assertEqual(dest_stat.st_blocks, 8)
- os.unlink(dest)
- def test_mkfs_extraopts(self):
- """Test wks option --mkfs-extraopts for empty and not empty partitions"""
- img = 'core-image-minimal'
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(
- ['part ext2 --fstype ext2 --source rootfs --mkfs-extraopts "-D -F -i 8192"\n',
- "part btrfs --fstype btrfs --source rootfs --size 40M --mkfs-extraopts='--quiet'\n",
- 'part squash --fstype squashfs --source rootfs --mkfs-extraopts "-no-sparse -b 4096"\n',
- 'part emptyvfat --fstype vfat --size 1M --mkfs-extraopts "-S 1024 -s 64"\n',
- 'part emptymsdos --fstype msdos --size 1M --mkfs-extraopts "-S 1024 -s 64"\n',
- 'part emptyext2 --fstype ext2 --size 1M --mkfs-extraopts "-D -F -i 8192"\n',
- 'part emptybtrfs --fstype btrfs --size 100M --mkfs-extraopts "--mixed -K"\n'])
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- out = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- self.assertEqual(1, len(out))
- @skipIfNotArch(['i586', 'i686', 'x86_64'])
- @OETestTag("runqemu")
- def test_expand_mbr_image(self):
- """Test wic write --expand command for mbr image"""
- # build an image
- config = 'IMAGE_FSTYPES = "wic"\nWKS_FILE = "directdisk.wks"\n'
- self.append_config(config)
- image = 'core-image-minimal'
- bitbake(image)
- # get path to the image
- bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_LINK_NAME'], image)
- image_path = os.path.join(bb_vars['DEPLOY_DIR_IMAGE'], '%s.wic' % bb_vars['IMAGE_LINK_NAME'])
- self.remove_config(config)
- try:
- # expand image to 1G
- new_image_path = None
- with NamedTemporaryFile(mode='wb', suffix='.wic.exp',
- dir=bb_vars['DEPLOY_DIR_IMAGE'], delete=False) as sparse:
- sparse.truncate(1024 ** 3)
- new_image_path = sparse.name
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- cmd = "wic write -n %s --expand 1:0 %s %s" % (sysroot, image_path, new_image_path)
- runCmd(cmd)
- # check if partitions are expanded
- orig = runCmd("wic ls %s -n %s" % (image_path, sysroot))
- exp = runCmd("wic ls %s -n %s" % (new_image_path, sysroot))
- orig_sizes = [int(line.split()[3]) for line in orig.output.split('\n')[1:]]
- exp_sizes = [int(line.split()[3]) for line in exp.output.split('\n')[1:]]
- self.assertEqual(orig_sizes[0], exp_sizes[0]) # first partition is not resized
- self.assertTrue(orig_sizes[1] < exp_sizes[1], msg="Parition size wasn't enlarged (%s vs %s)" % (orig_sizes[1], exp_sizes[1]))
- # Check if all free space is partitioned
- result = runCmd("%s/usr/sbin/sfdisk -F %s" % (sysroot, new_image_path))
- self.assertIn("0 B, 0 bytes, 0 sectors", result.output)
- os.rename(image_path, image_path + '.bak')
- os.rename(new_image_path, image_path)
- runqemu_params = get_bb_var('TEST_RUNQEMUPARAMS', 'core-image-minimal') or ""
- with runqemu('core-image-minimal', ssh=False, runqemuparams='%s nographic' % (runqemu_params)) as qemu:
- cmd = "ls /etc/"
- status, output = qemu.run_serial('true')
- self.assertEqual(1, status, 'Failed to run command "%s": %s' % (cmd, output))
- finally:
- if os.path.exists(new_image_path):
- os.unlink(new_image_path)
- if os.path.exists(image_path + '.bak'):
- os.rename(image_path + '.bak', image_path)
- def test_gpt_partition_name(self):
- """Test --part-name argument to set partition name in GPT table"""
- config = 'IMAGE_FSTYPES += "wic"\nWKS_FILE = "test_gpt_partition_name.wks"\n'
- self.append_config(config)
- image = 'core-image-minimal'
- bitbake(image)
- self.remove_config(config)
- deploy_dir = get_bb_var('DEPLOY_DIR_IMAGE')
- bb_vars = get_bb_vars(['DEPLOY_DIR_IMAGE', 'IMAGE_LINK_NAME'], image)
- image_path = os.path.join(bb_vars['DEPLOY_DIR_IMAGE'], '%s.wic' % bb_vars['IMAGE_LINK_NAME'])
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # Image is created
- self.assertTrue(os.path.exists(image_path), "image file %s doesn't exist" % image_path)
- # Check the names of the three partitions
- # as listed in test_gpt_partition_name.wks
- result = runCmd("%s/usr/sbin/sfdisk --part-label %s 1" % (sysroot, image_path))
- self.assertEqual('boot-A', result.output)
- result = runCmd("%s/usr/sbin/sfdisk --part-label %s 2" % (sysroot, image_path))
- self.assertEqual('root-A', result.output)
- # When the --part-name is not defined, the partition name is equal to the --label
- result = runCmd("%s/usr/sbin/sfdisk --part-label %s 3" % (sysroot, image_path))
- self.assertEqual('ext-space', result.output)
- def test_empty_zeroize_plugin(self):
- img = 'core-image-minimal'
- expected_size = [ 1024*1024, # 1M
- 512*1024, # 512K
- 2*1024*1024] # 2M
- # Check combination of sourceparams
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(
- ['part empty --source empty --sourceparams="fill" --ondisk sda --fixed-size 1M\n',
- 'part empty --source empty --sourceparams="size=512K" --ondisk sda --size 1M --align 1024\n',
- 'part empty --source empty --sourceparams="size=2048k,bs=512K" --ondisk sda --size 4M --align 1024\n'
- ])
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- runCmd(cmd)
- wksname = os.path.splitext(os.path.basename(wks.name))[0]
- wicout = glob(os.path.join(self.resultdir, "%s-*direct" % wksname))
- # Skip the complete image and just look at the single partitions
- for idx, value in enumerate(wicout[1:]):
- self.logger.info(wicout[idx])
- # Check if partitions are actually zeroized
- with open(wicout[idx], mode="rb") as fd:
- ba = bytearray(fd.read())
- for b in ba:
- self.assertEqual(b, 0)
- self.assertEqual(expected_size[idx], os.path.getsize(wicout[idx]))
- # Check inconsistancy check between "fill" and "--size" parameter
- with NamedTemporaryFile("w", suffix=".wks") as wks:
- wks.writelines(['part empty --source empty --sourceparams="fill" --ondisk sda --size 1M\n'])
- wks.flush()
- cmd = "wic create %s -e %s -o %s" % (wks.name, img, self.resultdir)
- result = runCmd(cmd, ignore_status=True)
- self.assertIn("Source parameter 'fill' only works with the '--fixed-size' option, exiting.", result.output)
- self.assertNotEqual(0, result.status)
- class ModifyTests(WicTestCase):
- def test_wic_ls(self):
- """Test listing image content using 'wic ls'"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- images = glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # list partitions
- result = runCmd("wic ls %s -n %s" % (images[0], sysroot))
- self.assertEqual(3, len(result.output.split('\n')))
- # list directory content of the first partition
- result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot))
- self.assertEqual(6, len(result.output.split('\n')))
- def test_wic_cp(self):
- """Test copy files and directories to the the wic image."""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- images = glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # list directory content of the first partition
- result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot))
- self.assertEqual(6, len(result.output.split('\n')))
- with NamedTemporaryFile("w", suffix=".wic-cp") as testfile:
- testfile.write("test")
- # copy file to the partition
- runCmd("wic cp %s %s:1/ -n %s" % (testfile.name, images[0], sysroot))
- # check if file is there
- result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot))
- self.assertEqual(7, len(result.output.split('\n')))
- self.assertIn(os.path.basename(testfile.name), result.output)
- # prepare directory
- testdir = os.path.join(self.resultdir, 'wic-test-cp-dir')
- testsubdir = os.path.join(testdir, 'subdir')
- os.makedirs(os.path.join(testsubdir))
- copy(testfile.name, testdir)
- # copy directory to the partition
- runCmd("wic cp %s %s:1/ -n %s" % (testdir, images[0], sysroot))
- # check if directory is there
- result = runCmd("wic ls %s:1/ -n %s" % (images[0], sysroot))
- self.assertEqual(8, len(result.output.split('\n')))
- self.assertIn(os.path.basename(testdir), result.output)
- # copy the file from the partition and check if it success
- dest = '%s-cp' % testfile.name
- runCmd("wic cp %s:1/%s %s -n %s" % (images[0],
- os.path.basename(testfile.name), dest, sysroot))
- self.assertTrue(os.path.exists(dest), msg="File %s wasn't generated as expected" % dest)
- def test_wic_rm(self):
- """Test removing files and directories from the the wic image."""
- runCmd("wic create mkefidisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- images = glob(os.path.join(self.resultdir, "mkefidisk-*.direct"))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # Not bulletproof but hopefully sufficient
- kerneltype = get_bb_var('KERNEL_IMAGETYPE', 'virtual/kernel')
- # list directory content of the first partition
- result = runCmd("wic ls %s:1 -n %s" % (images[0], sysroot))
- self.assertIn('\n%s ' % kerneltype.upper(), result.output)
- self.assertIn('\nEFI <DIR> ', result.output)
- # remove file. EFI partitions are case-insensitive so exercise that too
- runCmd("wic rm %s:1/%s -n %s" % (images[0], kerneltype.lower(), sysroot))
- # remove directory
- runCmd("wic rm %s:1/efi -n %s" % (images[0], sysroot))
- # check if they're removed
- result = runCmd("wic ls %s:1 -n %s" % (images[0], sysroot))
- self.assertNotIn('\n%s ' % kerneltype.upper(), result.output)
- self.assertNotIn('\nEFI <DIR> ', result.output)
- def test_wic_ls_ext(self):
- """Test listing content of the ext partition using 'wic ls'"""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- images = glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # list directory content of the second ext4 partition
- result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot))
- self.assertTrue(set(['bin', 'home', 'proc', 'usr', 'var', 'dev', 'lib', 'sbin']).issubset(
- set(line.split()[-1] for line in result.output.split('\n') if line)), msg="Expected directories not present %s" % result.output)
- def test_wic_cp_ext(self):
- """Test copy files and directories to the ext partition."""
- runCmd("wic create wictestdisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- images = glob(os.path.join(self.resultdir, "wictestdisk-*.direct"))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # list directory content of the ext4 partition
- result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot))
- dirs = set(line.split()[-1] for line in result.output.split('\n') if line)
- self.assertTrue(set(['bin', 'home', 'proc', 'usr', 'var', 'dev', 'lib', 'sbin']).issubset(dirs), msg="Expected directories not present %s" % dirs)
- with NamedTemporaryFile("w", suffix=".wic-cp") as testfile:
- testfile.write("test")
- # copy file to the partition
- runCmd("wic cp %s %s:2/ -n %s" % (testfile.name, images[0], sysroot))
- # check if file is there
- result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot))
- newdirs = set(line.split()[-1] for line in result.output.split('\n') if line)
- self.assertEqual(newdirs.difference(dirs), set([os.path.basename(testfile.name)]))
- # check if the file to copy is in the partition
- result = runCmd("wic ls %s:2/etc/ -n %s" % (images[0], sysroot))
- self.assertIn('fstab', [line.split()[-1] for line in result.output.split('\n') if line])
- # copy file from the partition, replace the temporary file content with it and
- # check for the file size to validate the copy
- runCmd("wic cp %s:2/etc/fstab %s -n %s" % (images[0], testfile.name, sysroot))
- self.assertTrue(os.stat(testfile.name).st_size > 0, msg="Filesize not as expected %s" % os.stat(testfile.name).st_size)
- def test_wic_rm_ext(self):
- """Test removing files from the ext partition."""
- runCmd("wic create mkefidisk "
- "--image-name=core-image-minimal "
- "-D -o %s" % self.resultdir)
- images = glob(os.path.join(self.resultdir, "mkefidisk-*.direct"))
- self.assertEqual(1, len(images))
- sysroot = get_bb_var('RECIPE_SYSROOT_NATIVE', 'wic-tools')
- # list directory content of the /etc directory on ext4 partition
- result = runCmd("wic ls %s:2/etc/ -n %s" % (images[0], sysroot))
- self.assertIn('fstab', [line.split()[-1] for line in result.output.split('\n') if line])
- # remove file
- runCmd("wic rm %s:2/etc/fstab -n %s" % (images[0], sysroot))
- # check if it's removed
- result = runCmd("wic ls %s:2/etc/ -n %s" % (images[0], sysroot))
- self.assertNotIn('fstab', [line.split()[-1] for line in result.output.split('\n') if line])
- # remove non-empty directory
- runCmd("wic rm -r %s:2/etc/ -n %s" % (images[0], sysroot))
- # check if it's removed
- result = runCmd("wic ls %s:2/ -n %s" % (images[0], sysroot))
- self.assertNotIn('etc', [line.split()[-1] for line in result.output.split('\n') if line])
|