| # SPDX-License-Identifier: GPL-2.0+ |
| # Copyright (c) 2018, Linaro Limited |
| # Author: Takahiro Akashi <takahiro.akashi@linaro.org> |
| |
| import os |
| import os.path |
| import pytest |
| import re |
| from subprocess import call, check_call, check_output, CalledProcessError |
| from fstest_defs import * |
| import u_boot_utils as util |
| |
| supported_fs_basic = ['fat16', 'fat32', 'ext4'] |
| supported_fs_ext = ['fat16', 'fat32'] |
| supported_fs_mkdir = ['fat16', 'fat32'] |
| supported_fs_unlink = ['fat16', 'fat32'] |
| supported_fs_symlink = ['ext4'] |
| |
| # |
| # Filesystem test specific setup |
| # |
| def pytest_addoption(parser): |
| """Enable --fs-type option. |
| |
| See pytest_configure() about how it works. |
| |
| Args: |
| parser: Pytest command-line parser. |
| |
| Returns: |
| Nothing. |
| """ |
| parser.addoption('--fs-type', action='append', default=None, |
| help='Targeting Filesystem Types') |
| |
| def pytest_configure(config): |
| """Restrict a file system(s) to be tested. |
| |
| A file system explicitly named with --fs-type option is selected |
| if it belongs to a default supported_fs_xxx list. |
| Multiple options can be specified. |
| |
| Args: |
| config: Pytest configuration. |
| |
| Returns: |
| Nothing. |
| """ |
| global supported_fs_basic |
| global supported_fs_ext |
| global supported_fs_mkdir |
| global supported_fs_unlink |
| global supported_fs_symlink |
| |
| def intersect(listA, listB): |
| return [x for x in listA if x in listB] |
| |
| supported_fs = config.getoption('fs_type') |
| if supported_fs: |
| print('*** FS TYPE modified: %s' % supported_fs) |
| supported_fs_basic = intersect(supported_fs, supported_fs_basic) |
| supported_fs_ext = intersect(supported_fs, supported_fs_ext) |
| supported_fs_mkdir = intersect(supported_fs, supported_fs_mkdir) |
| supported_fs_unlink = intersect(supported_fs, supported_fs_unlink) |
| supported_fs_symlink = intersect(supported_fs, supported_fs_symlink) |
| |
| def pytest_generate_tests(metafunc): |
| """Parametrize fixtures, fs_obj_xxx |
| |
| Each fixture will be parametrized with a corresponding support_fs_xxx |
| list. |
| |
| Args: |
| metafunc: Pytest test function. |
| |
| Returns: |
| Nothing. |
| """ |
| if 'fs_obj_basic' in metafunc.fixturenames: |
| metafunc.parametrize('fs_obj_basic', supported_fs_basic, |
| indirect=True, scope='module') |
| if 'fs_obj_ext' in metafunc.fixturenames: |
| metafunc.parametrize('fs_obj_ext', supported_fs_ext, |
| indirect=True, scope='module') |
| if 'fs_obj_mkdir' in metafunc.fixturenames: |
| metafunc.parametrize('fs_obj_mkdir', supported_fs_mkdir, |
| indirect=True, scope='module') |
| if 'fs_obj_unlink' in metafunc.fixturenames: |
| metafunc.parametrize('fs_obj_unlink', supported_fs_unlink, |
| indirect=True, scope='module') |
| if 'fs_obj_symlink' in metafunc.fixturenames: |
| metafunc.parametrize('fs_obj_symlink', supported_fs_symlink, |
| indirect=True, scope='module') |
| |
| # |
| # Helper functions |
| # |
| def fstype_to_ubname(fs_type): |
| """Convert a file system type to an U-boot specific string |
| |
| A generated string can be used as part of file system related commands |
| or a config name in u-boot. Currently fat16 and fat32 are handled |
| specifically. |
| |
| Args: |
| fs_type: File system type. |
| |
| Return: |
| A corresponding string for file system type. |
| """ |
| if re.match('fat', fs_type): |
| return 'fat' |
| else: |
| return fs_type |
| |
| def check_ubconfig(config, fs_type): |
| """Check whether a file system is enabled in u-boot configuration. |
| |
| This function is assumed to be called in a fixture function so that |
| the whole test cases will be skipped if a given file system is not |
| enabled. |
| |
| Args: |
| fs_type: File system type. |
| |
| Return: |
| Nothing. |
| """ |
| if not config.buildconfig.get('config_cmd_%s' % fs_type, None): |
| pytest.skip('.config feature "CMD_%s" not enabled' % fs_type.upper()) |
| if not config.buildconfig.get('config_%s_write' % fs_type, None): |
| pytest.skip('.config feature "%s_WRITE" not enabled' |
| % fs_type.upper()) |
| |
| def mk_fs(config, fs_type, size, id): |
| """Create a file system volume. |
| |
| Args: |
| fs_type: File system type. |
| size: Size of file system in MiB. |
| id: Prefix string of volume's file name. |
| |
| Return: |
| Nothing. |
| """ |
| fs_img = '%s.%s.img' % (id, fs_type) |
| fs_img = config.persistent_data_dir + '/' + fs_img |
| |
| if fs_type == 'fat16': |
| mkfs_opt = '-F 16' |
| elif fs_type == 'fat32': |
| mkfs_opt = '-F 32' |
| else: |
| mkfs_opt = '' |
| |
| if re.match('fat', fs_type): |
| fs_lnxtype = 'vfat' |
| else: |
| fs_lnxtype = fs_type |
| |
| count = (size + 1048576 - 1) / 1048576 |
| |
| # Some distributions do not add /sbin to the default PATH, where mkfs lives |
| if '/sbin' not in os.environ["PATH"].split(os.pathsep): |
| os.environ["PATH"] += os.pathsep + '/sbin' |
| |
| try: |
| check_call('rm -f %s' % fs_img, shell=True) |
| check_call('dd if=/dev/zero of=%s bs=1M count=%d' |
| % (fs_img, count), shell=True) |
| check_call('mkfs.%s %s %s' |
| % (fs_lnxtype, mkfs_opt, fs_img), shell=True) |
| if fs_type == 'ext4': |
| sb_content = check_output('tune2fs -l %s' % fs_img, shell=True).decode() |
| if 'metadata_csum' in sb_content: |
| check_call('tune2fs -O ^metadata_csum %s' % fs_img, shell=True) |
| return fs_img |
| except CalledProcessError: |
| call('rm -f %s' % fs_img, shell=True) |
| raise |
| |
| # from test/py/conftest.py |
| def tool_is_in_path(tool): |
| """Check whether a given command is available on host. |
| |
| Args: |
| tool: Command name. |
| |
| Return: |
| True if available, False if not. |
| """ |
| for path in os.environ['PATH'].split(os.pathsep): |
| fn = os.path.join(path, tool) |
| if os.path.isfile(fn) and os.access(fn, os.X_OK): |
| return True |
| return False |
| |
| fuse_mounted = False |
| |
| def mount_fs(fs_type, device, mount_point): |
| """Mount a volume. |
| |
| Args: |
| fs_type: File system type. |
| device: Volume's file name. |
| mount_point: Mount point. |
| |
| Return: |
| Nothing. |
| """ |
| global fuse_mounted |
| |
| try: |
| check_call('guestmount --pid-file guestmount.pid -a %s -m /dev/sda %s' |
| % (device, mount_point), shell=True) |
| fuse_mounted = True |
| return |
| except CalledProcessError: |
| fuse_mounted = False |
| |
| mount_opt = 'loop,rw' |
| if re.match('fat', fs_type): |
| mount_opt += ',umask=0000' |
| |
| check_call('sudo mount -o %s %s %s' |
| % (mount_opt, device, mount_point), shell=True) |
| |
| # may not be effective for some file systems |
| check_call('sudo chmod a+rw %s' % mount_point, shell=True) |
| |
| def umount_fs(mount_point): |
| """Unmount a volume. |
| |
| Args: |
| mount_point: Mount point. |
| |
| Return: |
| Nothing. |
| """ |
| if fuse_mounted: |
| call('sync') |
| call('guestunmount %s' % mount_point, shell=True) |
| |
| try: |
| with open("guestmount.pid", "r") as pidfile: |
| pid = int(pidfile.read()) |
| util.waitpid(pid, kill=True) |
| os.remove("guestmount.pid") |
| |
| except FileNotFoundError: |
| pass |
| |
| else: |
| call('sudo umount %s' % mount_point, shell=True) |
| |
| # |
| # Fixture for basic fs test |
| # derived from test/fs/fs-test.sh |
| # |
| @pytest.fixture() |
| def fs_obj_basic(request, u_boot_config): |
| """Set up a file system to be used in basic fs test. |
| |
| Args: |
| request: Pytest request object. |
| u_boot_config: U-boot configuration. |
| |
| Return: |
| A fixture for basic fs test, i.e. a triplet of file system type, |
| volume file name and a list of MD5 hashes. |
| """ |
| fs_type = request.param |
| fs_img = '' |
| |
| fs_ubtype = fstype_to_ubname(fs_type) |
| check_ubconfig(u_boot_config, fs_ubtype) |
| |
| mount_dir = u_boot_config.persistent_data_dir + '/mnt' |
| |
| small_file = mount_dir + '/' + SMALL_FILE |
| big_file = mount_dir + '/' + BIG_FILE |
| |
| try: |
| |
| # 3GiB volume |
| fs_img = mk_fs(u_boot_config, fs_type, 0xc0000000, '3GB') |
| except CalledProcessError as err: |
| pytest.skip('Creating failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| return |
| |
| try: |
| check_call('mkdir -p %s' % mount_dir, shell=True) |
| except CalledProcessError as err: |
| pytest.skip('Preparing mount folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Mount the image so we can populate it. |
| mount_fs(fs_type, fs_img, mount_dir) |
| except CalledProcessError as err: |
| pytest.skip('Mounting to folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Create a subdirectory. |
| check_call('mkdir %s/SUBDIR' % mount_dir, shell=True) |
| |
| # Create big file in this image. |
| # Note that we work only on the start 1MB, couple MBs in the 2GB range |
| # and the last 1 MB of the huge 2.5GB file. |
| # So, just put random values only in those areas. |
| check_call('dd if=/dev/urandom of=%s bs=1M count=1' |
| % big_file, shell=True) |
| check_call('dd if=/dev/urandom of=%s bs=1M count=2 seek=2047' |
| % big_file, shell=True) |
| check_call('dd if=/dev/urandom of=%s bs=1M count=1 seek=2499' |
| % big_file, shell=True) |
| |
| # Create a small file in this image. |
| check_call('dd if=/dev/urandom of=%s bs=1M count=1' |
| % small_file, shell=True) |
| |
| # Delete the small file copies which possibly are written as part of a |
| # previous test. |
| # check_call('rm -f "%s.w"' % MB1, shell=True) |
| # check_call('rm -f "%s.w2"' % MB1, shell=True) |
| |
| # Generate the md5sums of reads that we will test against small file |
| out = check_output( |
| 'dd if=%s bs=1M skip=0 count=1 2> /dev/null | md5sum' |
| % small_file, shell=True).decode() |
| md5val = [ out.split()[0] ] |
| |
| # Generate the md5sums of reads that we will test against big file |
| # One from beginning of file. |
| out = check_output( |
| 'dd if=%s bs=1M skip=0 count=1 2> /dev/null | md5sum' |
| % big_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| # One from end of file. |
| out = check_output( |
| 'dd if=%s bs=1M skip=2499 count=1 2> /dev/null | md5sum' |
| % big_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| # One from the last 1MB chunk of 2GB |
| out = check_output( |
| 'dd if=%s bs=1M skip=2047 count=1 2> /dev/null | md5sum' |
| % big_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| # One from the start 1MB chunk from 2GB |
| out = check_output( |
| 'dd if=%s bs=1M skip=2048 count=1 2> /dev/null | md5sum' |
| % big_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| # One 1MB chunk crossing the 2GB boundary |
| out = check_output( |
| 'dd if=%s bs=512K skip=4095 count=2 2> /dev/null | md5sum' |
| % big_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| except CalledProcessError as err: |
| pytest.skip('Setup failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| umount_fs(mount_dir) |
| return |
| else: |
| umount_fs(mount_dir) |
| yield [fs_ubtype, fs_img, md5val] |
| finally: |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| |
| # |
| # Fixture for extended fs test |
| # |
| @pytest.fixture() |
| def fs_obj_ext(request, u_boot_config): |
| """Set up a file system to be used in extended fs test. |
| |
| Args: |
| request: Pytest request object. |
| u_boot_config: U-boot configuration. |
| |
| Return: |
| A fixture for extended fs test, i.e. a triplet of file system type, |
| volume file name and a list of MD5 hashes. |
| """ |
| fs_type = request.param |
| fs_img = '' |
| |
| fs_ubtype = fstype_to_ubname(fs_type) |
| check_ubconfig(u_boot_config, fs_ubtype) |
| |
| mount_dir = u_boot_config.persistent_data_dir + '/mnt' |
| |
| min_file = mount_dir + '/' + MIN_FILE |
| tmp_file = mount_dir + '/tmpfile' |
| |
| try: |
| |
| # 128MiB volume |
| fs_img = mk_fs(u_boot_config, fs_type, 0x8000000, '128MB') |
| except CalledProcessError as err: |
| pytest.skip('Creating failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| return |
| |
| try: |
| check_call('mkdir -p %s' % mount_dir, shell=True) |
| except CalledProcessError as err: |
| pytest.skip('Preparing mount folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Mount the image so we can populate it. |
| mount_fs(fs_type, fs_img, mount_dir) |
| except CalledProcessError as err: |
| pytest.skip('Mounting to folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Create a test directory |
| check_call('mkdir %s/dir1' % mount_dir, shell=True) |
| |
| # Create a small file and calculate md5 |
| check_call('dd if=/dev/urandom of=%s bs=1K count=20' |
| % min_file, shell=True) |
| out = check_output( |
| 'dd if=%s bs=1K 2> /dev/null | md5sum' |
| % min_file, shell=True).decode() |
| md5val = [ out.split()[0] ] |
| |
| # Calculate md5sum of Test Case 4 |
| check_call('dd if=%s of=%s bs=1K count=20' |
| % (min_file, tmp_file), shell=True) |
| check_call('dd if=%s of=%s bs=1K seek=5 count=20' |
| % (min_file, tmp_file), shell=True) |
| out = check_output('dd if=%s bs=1K 2> /dev/null | md5sum' |
| % tmp_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| # Calculate md5sum of Test Case 5 |
| check_call('dd if=%s of=%s bs=1K count=20' |
| % (min_file, tmp_file), shell=True) |
| check_call('dd if=%s of=%s bs=1K seek=5 count=5' |
| % (min_file, tmp_file), shell=True) |
| out = check_output('dd if=%s bs=1K 2> /dev/null | md5sum' |
| % tmp_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| # Calculate md5sum of Test Case 7 |
| check_call('dd if=%s of=%s bs=1K count=20' |
| % (min_file, tmp_file), shell=True) |
| check_call('dd if=%s of=%s bs=1K seek=20 count=20' |
| % (min_file, tmp_file), shell=True) |
| out = check_output('dd if=%s bs=1K 2> /dev/null | md5sum' |
| % tmp_file, shell=True).decode() |
| md5val.append(out.split()[0]) |
| |
| check_call('rm %s' % tmp_file, shell=True) |
| except CalledProcessError: |
| pytest.skip('Setup failed for filesystem: ' + fs_type) |
| umount_fs(mount_dir) |
| return |
| else: |
| umount_fs(mount_dir) |
| yield [fs_ubtype, fs_img, md5val] |
| finally: |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| |
| # |
| # Fixture for mkdir test |
| # |
| @pytest.fixture() |
| def fs_obj_mkdir(request, u_boot_config): |
| """Set up a file system to be used in mkdir test. |
| |
| Args: |
| request: Pytest request object. |
| u_boot_config: U-boot configuration. |
| |
| Return: |
| A fixture for mkdir test, i.e. a duplet of file system type and |
| volume file name. |
| """ |
| fs_type = request.param |
| fs_img = '' |
| |
| fs_ubtype = fstype_to_ubname(fs_type) |
| check_ubconfig(u_boot_config, fs_ubtype) |
| |
| try: |
| # 128MiB volume |
| fs_img = mk_fs(u_boot_config, fs_type, 0x8000000, '128MB') |
| except: |
| pytest.skip('Setup failed for filesystem: ' + fs_type) |
| return |
| else: |
| yield [fs_ubtype, fs_img] |
| call('rm -f %s' % fs_img, shell=True) |
| |
| # |
| # Fixture for unlink test |
| # |
| @pytest.fixture() |
| def fs_obj_unlink(request, u_boot_config): |
| """Set up a file system to be used in unlink test. |
| |
| Args: |
| request: Pytest request object. |
| u_boot_config: U-boot configuration. |
| |
| Return: |
| A fixture for unlink test, i.e. a duplet of file system type and |
| volume file name. |
| """ |
| fs_type = request.param |
| fs_img = '' |
| |
| fs_ubtype = fstype_to_ubname(fs_type) |
| check_ubconfig(u_boot_config, fs_ubtype) |
| |
| mount_dir = u_boot_config.persistent_data_dir + '/mnt' |
| |
| try: |
| |
| # 128MiB volume |
| fs_img = mk_fs(u_boot_config, fs_type, 0x8000000, '128MB') |
| except CalledProcessError as err: |
| pytest.skip('Creating failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| return |
| |
| try: |
| check_call('mkdir -p %s' % mount_dir, shell=True) |
| except CalledProcessError as err: |
| pytest.skip('Preparing mount folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Mount the image so we can populate it. |
| mount_fs(fs_type, fs_img, mount_dir) |
| except CalledProcessError as err: |
| pytest.skip('Mounting to folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Test Case 1 & 3 |
| check_call('mkdir %s/dir1' % mount_dir, shell=True) |
| check_call('dd if=/dev/urandom of=%s/dir1/file1 bs=1K count=1' |
| % mount_dir, shell=True) |
| check_call('dd if=/dev/urandom of=%s/dir1/file2 bs=1K count=1' |
| % mount_dir, shell=True) |
| |
| # Test Case 2 |
| check_call('mkdir %s/dir2' % mount_dir, shell=True) |
| for i in range(0, 20): |
| check_call('mkdir %s/dir2/0123456789abcdef%02x' |
| % (mount_dir, i), shell=True) |
| |
| # Test Case 4 |
| check_call('mkdir %s/dir4' % mount_dir, shell=True) |
| |
| # Test Case 5, 6 & 7 |
| check_call('mkdir %s/dir5' % mount_dir, shell=True) |
| check_call('dd if=/dev/urandom of=%s/dir5/file1 bs=1K count=1' |
| % mount_dir, shell=True) |
| |
| except CalledProcessError: |
| pytest.skip('Setup failed for filesystem: ' + fs_type) |
| umount_fs(mount_dir) |
| return |
| else: |
| umount_fs(mount_dir) |
| yield [fs_ubtype, fs_img] |
| finally: |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| |
| # |
| # Fixture for symlink fs test |
| # |
| @pytest.fixture() |
| def fs_obj_symlink(request, u_boot_config): |
| """Set up a file system to be used in symlink fs test. |
| |
| Args: |
| request: Pytest request object. |
| u_boot_config: U-boot configuration. |
| |
| Return: |
| A fixture for basic fs test, i.e. a triplet of file system type, |
| volume file name and a list of MD5 hashes. |
| """ |
| fs_type = request.param |
| fs_img = '' |
| |
| fs_ubtype = fstype_to_ubname(fs_type) |
| check_ubconfig(u_boot_config, fs_ubtype) |
| |
| mount_dir = u_boot_config.persistent_data_dir + '/mnt' |
| |
| small_file = mount_dir + '/' + SMALL_FILE |
| medium_file = mount_dir + '/' + MEDIUM_FILE |
| |
| try: |
| |
| # 1GiB volume |
| fs_img = mk_fs(u_boot_config, fs_type, 0x40000000, '1GB') |
| except CalledProcessError as err: |
| pytest.skip('Creating failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| return |
| |
| try: |
| check_call('mkdir -p %s' % mount_dir, shell=True) |
| except CalledProcessError as err: |
| pytest.skip('Preparing mount folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Mount the image so we can populate it. |
| mount_fs(fs_type, fs_img, mount_dir) |
| except CalledProcessError as err: |
| pytest.skip('Mounting to folder failed for filesystem: ' + fs_type + '. {}'.format(err)) |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |
| return |
| |
| try: |
| # Create a subdirectory. |
| check_call('mkdir %s/SUBDIR' % mount_dir, shell=True) |
| |
| # Create a small file in this image. |
| check_call('dd if=/dev/urandom of=%s bs=1M count=1' |
| % small_file, shell=True) |
| |
| # Create a medium file in this image. |
| check_call('dd if=/dev/urandom of=%s bs=10M count=1' |
| % medium_file, shell=True) |
| |
| # Generate the md5sums of reads that we will test against small file |
| out = check_output( |
| 'dd if=%s bs=1M skip=0 count=1 2> /dev/null | md5sum' |
| % small_file, shell=True).decode() |
| md5val = [out.split()[0]] |
| out = check_output( |
| 'dd if=%s bs=10M skip=0 count=1 2> /dev/null | md5sum' |
| % medium_file, shell=True).decode() |
| md5val.extend([out.split()[0]]) |
| |
| except CalledProcessError: |
| pytest.skip('Setup failed for filesystem: ' + fs_type) |
| umount_fs(mount_dir) |
| return |
| else: |
| umount_fs(mount_dir) |
| yield [fs_ubtype, fs_img, md5val] |
| finally: |
| call('rmdir %s' % mount_dir, shell=True) |
| call('rm -f %s' % fs_img, shell=True) |