diff --git a/components/fatfs/fatfs_utils/boot_sector.py b/components/fatfs/fatfs_utils/boot_sector.py index e8f25bf398..615dd06511 100644 --- a/components/fatfs/fatfs_utils/boot_sector.py +++ b/components/fatfs/fatfs_utils/boot_sector.py @@ -3,7 +3,7 @@ from inspect import getmembers, isroutine from typing import Optional -from construct import Const, Int8ul, Int16ul, Int32ul, PaddedString, Struct +from construct import Const, Int8ul, Int16ul, Int32ul, PaddedString, Struct, core from .exceptions import InconsistentFATAttributes, NotInitialized from .fatfs_state import BootSectorState @@ -56,7 +56,7 @@ class BootSector: assert BOOT_SECTOR_HEADER.sizeof() == BOOT_HEADER_SIZE def __init__(self, boot_sector_state: Optional[BootSectorState] = None) -> None: - self._parsed_header = None + self._parsed_header: dict = {} self.boot_sector_state: BootSectorState = boot_sector_state def generate_boot_sector(self) -> None: @@ -97,8 +97,12 @@ class BootSector: ) def parse_boot_sector(self, binary_data: bytes) -> None: - self._parsed_header = BootSector.BOOT_SECTOR_HEADER.parse(binary_data) - if self._parsed_header is None: + """ + Checks the validity of the boot sector and derives the metadata from boot sector to the structured shape. + """ + try: + self._parsed_header = BootSector.BOOT_SECTOR_HEADER.parse(binary_data) + except core.StreamError: raise NotInitialized('The boot sector header is not parsed successfully!') if self._parsed_header['BPB_TotSec16'] != 0x00: @@ -141,9 +145,14 @@ class BootSector: assert self.boot_sector_state.file_sys_type in (f'FAT{self.boot_sector_state.fatfs_type} ', 'FAT ') def __str__(self) -> str: - if self._parsed_header is None: + """ + FATFS properties parser (internal helper tool for fatfsgen.py/fatfsparse.py) + Provides all the properties of given FATFS instance by parsing its boot sector (returns formatted string) + """ + + if self._parsed_header == {}: return 'Boot sector is not initialized!' - res: str = 'Properties of the FATFS:\n' + res: str = 'FATFS properties:\n' for member in getmembers(self.boot_sector_state, lambda a: not (isroutine(a))): prop_ = getattr(self.boot_sector_state, member[0]) if isinstance(prop_, int) or isinstance(prop_, str) and not member[0].startswith('_'): @@ -152,7 +161,8 @@ class BootSector: @property def binary_image(self) -> bytes: - if len(self.boot_sector_state.binary_image) == 0: - raise NotInitialized('Boot sector is not generated nor initialized!') + # when BootSector is not instantiated, self.boot_sector_state might be None + if self.boot_sector_state is None or len(self.boot_sector_state.binary_image) == 0: + raise NotInitialized('Boot sector is not initialized!') bin_image_: bytes = self.boot_sector_state.binary_image return bin_image_ diff --git a/components/fatfs/fatfs_utils/cluster.py b/components/fatfs/fatfs_utils/cluster.py index 4ea6c315e6..ced9b1f5c3 100644 --- a/components/fatfs/fatfs_utils/cluster.py +++ b/components/fatfs/fatfs_utils/cluster.py @@ -30,6 +30,14 @@ class Cluster: cluster_id: int, boot_sector_state: BootSectorState, init_: bool) -> None: + """ + Initially, if init_ is False, the cluster is virtual and is not allocated (doesn't do changes in the FAT). + :param cluster_id: the cluster ID - a key value linking the file's cluster, + the corresponding physical cluster (data region) and the FAT table cluster. + :param boot_sector_state: auxiliary structure holding the file-system's metadata + :param init_: True for allocation the cluster on instantiation, otherwise False. + :returns: None + """ self.id: int = cluster_id self.boot_sector_state: BootSectorState = boot_sector_state @@ -50,8 +58,19 @@ class Cluster: def next_cluster(self, value): # type: (Optional[Cluster]) -> None self._next_cluster = value - def _cluster_id_to_logical_position_in_bits(self, _id: int) -> int: - # computes address of the cluster in fat table + def _cluster_id_to_fat_position_in_bits(self, _id: int) -> int: + """ + This private method calculates the position of the memory block (cluster) in the FAT table. + + :param _id: the cluster ID - a key value linking the file's cluster, + the corresponding physical cluster (data region) and the FAT table cluster. + :returns: bit offset of the cluster in FAT + e.g.: + 00003000: 42 65 00 2E 00 74 00 78 00 74 00 0F 00 43 FF FF + + For FAT12 the third cluster has value = 0x02E and ID = 2. + Its bit-address is 24 (24 bits preceding, 0-indexed), because 0x2E starts at the bit-offset 24. + """ logical_position_: int = self.boot_sector_state.fatfs_type * _id return logical_position_ @@ -73,18 +92,10 @@ class Cluster: def _compute_cluster_data_address(self) -> int: return self.compute_cluster_data_address(self.boot_sector_state, self.id) - def _set_left_half_byte(self, address: int, value: int) -> None: - self.boot_sector_state.binary_image[address] &= 0x0f - self.boot_sector_state.binary_image[address] |= value << 4 - - def _set_right_half_byte(self, address: int, value: int) -> None: - self.boot_sector_state.binary_image[address] &= 0xf0 - self.boot_sector_state.binary_image[address] |= value - @property def fat_cluster_address(self) -> int: """Determines how many bits precede the first bit of the cluster in FAT""" - return self._cluster_id_to_logical_position_in_bits(self.id) + return self._cluster_id_to_fat_position_in_bits(self.id) @property def real_cluster_address(self) -> int: @@ -141,6 +152,27 @@ class Cluster: 2. if the cluster index is odd, we set the first half of the computed byte and the full consequent byte. Order of half bytes is 1, 3, 2. """ + + def _set_msb_half_byte(address: int, value_: int) -> None: + """ + Sets 4 most significant bits (msb half-byte) of 'boot_sector_state.binary_image' at given + 'address' to 'value_' (size of variable 'value_' is half byte) + + If a byte contents is 0b11110000, the msb half-byte would be 0b1111 + """ + self.boot_sector_state.binary_image[address] &= 0x0f + self.boot_sector_state.binary_image[address] |= value_ << 4 + + def _set_lsb_half_byte(address: int, value_: int) -> None: + """ + Sets 4 least significant bits (lsb half-byte) of 'boot_sector_state.binary_image' at given + 'address' to 'value_' (size of variable 'value_' is half byte) + + If a byte contents is 0b11110000, the lsb half-byte would be 0b0000 + """ + self.boot_sector_state.binary_image[address] &= 0xf0 + self.boot_sector_state.binary_image[address] |= value_ + # value must fit into number of bits of the fat (12, 16 or 32) assert value <= (1 << self.boot_sector_state.fatfs_type) - 1 half_bytes = split_by_half_byte_12_bit_little_endian(value) @@ -151,10 +183,10 @@ class Cluster: if self.fat_cluster_address % 8 == 0: # even block bin_img_[self.real_cluster_address] = build_byte(half_bytes[1], half_bytes[0]) - self._set_right_half_byte(self.real_cluster_address + 1, half_bytes[2]) + _set_lsb_half_byte(self.real_cluster_address + 1, half_bytes[2]) elif self.fat_cluster_address % 8 != 0: # odd block - self._set_left_half_byte(self.real_cluster_address, half_bytes[0]) + _set_msb_half_byte(self.real_cluster_address, half_bytes[0]) bin_img_[self.real_cluster_address + 1] = build_byte(half_bytes[2], half_bytes[1]) elif self.boot_sector_state.fatfs_type == FAT16: bin_img_[self.real_cluster_address:self.real_cluster_address + 2] = Int16ul.build(value) @@ -162,6 +194,11 @@ class Cluster: @property def is_root(self) -> bool: + """ + The FAT12/FAT16 contains only one root directory, + the root directory allocates the first cluster with the ID `ROOT_BLOCK_ID`. + The method checks if the cluster belongs to the root directory. + """ return self.id == Cluster.ROOT_BLOCK_ID def allocate_cluster(self) -> None: diff --git a/components/fatfs/fatfs_utils/exceptions.py b/components/fatfs/fatfs_utils/exceptions.py index b27c760770..a3a27df5d6 100644 --- a/components/fatfs/fatfs_utils/exceptions.py +++ b/components/fatfs/fatfs_utils/exceptions.py @@ -48,4 +48,7 @@ class FatalError(Exception): class InconsistentFATAttributes(Exception): + """ + Caused by e.g. wrong number of clusters for given FAT type + """ pass diff --git a/components/fatfs/fatfs_utils/fat.py b/components/fatfs/fatfs_utils/fat.py index 80a5fb4842..396075dd08 100644 --- a/components/fatfs/fatfs_utils/fat.py +++ b/components/fatfs/fatfs_utils/fat.py @@ -22,7 +22,7 @@ class FAT: self.clusters[Cluster.ROOT_BLOCK_ID].allocate_cluster() def __init__(self, boot_sector_state: BootSectorState, init_: bool) -> None: - self._first_free_cluster_id = 0 + self._first_free_cluster_id = 1 self.boot_sector_state = boot_sector_state self.clusters: List[Cluster] = [Cluster(cluster_id=i, boot_sector_state=self.boot_sector_state, @@ -31,10 +31,22 @@ class FAT: self.allocate_root_dir() def get_cluster_value(self, cluster_id_: int) -> int: + """ + The method retrieves the values of the FAT memory block. + E.g. in case of FAT12: + 00000000: F8 FF FF 55 05 00 00 00 00 00 00 00 00 00 00 00 + + The reserved value is 0xFF8, the value of first cluster if 0xFFF, thus is last in chain, + and the value of the second cluster is 0x555, so refers to the cluster number 0x555. + """ fat_cluster_value_: int = self.clusters[cluster_id_].get_from_fat() return fat_cluster_value_ def is_cluster_last(self, cluster_id_: int) -> bool: + """ + Checks if the cluster is last in its cluster chain. If the value of the cluster is + 0xFFF for FAT12, 0xFFFF for FAT16 or 0xFFFFFFFF for FAT32, the cluster is the last. + """ value_ = self.get_cluster_value(cluster_id_) is_cluster_last_: bool = value_ == (1 << self.boot_sector_state.fatfs_type) - 1 return is_cluster_last_ @@ -60,16 +72,26 @@ class FAT: return content_[:size] def find_free_cluster(self) -> Cluster: - # finds first empty cluster and allocates it - for cluster_id, cluster in enumerate(self.clusters[self._first_free_cluster_id:], - start=self._first_free_cluster_id): - if cluster.is_empty: - cluster.allocate_cluster() - self._first_free_cluster_id = cluster_id - return cluster - raise NoFreeClusterException('No free cluster available!') + """ + Returns the first free cluster and increments value of `self._first_free_cluster_id`. + The method works only in context of creating a partition from scratch. + In situations where the clusters are allocated and freed during the run of the program, + might the method cause `Out of space` error despite there would be free clusters. + """ + + if self._first_free_cluster_id + 1 >= len(self.clusters): + raise NoFreeClusterException('No free cluster available!') + cluster = self.clusters[self._first_free_cluster_id + 1] + if not cluster.is_empty: + raise NoFreeClusterException('No free cluster available!') + cluster.allocate_cluster() + self._first_free_cluster_id += 1 + return cluster def allocate_chain(self, first_cluster: Cluster, size: int) -> None: + """ + Allocates the linked list of clusters needed for the given file or directory. + """ current = first_cluster for _ in range(size - 1): free_cluster = self.find_free_cluster() diff --git a/components/fatfs/fatfsgen.py b/components/fatfs/fatfsgen.py index d70fb7ec60..11f8dd4545 100755 --- a/components/fatfs/fatfsgen.py +++ b/components/fatfs/fatfsgen.py @@ -79,7 +79,17 @@ class FATFS: extension: str = '', path_from_root: Optional[List[str]] = None, object_timestamp_: datetime = FATFS_INCEPTION) -> None: - # when path_from_root is None the dir is root + """ + Root directory recursively finds the parent directory of the new file, allocates cluster, + entry and appends a new file into the parent directory. + + When path_from_root is None the dir is root. + + :param name: The name of the file. + :param extension: The extension of the file. + :param path_from_root: List of strings containing names of the ancestor directories in the given order. + :param object_timestamp_: is not None, this will be propagated to the file's entry + """ self.root_directory.new_file(name=name, extension=extension, path_from_root=path_from_root, @@ -88,7 +98,18 @@ class FATFS: def create_directory(self, name: str, path_from_root: Optional[List[str]] = None, object_timestamp_: datetime = FATFS_INCEPTION) -> None: - # when path_from_root is None the dir is root + """ + Initially recursively finds a parent of the new directory + and then create a new directory inside the parent. + + When path_from_root is None the parent dir is root. + + :param name: The full name of the directory (excluding its path) + :param path_from_root: List of strings containing names of the ancestor directories in the given order. + :param object_timestamp_: in case the user preserves the timestamps, this will be propagated to the + metadata of the directory (to the corresponding entry) + :returns: None + """ parent_dir = self.root_directory if path_from_root: parent_dir = self.root_directory.recursive_search(path_from_root, self.root_directory) diff --git a/components/fatfs/test_fatfsgen/test_fatfsgen.py b/components/fatfs/test_fatfsgen/test_fatfsgen.py index 7d1f702390..5816991c0e 100755 --- a/components/fatfs/test_fatfsgen/test_fatfsgen.py +++ b/components/fatfs/test_fatfsgen/test_fatfsgen.py @@ -12,10 +12,15 @@ from test_utils import CFG, fill_sector, generate_test_dir_1, generate_test_dir_ sys.path.append(os.path.join(os.path.dirname(__file__), '..')) import fatfsgen # noqa E402 # pylint: disable=C0413 +from fatfs_utils.boot_sector import BootSector # noqa E402 # pylint: disable=C0413 +from fatfs_utils.cluster import Cluster # noqa E402 # pylint: disable=C0413 +from fatfs_utils.entry import Entry # noqa E402 # pylint: disable=C0413 from fatfs_utils.exceptions import InconsistentFATAttributes # noqa E402 # pylint: disable=C0413 +from fatfs_utils.exceptions import NotInitialized # noqa E402 # pylint: disable=C0413 from fatfs_utils.exceptions import TooLongNameException # noqa E402 # pylint: disable=C0413 from fatfs_utils.exceptions import WriteDirectoryException # noqa E402 # pylint: disable=C0413 from fatfs_utils.exceptions import LowerCaseException, NoFreeClusterException # noqa E402 # pylint: disable=C0413 +from fatfs_utils.utils import right_strip_string # noqa E402 # pylint: disable=C0413 from fatfs_utils.utils import FAT12, read_filesystem # noqa E402 # pylint: disable=C0413 @@ -473,6 +478,58 @@ class FatFSGen(unittest.TestCase): self.assertEqual(file_system[0x60d0: 0x60e0], b'e\x00l\x00l\x00o\x00h\x00\x00\x00e\x00l\x00') self.assertEqual(file_system[0x60e0: 0x60f0], b'HELLOH~\x02TXT \x00\x00\x00\x00') + def test_bs_not_initialized(self) -> None: + self.assertEqual(str(BootSector()), 'Boot sector is not initialized!') + self.assertRaises(NotInitialized, BootSector().generate_boot_sector) + self.assertRaises(NotInitialized, lambda: BootSector().binary_image) # encapsulate property to callable + + def test_bs_str(self) -> None: + fatfs = fatfsgen.FATFS() + bs = BootSector(fatfs.state.boot_sector_state) + bs.generate_boot_sector() + bs.parse_boot_sector(bs.binary_image) + x = 'FATFS properties:,clusters: 252,data_region_start: 24576,data_sectors: ' \ + '250,entries_root_count: 512,fat_table_start_address: 4096,fat_tables_cnt: 1,' \ + 'fatfs_type: 12,file_sys_type: FAT ,hidden_sectors: 0,media_type: 248,' \ + 'non_data_sectors: 6,num_heads: 255,oem_name: MSDOS5.0,reserved_sectors_cnt: 1,' \ + 'root_dir_sectors_cnt: 4,root_directory_start: 8192,sec_per_track: 63,sector_size: 4096,' \ + 'sectors_count: 256,sectors_per_cluster: 1,sectors_per_fat_cnt: 1,size: 1048576,' \ + 'volume_label: Espressif ,volume_uuid: 1144419653,' + self.assertEqual(x.split(',')[:-2], str(bs).split('\n')[:-2]) # except for volume id + + def test_parsing_error(self) -> None: + self.assertRaises(NotInitialized, BootSector().parse_boot_sector, b'') + + def test_not_implemented_fat32(self) -> None: + self.assertEqual( + Entry.get_cluster_id( + Entry.ENTRY_FORMAT_SHORT_NAME.parse( + bytearray(b'AHOJ \x18\x00\xb0[&U&U\x00\x00\xb0[&U\x02\x00\x08\x00\x00\x00'))), + 2) + + def test_get_cluster_value_from_fat(self) -> None: + fatfs = fatfsgen.FATFS() + self.assertEqual(fatfs.fat.get_cluster_value(1), 0xFFF) + + def test_is_cluster_last(self) -> None: + fatfs = fatfsgen.FATFS() + self.assertEqual(fatfs.fat.is_cluster_last(2), False) + + def test_chain_in_fat(self) -> None: + fatfs = fatfsgen.FATFS() + self.assertEqual(fatfs.fat.get_chained_content(1), b'\x00' * 0x1000) + + def test_retrieve_file_chaining(self) -> None: + fatfs = fatfsgen.FATFS() + fatfs.create_file('WRITEF', extension='TXT') + fatfs.write_content(path_from_root=['WRITEF.TXT'], content=CFG['sector_size'] * b'a' + b'a') + fatfs.write_filesystem(CFG['output_file']) + self.assertEqual(fatfs.fat.get_chained_content(1)[:15], b'WRITEF TXT \x00\x00\x00') + self.assertEqual(fatfs.fat.get_chained_content(2)[:15], b'aaaaaaaaaaaaaaa') + + def test_lstrip(self) -> None: + self.assertEqual(right_strip_string('\x20\x20\x20thisistest\x20\x20\x20'), ' thisistest') + if __name__ == '__main__': unittest.main() diff --git a/components/fatfs/test_fatfsgen/test_fatfsparse.py b/components/fatfs/test_fatfsgen/test_fatfsparse.py index aa5f28dfb7..15a7f3d866 100755 --- a/components/fatfs/test_fatfsgen/test_fatfsparse.py +++ b/components/fatfs/test_fatfsgen/test_fatfsparse.py @@ -12,6 +12,7 @@ from test_utils import compare_folders, fill_sector, generate_local_folder_struc sys.path.append(os.path.join(os.path.dirname(__file__), '..')) import fatfsgen # noqa E402 # pylint: disable=C0413 +from fatfs_utils.entry import Entry # noqa E402 # pylint: disable=C0413 class FatFSGen(unittest.TestCase): @@ -323,6 +324,31 @@ class FatFSGen(unittest.TestCase): run(['python', '../fatfsparse.py', 'fatfs_image.img'], stderr=STDOUT) assert compare_folders('testf', 'Espressif') + def test_parse_long_name(self) -> None: + self.assertEqual( + Entry.parse_entry_long( + b'\x01t\x00h\x00i\x00s\x00_\x00\x0f\x00\xfbi\x00s\x00_\x00l\x00o\x00n\x00\x00\x00g\x00_\x00', 251), + { + 'order': 1, + 'name1': b't\x00h\x00i\x00s\x00_\x00', + 'name2': b'i\x00s\x00_\x00l\x00o\x00n\x00', + 'name3': b'g\x00_\x00', + 'is_last': False + } + ) + self.assertEqual( + Entry.parse_entry_long( + b'\x01t\x00h\x00i\x00s\x00_\x00\x0f\x00\xfbi\x00s\x00_\x00l\x00o\x00n\x00\x00\x00g\x00_\x00', 252 + ), + {} + ) + self.assertEqual( + Entry.parse_entry_long( + b'\x01t\x00h\x00i\x00s\x00_\x00\x0f\x01\xfbi\x00s\x00_\x00l\x00o\x00n\x00\x00\x00g\x00_\x00', 251 + ), + {} + ) + if __name__ == '__main__': unittest.main() diff --git a/components/fatfs/test_fatfsgen/test_wl_fatfsgen.py b/components/fatfs/test_fatfsgen/test_wl_fatfsgen.py index e39d7cd8c4..2b3e28c7eb 100755 --- a/components/fatfs/test_fatfsgen/test_wl_fatfsgen.py +++ b/components/fatfs/test_fatfsgen/test_wl_fatfsgen.py @@ -25,7 +25,7 @@ class WLFatFSGen(unittest.TestCase): def test_empty_file_sn_fat12(self) -> None: fatfs = wl_fatfsgen.WLFATFS() - fatfs.wl_create_file('TESTFILE') + fatfs.plain_fatfs.create_file('TESTFILE') fatfs.init_wl() fatfs.wl_write_filesystem(CFG['output_file']) with open(CFG['output_file'], 'rb') as fs_file: @@ -36,7 +36,7 @@ class WLFatFSGen(unittest.TestCase): def test_directory_sn_fat12(self) -> None: fatfs = wl_fatfsgen.WLFATFS(device_id=3750448905) - fatfs.wl_create_directory('TESTFOLD') + fatfs.plain_fatfs.create_directory('TESTFOLD') fatfs.init_wl() fatfs.wl_write_filesystem(CFG['output_file']) @@ -73,7 +73,7 @@ class WLFatFSGen(unittest.TestCase): def test_directory_sn_fat122mb(self) -> None: fatfs = wl_fatfsgen.WLFATFS(device_id=3750448905, size=2 * 1024 * 1024) - fatfs.wl_create_directory('TESTFOLD') + fatfs.plain_fatfs.create_directory('TESTFOLD') fatfs.init_wl() fatfs.wl_write_filesystem(CFG['output_file']) @@ -102,12 +102,12 @@ class WLFatFSGen(unittest.TestCase): def test_write_not_initialized_wlfatfs(self) -> None: fatfs = wl_fatfsgen.WLFATFS() - fatfs.wl_create_directory('TESTFOLD') + fatfs.plain_fatfs.create_directory('TESTFOLD') self.assertRaises(WLNotInitialized, fatfs.wl_write_filesystem, CFG['output_file']) def test_e2e_deep_folder_into_image_ext(self) -> None: fatfs = wl_fatfsgen.WLFATFS() - fatfs.wl_generate(CFG['test_dir2']) + fatfs.plain_fatfs.generate(CFG['test_dir2']) fatfs.init_wl() fatfs.wl_write_filesystem(CFG['output_file']) with open(CFG['output_file'], 'rb') as fs_file: @@ -124,7 +124,7 @@ class WLFatFSGen(unittest.TestCase): def test_e2e_deep_folder_into_image(self) -> None: fatfs = wl_fatfsgen.WLFATFS() - fatfs.wl_generate(CFG['test_dir']) + fatfs.plain_fatfs.generate(CFG['test_dir']) fatfs.init_wl() fatfs.wl_write_filesystem(CFG['output_file']) with open(CFG['output_file'], 'rb') as fs_file: diff --git a/components/fatfs/wl_fatfsgen.py b/components/fatfs/wl_fatfsgen.py index b6c5dd4fe7..a70d00d6a7 100755 --- a/components/fatfs/wl_fatfsgen.py +++ b/components/fatfs/wl_fatfsgen.py @@ -2,8 +2,6 @@ # SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD # SPDX-License-Identifier: Apache-2.0 -from typing import List, Optional - from construct import Const, Int32ul, Struct from fatfs_utils.exceptions import WLNotInitialized from fatfs_utils.utils import (FULL_BYTE, UINT32_MAX, FATDefaults, crc32, generate_4bytes_random, @@ -200,21 +198,6 @@ class WLFATFS: with open(output_path, 'wb') as output: output.write(bytearray(self.fatfs_binary_image)) - def wl_generate(self, input_directory: str) -> None: - """ - Normalize path to folder and recursively encode folder to binary image - """ - self.plain_fatfs.generate(input_directory=input_directory) - - def wl_create_file(self, name: str, extension: str = '', path_from_root: Optional[List[str]] = None) -> None: - self.plain_fatfs.create_file(name, extension, path_from_root) - - def wl_create_directory(self, name: str, path_from_root: Optional[List[str]] = None) -> None: - self.plain_fatfs.create_directory(name, path_from_root) - - def wl_write_content(self, path_from_root: List[str], content: bytes) -> None: - self.plain_fatfs.write_content(path_from_root, content) - if __name__ == '__main__': desc = 'Create a FAT filesystem with support for wear levelling and populate it with directory content' @@ -228,6 +211,6 @@ if __name__ == '__main__': long_names_enabled=args.long_name_support, use_default_datetime=args.use_default_datetime) - wl_fatfs.wl_generate(args.input_directory) + wl_fatfs.plain_fatfs.generate(args.input_directory) wl_fatfs.init_wl() wl_fatfs.wl_write_filesystem(args.output_file)