Commit 9baa4301 authored by Michael Anthony Knyszek's avatar Michael Anthony Knyszek Committed by Michael Knyszek

runtime: merge all treaps into one implementation

This change modifies the treap implementation to support holding all
spans in a single treap, instead of keeping them all in separate treaps.

This improves ergonomics for nearly all treap-related callsites.
With that said, iteration is now more expensive, but it never occurs on
the fast path, only on scavenging-related paths.

This change opens up the opportunity for further optimizations, such as
splitting spans without treap removal (taking treap removal off the span
allocator's critical path) as well as improvements to treap iteration
(building linked lists for each iteration type and managing them on
insert/removal, since those operations should be less frequent).

For #30333.

Change-Id: I3dac97afd3682a37fda09ae8656a770e1369d0a9
Reviewed-on: https://go-review.googlesource.com/c/go/+/174398
Run-TryBot: Michael Knyszek <mknyszek@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarAustin Clements <austin@google.com>
parent 1ad2298c
...@@ -339,7 +339,7 @@ func ReadMemStatsSlow() (base, slow MemStats) { ...@@ -339,7 +339,7 @@ func ReadMemStatsSlow() (base, slow MemStats) {
slow.BySize[i].Frees = bySize[i].Frees slow.BySize[i].Frees = bySize[i].Frees
} }
for i := mheap_.scav.start(); i.valid(); i = i.next() { for i := mheap_.free.start(0, 0); i.valid(); i = i.next() {
slow.HeapReleased += uint64(i.span().released()) slow.HeapReleased += uint64(i.span().released())
} }
...@@ -522,11 +522,12 @@ type Span struct { ...@@ -522,11 +522,12 @@ type Span struct {
*mspan *mspan
} }
func AllocSpan(base, npages uintptr) Span { func AllocSpan(base, npages uintptr, scavenged bool) Span {
lock(&mheap_.lock) lock(&mheap_.lock)
s := (*mspan)(mheap_.spanalloc.alloc()) s := (*mspan)(mheap_.spanalloc.alloc())
unlock(&mheap_.lock) unlock(&mheap_.lock)
s.init(base, npages) s.init(base, npages)
s.scavenged = scavenged
return Span{s} return Span{s}
} }
...@@ -545,6 +546,17 @@ func (s Span) Pages() uintptr { ...@@ -545,6 +546,17 @@ func (s Span) Pages() uintptr {
return s.mspan.npages return s.mspan.npages
} }
type TreapIterType int
const (
TreapIterScav TreapIterType = TreapIterType(treapIterScav)
TreapIterBits = treapIterBits
)
func (s Span) MatchesIter(mask, match TreapIterType) bool {
return s.mspan.matchesIter(treapIterType(mask), treapIterType(match))
}
type TreapIter struct { type TreapIter struct {
treapIter treapIter
} }
...@@ -575,12 +587,12 @@ type Treap struct { ...@@ -575,12 +587,12 @@ type Treap struct {
mTreap mTreap
} }
func (t *Treap) Start() TreapIter { func (t *Treap) Start(mask, match TreapIterType) TreapIter {
return TreapIter{t.start()} return TreapIter{t.start(treapIterType(mask), treapIterType(match))}
} }
func (t *Treap) End() TreapIter { func (t *Treap) End(mask, match TreapIterType) TreapIter {
return TreapIter{t.end()} return TreapIter{t.end(treapIterType(mask), treapIterType(match))}
} }
func (t *Treap) Insert(s Span) { func (t *Treap) Insert(s Span) {
......
...@@ -185,6 +185,41 @@ func (t *treapNode) validateMaxPages() uintptr { ...@@ -185,6 +185,41 @@ func (t *treapNode) validateMaxPages() uintptr {
return max return max
} }
// treapIterType represents the type of iteration to perform
// over the treap. Each choice effectively represents a filter,
// i.e. spans that do not satisfy the conditions of the iteration
// type will be skipped over.
type treapIterType uint8
const (
treapIterScav treapIterType = 1 << iota // scavenged spans
treapIterBits = iota
)
// matches returns true if t satisfies the filter given by mask and match. mask
// is a bit-set of span properties to filter on.
//
// In other words, matches returns true if all properties set in mask have the
// value given by the corresponding bits in match.
func (t treapIterType) matches(mask, match treapIterType) bool {
return t&mask == match
}
// iterType returns the treapIterType associated with this span.
func (s *mspan) iterType() treapIterType {
have := treapIterType(0)
if s.scavenged {
have |= treapIterScav
}
return have
}
// matchesIter is a convenience method which checks if a span
// meets the criteria of the mask and match for an iter type.
func (s *mspan) matchesIter(mask, match treapIterType) bool {
return s.iterType().matches(mask, match)
}
// treapIter is a bidirectional iterator type which may be used to iterate over a // treapIter is a bidirectional iterator type which may be used to iterate over a
// an mTreap in-order forwards (increasing order) or backwards (decreasing order). // an mTreap in-order forwards (increasing order) or backwards (decreasing order).
// Its purpose is to hide details about the treap from users when trying to iterate // Its purpose is to hide details about the treap from users when trying to iterate
...@@ -192,7 +227,8 @@ func (t *treapNode) validateMaxPages() uintptr { ...@@ -192,7 +227,8 @@ func (t *treapNode) validateMaxPages() uintptr {
// //
// To create iterators over the treap, call start or end on an mTreap. // To create iterators over the treap, call start or end on an mTreap.
type treapIter struct { type treapIter struct {
t *treapNode mask, match treapIterType
t *treapNode
} }
// span returns the span at the current position in the treap. // span returns the span at the current position in the treap.
...@@ -211,6 +247,9 @@ func (i *treapIter) valid() bool { ...@@ -211,6 +247,9 @@ func (i *treapIter) valid() bool {
// ceases to be valid, calling next will panic. // ceases to be valid, calling next will panic.
func (i treapIter) next() treapIter { func (i treapIter) next() treapIter {
i.t = i.t.succ() i.t = i.t.succ()
for i.valid() && !i.span().matchesIter(i.mask, i.match) {
i.t = i.t.succ()
}
return i return i
} }
...@@ -218,12 +257,15 @@ func (i treapIter) next() treapIter { ...@@ -218,12 +257,15 @@ func (i treapIter) next() treapIter {
// ceases to be valid, calling prev will panic. // ceases to be valid, calling prev will panic.
func (i treapIter) prev() treapIter { func (i treapIter) prev() treapIter {
i.t = i.t.pred() i.t = i.t.pred()
for i.valid() && !i.span().matchesIter(i.mask, i.match) {
i.t = i.t.pred()
}
return i return i
} }
// start returns an iterator which points to the start of the treap (the // start returns an iterator which points to the start of the treap (the
// left-most node in the treap). // left-most node in the treap) subject to mask and match constraints.
func (root *mTreap) start() treapIter { func (root *mTreap) start(mask, match treapIterType) treapIter {
t := root.treap t := root.treap
if t == nil { if t == nil {
return treapIter{} return treapIter{}
...@@ -231,12 +273,15 @@ func (root *mTreap) start() treapIter { ...@@ -231,12 +273,15 @@ func (root *mTreap) start() treapIter {
for t.left != nil { for t.left != nil {
t = t.left t = t.left
} }
return treapIter{t: t} for t != nil && !t.span.matchesIter(mask, match) {
t = t.succ()
}
return treapIter{mask, match, t}
} }
// end returns an iterator which points to the end of the treap (the // end returns an iterator which points to the end of the treap (the
// right-most node in the treap). // right-most node in the treap) subject to mask and match constraints.
func (root *mTreap) end() treapIter { func (root *mTreap) end(mask, match treapIterType) treapIter {
t := root.treap t := root.treap
if t == nil { if t == nil {
return treapIter{} return treapIter{}
...@@ -244,7 +289,10 @@ func (root *mTreap) end() treapIter { ...@@ -244,7 +289,10 @@ func (root *mTreap) end() treapIter {
for t.right != nil { for t.right != nil {
t = t.right t = t.right
} }
return treapIter{t: t} for t != nil && !t.span.matchesIter(mask, match) {
t = t.pred()
}
return treapIter{mask, match, t}
} }
// insert adds span to the large span treap. // insert adds span to the large span treap.
...@@ -348,9 +396,10 @@ func (root *mTreap) removeNode(t *treapNode) { ...@@ -348,9 +396,10 @@ func (root *mTreap) removeNode(t *treapNode) {
mheap_.treapalloc.free(unsafe.Pointer(t)) mheap_.treapalloc.free(unsafe.Pointer(t))
} }
// find searches for, finds, and returns the treap iterator representing the // find searches for, finds, and returns the treap iterator over all spans
// position of the span with the smallest base address which is at least npages // representing the position of the span with the smallest base address which is
// in size. If no span has at least npages it returns an invalid iterator. // at least npages in size. If no span has at least npages it returns an invalid
// iterator.
// //
// This algorithm is as follows: // This algorithm is as follows:
// * If there's a left child and its subtree can satisfy this allocation, // * If there's a left child and its subtree can satisfy this allocation,
...@@ -391,7 +440,7 @@ func (root *mTreap) find(npages uintptr) treapIter { ...@@ -391,7 +440,7 @@ func (root *mTreap) find(npages uintptr) treapIter {
t = nil t = nil
} }
} }
return treapIter{t} return treapIter{t: t}
} }
// removeSpan searches for, finds, deletes span along with // removeSpan searches for, finds, deletes span along with
......
...@@ -30,8 +30,7 @@ const minPhysPageSize = 4096 ...@@ -30,8 +30,7 @@ const minPhysPageSize = 4096
//go:notinheap //go:notinheap
type mheap struct { type mheap struct {
lock mutex lock mutex
free mTreap // free and non-scavenged spans free mTreap // free spans
scav mTreap // free and scavenged spans
sweepgen uint32 // sweep generation, see comment in mspan sweepgen uint32 // sweep generation, see comment in mspan
sweepdone uint32 // all spans are swept sweepdone uint32 // all spans are swept
sweepers uint32 // number of active sweepone calls sweepers uint32 // number of active sweepone calls
...@@ -60,7 +59,7 @@ type mheap struct { ...@@ -60,7 +59,7 @@ type mheap struct {
// on the swept stack. // on the swept stack.
sweepSpans [2]gcSweepBuf sweepSpans [2]gcSweepBuf
_ uint32 // align uint64 fields on 32-bit for atomics // _ uint32 // align uint64 fields on 32-bit for atomics
// Proportional sweep // Proportional sweep
// //
...@@ -464,7 +463,7 @@ func (h *mheap) coalesce(s *mspan) { ...@@ -464,7 +463,7 @@ func (h *mheap) coalesce(s *mspan) {
// The size is potentially changing so the treap needs to delete adjacent nodes and // The size is potentially changing so the treap needs to delete adjacent nodes and
// insert back as a combined node. // insert back as a combined node.
h.treapForSpan(other).removeSpan(other) h.free.removeSpan(other)
other.state = mSpanDead other.state = mSpanDead
h.spanalloc.free(unsafe.Pointer(other)) h.spanalloc.free(unsafe.Pointer(other))
} }
...@@ -482,7 +481,7 @@ func (h *mheap) coalesce(s *mspan) { ...@@ -482,7 +481,7 @@ func (h *mheap) coalesce(s *mspan) {
return return
} }
// Since we're resizing other, we must remove it from the treap. // Since we're resizing other, we must remove it from the treap.
h.treapForSpan(other).removeSpan(other) h.free.removeSpan(other)
// Round boundary to the nearest physical page size, toward the // Round boundary to the nearest physical page size, toward the
// scavenged span. // scavenged span.
...@@ -500,7 +499,7 @@ func (h *mheap) coalesce(s *mspan) { ...@@ -500,7 +499,7 @@ func (h *mheap) coalesce(s *mspan) {
h.setSpan(boundary, b) h.setSpan(boundary, b)
// Re-insert other now that it has a new size. // Re-insert other now that it has a new size.
h.treapForSpan(other).insert(other) h.free.insert(other)
} }
// Coalesce with earlier, later spans. // Coalesce with earlier, later spans.
...@@ -1101,57 +1100,27 @@ func (h *mheap) setSpans(base, npage uintptr, s *mspan) { ...@@ -1101,57 +1100,27 @@ func (h *mheap) setSpans(base, npage uintptr, s *mspan) {
} }
} }
// treapForSpan returns the appropriate treap for a span for
// insertion and removal.
func (h *mheap) treapForSpan(span *mspan) *mTreap {
if span.scavenged {
return &h.scav
}
return &h.free
}
// pickFreeSpan acquires a free span from internal free list
// structures if one is available. Otherwise returns nil.
// h must be locked.
func (h *mheap) pickFreeSpan(npage uintptr) *mspan {
tf := h.free.find(npage)
ts := h.scav.find(npage)
// Check for whichever treap gave us the smaller, non-nil result.
// Note that we want the _smaller_ free span, i.e. the free span
// closer in size to the amount we requested (npage).
var s *mspan
if tf.valid() && (!ts.valid() || tf.span().base() <= ts.span().base()) {
s = tf.span()
h.free.erase(tf)
} else if ts.valid() && (!tf.valid() || tf.span().base() > ts.span().base()) {
s = ts.span()
h.scav.erase(ts)
}
return s
}
// Allocates a span of the given size. h must be locked. // Allocates a span of the given size. h must be locked.
// The returned span has been removed from the // The returned span has been removed from the
// free structures, but its state is still mSpanFree. // free structures, but its state is still mSpanFree.
func (h *mheap) allocSpanLocked(npage uintptr, stat *uint64) *mspan { func (h *mheap) allocSpanLocked(npage uintptr, stat *uint64) *mspan {
var s *mspan t := h.free.find(npage)
if t.valid() {
s = h.pickFreeSpan(npage)
if s != nil {
goto HaveSpan goto HaveSpan
} }
// On failure, grow the heap and try again.
if !h.grow(npage) { if !h.grow(npage) {
return nil return nil
} }
s = h.pickFreeSpan(npage) t = h.free.find(npage)
if s != nil { if t.valid() {
goto HaveSpan goto HaveSpan
} }
throw("grew heap, but no adequate free span found") throw("grew heap, but no adequate free span found")
HaveSpan: HaveSpan:
s := t.span()
h.free.erase(t)
// Mark span in use. // Mark span in use.
if s.state != mSpanFree { if s.state != mSpanFree {
throw("candidate mspan for allocation is not free") throw("candidate mspan for allocation is not free")
...@@ -1339,8 +1308,8 @@ func (h *mheap) freeSpanLocked(s *mspan, acctinuse, acctidle bool, unusedsince i ...@@ -1339,8 +1308,8 @@ func (h *mheap) freeSpanLocked(s *mspan, acctinuse, acctidle bool, unusedsince i
// Coalesce span with neighbors. // Coalesce span with neighbors.
h.coalesce(s) h.coalesce(s)
// Insert s into the appropriate treap. // Insert s into the treap.
h.treapForSpan(s).insert(s) h.free.insert(s)
} }
// scavengeLocked scavenges nbytes worth of spans in the free treap by // scavengeLocked scavenges nbytes worth of spans in the free treap by
...@@ -1355,10 +1324,11 @@ func (h *mheap) scavengeLocked(nbytes uintptr) { ...@@ -1355,10 +1324,11 @@ func (h *mheap) scavengeLocked(nbytes uintptr) {
h.scavengeCredit -= nbytes h.scavengeCredit -= nbytes
return return
} }
// Iterate over the treap backwards (from highest address to lowest address) // Iterate over the unscavenged spans in the treap backwards (from highest
// scavenging spans until we've reached our quota of nbytes. // address to lowest address) scavenging spans until we've reached our
// quota of nbytes.
released := uintptr(0) released := uintptr(0)
for t := h.free.end(); released < nbytes && t.valid(); { for t := h.free.end(treapIterScav, 0); released < nbytes && t.valid(); {
s := t.span() s := t.span()
r := s.scavenge() r := s.scavenge()
if r == 0 { if r == 0 {
...@@ -1373,7 +1343,7 @@ func (h *mheap) scavengeLocked(nbytes uintptr) { ...@@ -1373,7 +1343,7 @@ func (h *mheap) scavengeLocked(nbytes uintptr) {
// the same scavenged state adjacent to each other. // the same scavenged state adjacent to each other.
h.coalesce(s) h.coalesce(s)
t = n t = n
h.scav.insert(s) h.free.insert(s)
released += r released += r
} }
// If we over-scavenged, turn that extra amount into credit. // If we over-scavenged, turn that extra amount into credit.
...@@ -1386,20 +1356,19 @@ func (h *mheap) scavengeLocked(nbytes uintptr) { ...@@ -1386,20 +1356,19 @@ func (h *mheap) scavengeLocked(nbytes uintptr) {
// treapNode's span. It then removes the scavenged span from // treapNode's span. It then removes the scavenged span from
// unscav and adds it into scav before continuing. h must be locked. // unscav and adds it into scav before continuing. h must be locked.
func (h *mheap) scavengeAllLocked(now, limit uint64) uintptr { func (h *mheap) scavengeAllLocked(now, limit uint64) uintptr {
// Iterate over the treap scavenging spans if unused for at least limit time. // Iterate over the unscavenged spans in the treap scavenging spans
// if unused for at least limit time.
released := uintptr(0) released := uintptr(0)
for t := h.free.start(); t.valid(); { for t := h.free.start(treapIterScav, 0); t.valid(); {
s := t.span() s := t.span()
n := t.next() n := t.next()
if (now - uint64(s.unusedsince)) > limit { if (now - uint64(s.unusedsince)) > limit {
r := s.scavenge() r := s.scavenge()
if r != 0 { if r != 0 {
h.free.erase(t) h.free.erase(t)
// Now that s is scavenged, we must eagerly coalesce it // See (*mheap).scavenge.
// with its neighbors to prevent having two spans with
// the same scavenged state adjacent to each other.
h.coalesce(s) h.coalesce(s)
h.scav.insert(s) h.free.insert(s)
released += r released += r
} }
} }
......
...@@ -5,19 +5,23 @@ ...@@ -5,19 +5,23 @@
package runtime_test package runtime_test
import ( import (
"fmt"
"runtime" "runtime"
"testing" "testing"
) )
var spanDesc = map[uintptr]uintptr{ var spanDesc = map[uintptr]struct {
0xc0000000: 2, pages uintptr
0xc0006000: 1, scav bool
0xc0010000: 8, }{
0xc0022000: 7, 0xc0000000: {2, false},
0xc0034000: 4, 0xc0006000: {1, false},
0xc0040000: 5, 0xc0010000: {8, false},
0xc0050000: 5, 0xc0022000: {7, false},
0xc0060000: 5000, 0xc0034000: {4, true},
0xc0040000: {5, false},
0xc0050000: {5, true},
0xc0060000: {5000, false},
} }
// Wrap the Treap one more time because go:notinheap doesn't // Wrap the Treap one more time because go:notinheap doesn't
...@@ -28,6 +32,10 @@ type treap struct { ...@@ -28,6 +32,10 @@ type treap struct {
runtime.Treap runtime.Treap
} }
func maskMatchName(mask, match runtime.TreapIterType) string {
return fmt.Sprintf("%0*b-%0*b", runtime.TreapIterBits, uint8(mask), runtime.TreapIterBits, uint8(match))
}
// This test ensures that the treap implementation in the runtime // This test ensures that the treap implementation in the runtime
// maintains all stated invariants after different sequences of // maintains all stated invariants after different sequences of
// insert, removeSpan, find, and erase. Invariants specific to the // insert, removeSpan, find, and erase. Invariants specific to the
...@@ -36,12 +44,37 @@ type treap struct { ...@@ -36,12 +44,37 @@ type treap struct {
// treap. // treap.
func TestTreap(t *testing.T) { func TestTreap(t *testing.T) {
// Set up a bunch of spans allocated into mheap_. // Set up a bunch of spans allocated into mheap_.
// Also, derive a set of typeCounts of each type of span
// according to runtime.TreapIterType so we can verify against
// them later.
spans := make([]runtime.Span, 0, len(spanDesc)) spans := make([]runtime.Span, 0, len(spanDesc))
for base, pages := range spanDesc { typeCounts := [1 << runtime.TreapIterBits][1 << runtime.TreapIterBits]int{}
s := runtime.AllocSpan(base, pages) for base, de := range spanDesc {
s := runtime.AllocSpan(base, de.pages, de.scav)
defer s.Free() defer s.Free()
spans = append(spans, s) spans = append(spans, s)
for i := runtime.TreapIterType(0); i < 1<<runtime.TreapIterBits; i++ {
for j := runtime.TreapIterType(0); j < 1<<runtime.TreapIterBits; j++ {
if s.MatchesIter(i, j) {
typeCounts[i][j]++
}
}
}
} }
t.Run("TypeCountsSanity", func(t *testing.T) {
// Just sanity check type counts for a few values.
check := func(mask, match runtime.TreapIterType, count int) {
tc := typeCounts[mask][match]
if tc != count {
name := maskMatchName(mask, match)
t.Fatalf("failed a sanity check for mask/match %s counts: got %d, wanted %d", name, tc, count)
}
}
check(0, 0, len(spanDesc))
check(runtime.TreapIterScav, 0, 6)
check(runtime.TreapIterScav, runtime.TreapIterScav, 2)
})
t.Run("Insert", func(t *testing.T) { t.Run("Insert", func(t *testing.T) {
tr := treap{} tr := treap{}
// Test just a very basic insert/remove for sanity. // Test just a very basic insert/remove for sanity.
...@@ -77,61 +110,74 @@ func TestTreap(t *testing.T) { ...@@ -77,61 +110,74 @@ func TestTreap(t *testing.T) {
} }
}) })
t.Run("Iterate", func(t *testing.T) { t.Run("Iterate", func(t *testing.T) {
t.Run("StartToEnd", func(t *testing.T) { for mask := runtime.TreapIterType(0); mask < 1<<runtime.TreapIterBits; mask++ {
// Ensure progressing an iterator actually goes over the whole treap for match := runtime.TreapIterType(0); match < 1<<runtime.TreapIterBits; match++ {
// from the start and that it iterates over the elements in order. iterName := maskMatchName(mask, match)
// Also ensures that Start returns a valid iterator. t.Run(iterName, func(t *testing.T) {
tr := treap{} t.Run("StartToEnd", func(t *testing.T) {
for _, s := range spans { // Ensure progressing an iterator actually goes over the whole treap
tr.Insert(s) // from the start and that it iterates over the elements in order.
// Furthermore, ensure that it only iterates over the relevant parts
// of the treap.
// Finally, ensures that Start returns a valid iterator.
tr := treap{}
for _, s := range spans {
tr.Insert(s)
}
nspans := 0
lastBase := uintptr(0)
for i := tr.Start(mask, match); i.Valid(); i = i.Next() {
nspans++
if lastBase > i.Span().Base() {
t.Fatalf("not iterating in correct order: encountered base %x before %x", lastBase, i.Span().Base())
}
lastBase = i.Span().Base()
if !i.Span().MatchesIter(mask, match) {
t.Fatalf("found non-matching span while iteration over mask/match %s: base %x", iterName, i.Span().Base())
}
}
if nspans != typeCounts[mask][match] {
t.Fatal("failed to iterate forwards over full treap")
}
for _, s := range spans {
tr.RemoveSpan(s)
}
})
t.Run("EndToStart", func(t *testing.T) {
// See StartToEnd tests.
tr := treap{}
for _, s := range spans {
tr.Insert(s)
}
nspans := 0
lastBase := ^uintptr(0)
for i := tr.End(mask, match); i.Valid(); i = i.Prev() {
nspans++
if lastBase < i.Span().Base() {
t.Fatalf("not iterating in correct order: encountered base %x before %x", lastBase, i.Span().Base())
}
lastBase = i.Span().Base()
if !i.Span().MatchesIter(mask, match) {
t.Fatalf("found non-matching span while iteration over mask/match %s: base %x", iterName, i.Span().Base())
}
}
if nspans != typeCounts[mask][match] {
t.Fatal("failed to iterate backwards over full treap")
}
for _, s := range spans {
tr.RemoveSpan(s)
}
})
})
} }
nspans := 0 }
lastBase := uintptr(0)
for i := tr.Start(); i.Valid(); i = i.Next() {
nspans++
if lastBase > i.Span().Base() {
t.Fatalf("not iterating in correct order: encountered base %x before %x", lastBase, i.Span().Base())
}
lastBase = i.Span().Base()
}
if nspans != len(spans) {
t.Fatal("failed to iterate forwards over full treap")
}
for _, s := range spans {
tr.RemoveSpan(s)
}
})
t.Run("EndToStart", func(t *testing.T) {
// Ensure progressing an iterator actually goes over the whole treap
// from the end and that it iterates over the elements in reverse
// order. Also ensures that End returns a valid iterator.
tr := treap{}
for _, s := range spans {
tr.Insert(s)
}
nspans := 0
lastBase := ^uintptr(0)
for i := tr.End(); i.Valid(); i = i.Prev() {
nspans++
if lastBase < i.Span().Base() {
t.Fatalf("not iterating in correct order: encountered base %x before %x", lastBase, i.Span().Base())
}
lastBase = i.Span().Base()
}
if nspans != len(spans) {
t.Fatal("failed to iterate backwards over full treap")
}
for _, s := range spans {
tr.RemoveSpan(s)
}
})
t.Run("Prev", func(t *testing.T) { t.Run("Prev", func(t *testing.T) {
// Test the iterator invariant that i.prev().next() == i. // Test the iterator invariant that i.prev().next() == i.
tr := treap{} tr := treap{}
for _, s := range spans { for _, s := range spans {
tr.Insert(s) tr.Insert(s)
} }
i := tr.Start().Next().Next() i := tr.Start(0, 0).Next().Next()
p := i.Prev() p := i.Prev()
if !p.Valid() { if !p.Valid() {
t.Fatal("i.prev() is invalid") t.Fatal("i.prev() is invalid")
...@@ -149,7 +195,7 @@ func TestTreap(t *testing.T) { ...@@ -149,7 +195,7 @@ func TestTreap(t *testing.T) {
for _, s := range spans { for _, s := range spans {
tr.Insert(s) tr.Insert(s)
} }
i := tr.Start().Next().Next() i := tr.Start(0, 0).Next().Next()
n := i.Next() n := i.Next()
if !n.Valid() { if !n.Valid() {
t.Fatal("i.next() is invalid") t.Fatal("i.next() is invalid")
...@@ -169,7 +215,7 @@ func TestTreap(t *testing.T) { ...@@ -169,7 +215,7 @@ func TestTreap(t *testing.T) {
for _, s := range spans { for _, s := range spans {
tr.Insert(s) tr.Insert(s)
} }
i := tr.Start().Next().Next().Next() i := tr.Start(0, 0).Next().Next().Next()
s := i.Span() s := i.Span()
n := i.Next() n := i.Next()
p := i.Prev() p := i.Prev()
...@@ -191,7 +237,7 @@ func TestTreap(t *testing.T) { ...@@ -191,7 +237,7 @@ func TestTreap(t *testing.T) {
for _, s := range spans { for _, s := range spans {
tr.Insert(s) tr.Insert(s)
} }
for i := tr.Start(); i.Valid(); { for i := tr.Start(0, 0); i.Valid(); {
n := i.Next() n := i.Next()
tr.Erase(i) tr.Erase(i)
i = n i = n
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment