mirror of
https://github.com/littlefs-project/littlefs.git
synced 2026-04-01 20:24:46 +00:00
This started with adding -d/--diff support to dbgflags.py, which is very
useful for comparing flags during test failure.
The flag asserts in our tests generally look like this:
tests/test_mount.toml:171:assert: assert failed with 33570064,
expected eq 33570576
assert(fsinfo.flags == (
Which can now be quickly compared with dbgflags.py:
$ ./scripts/dbgflags.py +i 33570064 -d 33570576
LFS3_I_GBMAP 0x02000000 Global on-disk block-map in use
LFS3_I_REVPERTURB 0x00000010 Mounted with LFS3_M_REVPERTURB
LFS3_I_MKCONSISTENT 0x00000100 Filesystem needs mkconsistent to write
-LFS3_I_LOOKAHEAD 0x00000200 Lookahead buffer is not full
LFS3_I_PREERASE 0x00000400 Blocks can be pre-erased
LFS3_I_COMPACT 0x00000800 Filesystem may have uncompacted metadata
LFS3_I_CKMETA 0x00001000 Metadata checksums not checked recently
LFS3_I_CKDATA 0x00002000 Data checksums not checked recently
The assert print is a bit more annoying than it needs to be, as it only
prints in decimal. But, since our prettyasserts.py only works at the
syntax layer, it's not possible to make it any smarter.
---
To make this diffing work required a couple more features in our
self-parsing Flag class:
- Keep track of lineno, mainly for ordering things
- Moved find logic into a staticmethod on all classes
- Added _sentinel based defaults to find functions
- Allowed self to be non-class in line functions to deduplicate "Unknown
flag" messages
I went ahead and extended these to the other self-parsing classes (Err
and Tag) in case they're useful in the future.
171 lines
4.7 KiB
Python
Executable File
171 lines
4.7 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
# prevent local imports
|
|
if __name__ == "__main__":
|
|
__import__('sys').path.pop(0)
|
|
|
|
import functools as ft
|
|
|
|
|
|
# Error codes
|
|
ERR_OK = 0 # No error
|
|
ERR_UNKNOWN = -1 # Unknown error
|
|
ERR_INVAL = -22 # Invalid parameter
|
|
ERR_NOTSUP = -95 # Operation not supported
|
|
ERR_BUSY = -16 # Device or resource busy
|
|
ERR_IO = -5 # Error during device operation
|
|
ERR_CORRUPT = -84 # Corrupted
|
|
ERR_NOENT = -2 # No directory entry
|
|
ERR_EXIST = -17 # Entry already exists
|
|
ERR_NOTDIR = -20 # Entry is not a dir
|
|
ERR_ISDIR = -21 # Entry is a dir
|
|
ERR_NOTEMPTY = -39 # Dir is not empty
|
|
ERR_FBIG = -27 # File too large
|
|
ERR_NOSPC = -28 # No space left on device
|
|
ERR_NOMEM = -12 # No more memory available
|
|
ERR_NOATTR = -61 # No data/attr available
|
|
ERR_NAMETOOLONG = -36 # File name too long
|
|
ERR_RANGE = -34 # Result out of range
|
|
|
|
|
|
# self-parsing error codes
|
|
class Err:
|
|
def __init__(self, name, code, help, *,
|
|
lineno=0):
|
|
self.name = name
|
|
self.code = code
|
|
self.help = help
|
|
self.lineno = lineno
|
|
|
|
def __repr__(self):
|
|
return 'Err(%r, %r, %r)' % (
|
|
self.name,
|
|
self.code,
|
|
self.help)
|
|
|
|
def __eq__(self, other):
|
|
return self.name == other.name
|
|
|
|
def __ne__(self, other):
|
|
return self.name != other.name
|
|
|
|
def __hash__(self):
|
|
return hash(self.name)
|
|
|
|
def line(self):
|
|
if isinstance(self, Err):
|
|
return ('LFS3_%s' % self.name, '%d' % self.code, self.help)
|
|
else:
|
|
return ('?', str(self), 'Unknown err code')
|
|
|
|
@staticmethod
|
|
@ft.cache
|
|
def errs():
|
|
# parse our script's source to figure out errs
|
|
import inspect
|
|
import re
|
|
errs = []
|
|
err_pattern = re.compile(
|
|
'^(?P<name>ERR_[^ ]*) *= *(?P<code>[^#]*?) *'
|
|
'#+ *(?P<help>.*)$')
|
|
for i, line in enumerate(
|
|
inspect.getsource(inspect.getmodule(inspect.currentframe()))
|
|
.replace('\\\n', '')
|
|
.splitlines()):
|
|
m = err_pattern.match(line)
|
|
if m:
|
|
errs.append(Err(
|
|
m.group('name'),
|
|
globals()[m.group('name')],
|
|
m.group('help'),
|
|
lineno=1+i))
|
|
return errs
|
|
|
|
_sentinel = object()
|
|
@staticmethod
|
|
def find(e_, *, default=_sentinel):
|
|
# find errs, note this is cached
|
|
errs__ = Err.errs()
|
|
|
|
# find by LFS3_ERR_+name
|
|
for e in errs__:
|
|
if 'LFS3_%s' % e.name.upper() == e_.upper():
|
|
return e
|
|
# find by ERR_+name
|
|
for e in errs__:
|
|
if e.name.upper() == e_.upper():
|
|
return e
|
|
# find by name
|
|
for e in errs__:
|
|
if e.name.split('_', 1)[1] == e_.upper():
|
|
return e
|
|
# find by E+name
|
|
for e in errs__:
|
|
if 'E%s' % e.name.split('_', 1)[1].upper() == e_.upper():
|
|
return e
|
|
try:
|
|
# find by err code
|
|
for e in errs__:
|
|
if e.code == int(e_, 0):
|
|
return e
|
|
# find by negated err code
|
|
for e in errs__:
|
|
if e.code == -int(e_, 0):
|
|
return e
|
|
except ValueError:
|
|
pass
|
|
# not found
|
|
if default is Err._sentinel:
|
|
raise KeyError(e_)
|
|
else:
|
|
return default
|
|
|
|
|
|
def main(errs, *,
|
|
list=False):
|
|
import builtins
|
|
list_, list = list, builtins.list
|
|
|
|
lines = []
|
|
# list all known error codes
|
|
if list_:
|
|
for e in Err.errs():
|
|
lines.append(e.line())
|
|
|
|
# find errs by name or value
|
|
else:
|
|
for e_ in errs:
|
|
lines.append(Err.line(Err.find(e_, default=e_)))
|
|
|
|
# first find widths
|
|
w = [0, 0]
|
|
for l in lines:
|
|
w[0] = max(w[0], len(l[0]))
|
|
w[1] = max(w[1], len(l[1]))
|
|
|
|
# then print results
|
|
for l in lines:
|
|
print('%-*s %-*s %s' % (
|
|
w[0], l[0],
|
|
w[1], l[1],
|
|
l[2]))
|
|
|
|
|
|
if __name__ == "__main__":
|
|
import argparse
|
|
import sys
|
|
parser = argparse.ArgumentParser(
|
|
description="Decode littlefs error codes.",
|
|
allow_abbrev=False)
|
|
parser.add_argument(
|
|
'errs',
|
|
nargs='*',
|
|
help="Error codes or error names to decode.")
|
|
parser.add_argument(
|
|
'-l', '--list',
|
|
action='store_true',
|
|
help="List all known error codes.")
|
|
sys.exit(main(**{k: v
|
|
for k, v in vars(parser.parse_intermixed_args()).items()
|
|
if v is not None}))
|