diff options
-rw-r--r-- | lib/btrfsprogs/btrfsinspect/rebuildnodes/rebuild.go | 77 |
1 files changed, 55 insertions, 22 deletions
diff --git a/lib/btrfsprogs/btrfsinspect/rebuildnodes/rebuild.go b/lib/btrfsprogs/btrfsinspect/rebuildnodes/rebuild.go index d77a4ea..0116abb 100644 --- a/lib/btrfsprogs/btrfsinspect/rebuildnodes/rebuild.go +++ b/lib/btrfsprogs/btrfsinspect/rebuildnodes/rebuild.go @@ -10,6 +10,7 @@ import ( "fmt" "runtime" "sort" + "strings" "time" "github.com/datawire/dlib/dgroup" @@ -62,6 +63,7 @@ type rebuilder struct { } type treeAugmentQueue struct { + keyBuf strings.Builder zero map[string]struct{} single map[string]btrfsvol.LogicalAddr multi map[string]containers.Set[btrfsvol.LogicalAddr] @@ -469,26 +471,38 @@ func (o *rebuilder) resolveTreeAugments(ctx context.Context, treeID btrfsprim.Ob // Free some memory o.augmentQueue[treeID].single = nil o.augmentQueue[treeID].multi = nil + o.augmentQueue[treeID].keyBuf.Reset() return ret } //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// -func (o *rebuilder) hasAugment(treeID btrfsprim.ObjID, wantKey string) bool { - if treeQueue, ok := o.augmentQueue[treeID]; ok { +func shortenWantKey(wantKey string) string { + if !strings.HasPrefix(wantKey, "tree=") { + panic("should not happen") + } + sp := strings.IndexByte(wantKey, ' ') + if sp < 0 { + panic("should not happen") + } + return wantKey[sp+1:] +} + +func (treeQueue *treeAugmentQueue) has(wantKey string) bool { + if treeQueue != nil { if treeQueue.zero != nil { - if _, ok := treeQueue.zero[wantKey]; ok { + if _, ok := treeQueue.zero[shortenWantKey(wantKey)]; ok { return true } } if treeQueue.single != nil { - if _, ok := treeQueue.single[wantKey]; ok { + if _, ok := treeQueue.single[shortenWantKey(wantKey)]; ok { return true } } if treeQueue.multi != nil { - if _, ok := treeQueue.multi[wantKey]; ok { + if _, ok := treeQueue.multi[shortenWantKey(wantKey)]; ok { return true } } @@ -496,31 +510,50 @@ func (o *rebuilder) hasAugment(treeID btrfsprim.ObjID, wantKey string) bool { return false } -func (o *rebuilder) wantAugment(ctx context.Context, treeID btrfsprim.ObjID, wantKey string, choices containers.Set[btrfsvol.LogicalAddr]) { - if o.augmentQueue[treeID] == nil { - o.augmentQueue[treeID] = new(treeAugmentQueue) +func (treeQueue *treeAugmentQueue) store(wantKey string, choices containers.Set[btrfsvol.LogicalAddr]) { + wantKey = shortenWantKey(wantKey) + beg := treeQueue.keyBuf.Len() + if treeQueue.keyBuf.Cap()-beg < len(wantKey) { + treeQueue.keyBuf.Reset() + treeQueue.keyBuf.Grow(textui.Tunable(4096)) + beg = 0 } + treeQueue.keyBuf.WriteString(wantKey) + wantKey = treeQueue.keyBuf.String()[beg:] + switch len(choices) { case 0: - dlog.Error(ctx, "could not find wanted item") - if o.augmentQueue[treeID].zero == nil { - o.augmentQueue[treeID].zero = make(map[string]struct{}) + if treeQueue.zero == nil { + treeQueue.zero = make(map[string]struct{}) } - o.augmentQueue[treeID].zero[wantKey] = struct{}{} - o.numAugmentFailures++ + treeQueue.zero[wantKey] = struct{}{} case 1: - dlog.Infof(ctx, "choices=%v", maps.SortedKeys(choices)) - if o.augmentQueue[treeID].single == nil { - o.augmentQueue[treeID].single = make(map[string]btrfsvol.LogicalAddr) + if treeQueue.single == nil { + treeQueue.single = make(map[string]btrfsvol.LogicalAddr) } - o.augmentQueue[treeID].single[wantKey] = choices.TakeOne() - o.numAugments++ + treeQueue.single[wantKey] = choices.TakeOne() default: - dlog.Infof(ctx, "choices=%v", maps.SortedKeys(choices)) - if o.augmentQueue[treeID].multi == nil { - o.augmentQueue[treeID].multi = make(map[string]containers.Set[btrfsvol.LogicalAddr]) + if treeQueue.multi == nil { + treeQueue.multi = make(map[string]containers.Set[btrfsvol.LogicalAddr]) } - o.augmentQueue[treeID].multi[wantKey] = choices + treeQueue.multi[wantKey] = choices + } +} + +func (o *rebuilder) hasAugment(treeID btrfsprim.ObjID, wantKey string) bool { + return o.augmentQueue[treeID].has(wantKey) +} + +func (o *rebuilder) wantAugment(ctx context.Context, treeID btrfsprim.ObjID, wantKey string, choices containers.Set[btrfsvol.LogicalAddr]) { + if o.augmentQueue[treeID] == nil { + o.augmentQueue[treeID] = new(treeAugmentQueue) + } + o.augmentQueue[treeID].store(wantKey, choices) + if len(choices) == 0 { + dlog.Error(ctx, "could not find wanted item") + o.numAugmentFailures++ + } else { + dlog.Infof(ctx, "choices=%v", maps.SortedKeys(choices)) o.numAugments++ } } |