diff options
author | Luke Shumaker <lukeshu@lukeshu.com> | 2022-06-30 02:57:11 -0600 |
---|---|---|
committer | Luke Shumaker <lukeshu@lukeshu.com> | 2022-06-30 02:57:11 -0600 |
commit | 428d86a1816d13ea6969793a42893c739487cf3d (patch) | |
tree | f7e05f2cf8eee25d86eca68fbddd539f43f8da35 /pkg | |
parent | 74f15aa6a5c04bab0615262d005eace4fc2baedb (diff) |
wip
Diffstat (limited to 'pkg')
-rw-r--r-- | pkg/btrfs/btrfsvol/lvm.go | 67 | ||||
-rw-r--r-- | pkg/rbtree/range.go | 8 |
2 files changed, 42 insertions, 33 deletions
diff --git a/pkg/btrfs/btrfsvol/lvm.go b/pkg/btrfs/btrfsvol/lvm.go index d11b2f0..01ecaf3 100644 --- a/pkg/btrfs/btrfsvol/lvm.go +++ b/pkg/btrfs/btrfsvol/lvm.go @@ -7,6 +7,7 @@ import ( "reflect" "sort" + "lukeshu.com/btrfs-tools/pkg/rbtree" "lukeshu.com/btrfs-tools/pkg/util" ) @@ -15,12 +16,34 @@ type LogicalVolume[PhysicalVolume util.File[PhysicalAddr]] struct { id2pv map[DeviceID]PhysicalVolume - logical2physical []chunkMapping - physical2logical map[DeviceID][]devextMapping + logical2physical *rbtree.Tree[LogicalAddr, chunkMapping] + physical2logical map[DeviceID]*rbtree.Tree[PhysicalAddr, devextMapping] } var _ util.File[LogicalAddr] = (*LogicalVolume[util.File[PhysicalAddr]])(nil) +func (lv *LogicalVolume[PhysicalVolume]) init() { + if lv.id2pv == nil { + lv.id2pv = make(map[DeviceID]PhysicalVolume) + } + if lv.logical2physical == nil { + lv.logical2physical = &rbtree.Tree[LogicalAddr, chunkMapping]{ + KeyFn: func(chunk chunkMapping) LogicalAddr { + return chunk.LAddr + }, + } + } + for devid := range lv.id2pv { + if _, ok := lv.physical2logical[devid]; !ok { + lv.physical2logical[devid] = &rbtree.Tree[LogicalAddr, chunkMapping]{ + KeyFn: func(ext devextMapping) LogicalAddr { + return ext.PAddr + }, + } + } + } +} + func (lv *LogicalVolume[PhysicalVolume]) SetName(name string) { lv.name = name } @@ -30,22 +53,26 @@ func (lv *LogicalVolume[PhysicalVolume]) Name() string { } func (lv *LogicalVolume[PhysicalVolume]) Size() (LogicalAddr, error) { - if len(lv.logical2physical) == 0 { + lv.init() + lastChunk := lv.logical2physical.Max() + if lastChunk == nil { return 0, nil } - lastChunk := lv.logical2physical[len(lv.logical2physical)-1] - return lastChunk.LAddr.Add(lastChunk.Size), nil + return lastChunk.Value.LAddr.Add(lastChunk.Value.Size), nil } func (lv *LogicalVolume[PhysicalVolume]) AddPhysicalVolume(id DeviceID, dev PhysicalVolume) error { - if lv.id2pv == nil { - lv.id2pv = make(map[DeviceID]PhysicalVolume) - } + lv.init() if other, exists := lv.id2pv[id]; exists { return fmt.Errorf("(%p).AddPhysicalVolume: cannot add physical volume %q: already have physical volume %q with id=%v", lv, dev.Name(), other.Name(), id) } lv.id2pv[id] = dev + lv.physical2logical[id] = &rbtree.Tree[LogicalAddr, chunkMapping]{ + KeyFn: func(ext devextMapping) LogicalAddr { + return ext.PAddr + }, + } return nil } @@ -70,14 +97,12 @@ type Mapping struct { } func (lv *LogicalVolume[PhysicalVolume]) AddMapping(m Mapping) error { + lv.init() // sanity check if _, haveDev := lv.id2pv[m.PAddr.Dev]; !haveDev { return fmt.Errorf("(%p).AddMapping: do not have a physical volume with id=%v", lv, m.PAddr.Dev) } - if lv.physical2logical == nil { - lv.physical2logical = make(map[DeviceID][]devextMapping) - } // logical2physical newChunk := chunkMapping{ @@ -86,15 +111,7 @@ func (lv *LogicalVolume[PhysicalVolume]) AddMapping(m Mapping) error { Size: m.Size, Flags: m.Flags, } - var logicalOverlaps []chunkMapping - for _, chunk := range lv.logical2physical { - switch newChunk.cmpRange(chunk) { - case 0: - logicalOverlaps = append(logicalOverlaps, chunk) - case 1: - break - } - } + logicalOverlaps := lv.logical2physical.SearchRange(newChunk.cmpRange) var err error newChunk, err = newChunk.union(logicalOverlaps...) if err != nil { @@ -108,15 +125,7 @@ func (lv *LogicalVolume[PhysicalVolume]) AddMapping(m Mapping) error { Size: m.Size, Flags: m.Flags, } - var physicalOverlaps []devextMapping - for _, ext := range lv.physical2logical[m.PAddr.Dev] { - switch newExt.cmpRange(ext) { - case 0: - physicalOverlaps = append(physicalOverlaps, ext) - case 1: - break - } - } + physicalOverlaps := lv.physical2logical[m.PAddr.Dev].SearchRange(newExt.cmpRange) newExt, err = newExt.union(physicalOverlaps...) if err != nil { return fmt.Errorf("(%p).AddMapping: %w", lv, err) diff --git a/pkg/rbtree/range.go b/pkg/rbtree/range.go index d7325ee..9f67078 100644 --- a/pkg/rbtree/range.go +++ b/pkg/rbtree/range.go @@ -6,18 +6,18 @@ import ( // SearchRange is like Search, but returns all nodes that match the // function; assuming that they are contiguous. -func (t *Tree[K, V]) SearchRange(fn func(V) int) []*Node[V] { +func (t *Tree[K, V]) SearchRange(fn func(V) int) []V { middle := t.Search(fn) if middle == nil { return nil } - ret := []*Node[V]{middle} + ret := []V{middle.Value} for node := t.Prev(middle); node != nil && fn(node.Value) == 0; node = t.Prev(node) { - ret = append(ret, node) + ret = append(ret, node.Value) } util.ReverseSlice(ret) for node := t.Next(middle); node != nil && fn(node.Value) == 0; node = t.Next(node) { - ret = append(ret, node) + ret = append(ret, node.Value) } return ret } |