diff options
Diffstat (limited to 'lib/btrfsprogs')
| -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++  	}  } | 
