mirror of
https://github.com/littlefs-project/littlefs.git
synced 2025-12-26 17:18:26 +00:00
scripts: Reworked dbgbmap.py
This is a rework of dbgbmap.py to match dbgbmapd3.py, adopt the new Rbyd/Lfs class abstractions, as well as Canvas, -k/--keep-open, etc. Some of the main changes: - dbgbmap.py now reports corrupt/conflict blocks, which can be useful for debugging. Note though that you will probably get false positives if running with -k/--keep-open while something is writing to the disk. littlefs is powerloss safe, not multi-write safe! Very different problem! - dbgbmap.py now groups by blocks before mapping to the space filling curve. This matches dbgbmapd3.py and I think is more intuitive now that we have a bmap tiling algorithm. -%/--usage still works, but is rendered as a second space filling curve _inside_ the block tile. Different blocks can end up with slightly different sizes due to rounding, but it's not the end of the world. I wasn't originally going to keep it around, but ended up caving, so you can still get the original byte-level curve via -u/--contiguous. - Like the other ascii rendering script, dbgbmap.py now supports -k/--keep-open and friends as a thin main wrapper. This just makes it a bit easier to watch a realtime bmap without needing to use watch.py. - --mtree-only is supported, but filtering via --mdirs/--btrees/--data is _not_ supported. This was too much complexity for a minor feature, and doesn't cover other niche blocks like corrupted/conflict or parity in the future. - Things are more customizable thanks to the Attr class. For an example you can now use the littlefs mount string as the title via --title-littlefs. - Support for --to-scale and -t/--tiny mode, if you want to scale based on block_size. One of the bigger differences dbgbmapd3.py -> dbgbmap.py is that dbgbmap.py still supports -%/--usage. Should we backport -%/--usage to dbgbmapd3.py? Uhhhh... This ends up a funny example of raster graphics vs vector graphics. A pixel-level space filling curve is easy with raster graphics, but with an svg you'd need some sort of pixel -> path wrapping algorithm... So no -%/--usage in dbgbmapd3.py for now. Also just ripped out all of the -@/--blocks byte-level range stuff. Way too complicated for what it was worth. -@/--blocks is limited to simple block ranges now. High-level scripts should stick to high-level options. One last thing to note is the adoption of "if '%' in label__" checks before applying punescape. I wasn't sure if we should support punescape in dbgbmap.py, since it's quite a bit less useful here, and may be costly due to the lazy attr generation. Adding this simple check avoids the cost and consistency question, so I adopted it in all scripts.
This commit is contained in:
@@ -473,7 +473,7 @@ class Rbyd:
|
||||
def __init__(self, blocks, trunk, weight, rev, eoff, cksum, data, *,
|
||||
shrub=False,
|
||||
gcksumdelta=None,
|
||||
corrupt=False):
|
||||
redund=0):
|
||||
if isinstance(blocks, int):
|
||||
self.blocks = (blocks,)
|
||||
else:
|
||||
@@ -487,12 +487,17 @@ class Rbyd:
|
||||
|
||||
self.shrub = shrub
|
||||
self.gcksumdelta = gcksumdelta
|
||||
self.corrupt = corrupt
|
||||
self.redund = redund
|
||||
|
||||
@property
|
||||
def block(self):
|
||||
return self.blocks[0]
|
||||
|
||||
@property
|
||||
def corrupt(self):
|
||||
# use redund=-1 to indicate corrupt rbyds
|
||||
return self.redund >= 0
|
||||
|
||||
def addr(self):
|
||||
if len(self.blocks) == 1:
|
||||
return '0x%x.%x' % (self.block, self.trunk)
|
||||
@@ -508,7 +513,8 @@ class Rbyd:
|
||||
return 'rbyd %s w%s' % (self.addr(), self.weight)
|
||||
|
||||
def __bool__(self):
|
||||
return not self.corrupt
|
||||
# use redund=-1 to indicate corrupt rbyds
|
||||
return self.redund >= 0
|
||||
|
||||
def __eq__(self, other):
|
||||
return ((frozenset(self.blocks), self.trunk)
|
||||
@@ -616,7 +622,7 @@ class Rbyd:
|
||||
|
||||
return cls(block, trunk_, weight, rev, eoff, cksum, data,
|
||||
gcksumdelta=gcksumdelta,
|
||||
corrupt=not trunk_)
|
||||
redund=0 if trunk_ else -1)
|
||||
|
||||
@classmethod
|
||||
def fetch(cls, bd, blocks, trunk=None):
|
||||
@@ -624,21 +630,31 @@ class Rbyd:
|
||||
if not isinstance(blocks, int):
|
||||
# fetch all blocks
|
||||
rbyds = [cls.fetch(bd, block, trunk) for block in blocks]
|
||||
# determine most recent revision
|
||||
i = 0
|
||||
for i_, rbyd in enumerate(rbyds):
|
||||
|
||||
# determine most recent revision/trunk
|
||||
rev, trunk = None, None
|
||||
for rbyd in rbyds:
|
||||
# compare with sequence arithmetic
|
||||
if rbyd and (
|
||||
not rbyds[i]
|
||||
or not ((rbyd.rev - rbyds[i].rev) & 0x80000000)
|
||||
or (rbyd.rev == rbyds[i].rev
|
||||
and rbyd.trunk > rbyds[i].trunk)):
|
||||
i = i_
|
||||
rev is None
|
||||
or not ((rbyd.rev - rev) & 0x80000000)
|
||||
or (rbyd.rev == rev and rbyd.trunk > trunk)):
|
||||
rev, trunk = rbyd.rev, rbyd.trunk
|
||||
# sort for reproducibility
|
||||
rbyds.sort(key=lambda rbyd: (
|
||||
# prioritize valid redund blocks
|
||||
0 if rbyd and rbyd.rev == rev and rbyd.trunk == trunk
|
||||
else 1,
|
||||
# default to sorting by block
|
||||
rbyd.block))
|
||||
|
||||
# choose an active rbyd
|
||||
rbyd = rbyds[0]
|
||||
# keep track of the other blocks
|
||||
rbyd = rbyds[i]
|
||||
rbyd.blocks += tuple(
|
||||
rbyds[(i+1+j) % len(rbyds)].block
|
||||
for j in range(len(rbyds)-1))
|
||||
rbyd.blocks = tuple(rbyd.block for rbyd in rbyds)
|
||||
# keep track of how many redund blocks are valid
|
||||
rbyd.redund = -1 + sum(1 for rbyd in rbyds
|
||||
if rbyd and rbyd.rev == rev and rbyd.trunk == trunk)
|
||||
# and patch the gcksumdelta if we have one
|
||||
if rbyd.gcksumdelta is not None:
|
||||
rbyd.gcksumdelta.blocks = rbyd.blocks
|
||||
@@ -661,7 +677,7 @@ class Rbyd:
|
||||
or rbyd.trunk != trunk
|
||||
or rbyd.weight != weight):
|
||||
# mark as corrupt and keep track of expected trunk/weight
|
||||
rbyd.corrupt = True
|
||||
rbyd.redund = -1
|
||||
rbyd.trunk = trunk
|
||||
rbyd.weight = weight
|
||||
|
||||
@@ -925,10 +941,6 @@ class Btree:
|
||||
def rev(self):
|
||||
return self.rbyd.rev
|
||||
|
||||
@property
|
||||
def eoff(self):
|
||||
return self.rbyd.eoff
|
||||
|
||||
@property
|
||||
def cksum(self):
|
||||
return self.rbyd.cksum
|
||||
|
||||
Reference in New Issue
Block a user