1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-13 08:50:56 +00:00

core/sdcard: fix invalid state when filesystem mounting fails

This commit is contained in:
matejcik 2020-02-21 15:53:16 +01:00 committed by matejcik
parent f9097b16e6
commit e9c275c24f
2 changed files with 46 additions and 12 deletions

View File

@ -7,7 +7,7 @@ if False:
class FilesystemWrapper: class FilesystemWrapper:
_INSTANCE = None # type: Optional[FilesystemWrapper] _INSTANCE = None # type: Optional[FilesystemWrapper]
def __init__(self, mounted: bool = True) -> None: def __init__(self, mounted: bool) -> None:
self.fs = FatFS() self.fs = FatFS()
self.mounted = mounted self.mounted = mounted
self.counter = 0 self.counter = 0
@ -20,23 +20,30 @@ class FilesystemWrapper:
raise RuntimeError # cannot request mounted and non-mounted instance at the same time raise RuntimeError # cannot request mounted and non-mounted instance at the same time
return cls._INSTANCE return cls._INSTANCE
def _deinit_instance(self) -> None:
if self.mounted:
self.fs.unmount()
sdcard.power_off()
FilesystemWrapper._INSTANCE = None
def __enter__(self) -> "FatFS": def __enter__(self) -> "FatFS":
if self.counter <= 0: try:
self.counter = 0 if self.counter <= 0:
sdcard.power_on() self.counter = 0
if self.mounted: sdcard.power_on()
self.fs.mount() if self.mounted:
self.counter += 1 self.fs.mount()
return self.fs self.counter += 1
return self.fs
except Exception:
self._deinit_instance()
raise
def __exit__(self, exc_type: Any, exc_val: Any, tb: Any) -> None: def __exit__(self, exc_type: Any, exc_val: Any, tb: Any) -> None:
self.counter -= 1 self.counter -= 1
if self.counter <= 0: if self.counter <= 0:
self.counter = 0 self.counter = 0
if self.mounted: self._deinit_instance()
self.fs.unmount()
sdcard.power_off()
FilesystemWrapper._INSTANCE = None
def get_filesystem(mounted: bool = True) -> FilesystemWrapper: def get_filesystem(mounted: bool = True) -> FilesystemWrapper:

View File

@ -62,6 +62,33 @@ class TestTrezorSdcard(unittest.TestCase):
with sdcard.get_filesystem(mounted=True): with sdcard.get_filesystem(mounted=True):
pass pass
def test_failed_mount(self):
# set up a filesystem first
with sdcard.get_filesystem(mounted=False) as fs:
fs.mkfs()
with sdcard.get_filesystem() as fs:
# the following should succeed
fs.listdir("/")
# trash filesystem
io.sdcard.power_on()
io.sdcard.write(0, bytes([0xFF] * io.sdcard.BLOCK_SIZE))
io.sdcard.power_off()
# mounting should now fail
with self.assertRaises(OSError):
with sdcard.get_filesystem() as fs:
pass
# it should be possible to create an unmounted instance
with sdcard.get_filesystem(mounted=False) as fs:
fs.mkfs()
# mounting should now succeed
with sdcard.get_filesystem() as fs:
fs.listdir("/")
if __name__ == "__main__": if __name__ == "__main__":