mirror of https://github.com/etcd-io/bbolt.git
188 lines
4.5 KiB
Go
188 lines
4.5 KiB
Go
package freelist
|
|
|
|
import (
|
|
"math/rand"
|
|
"reflect"
|
|
"sort"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"go.etcd.io/bbolt/internal/common"
|
|
)
|
|
|
|
func TestFreelistHashmap_init_panics(t *testing.T) {
|
|
f := NewHashMapFreelist()
|
|
require.Panics(t, func() {
|
|
// init expects sorted input
|
|
f.Init([]common.Pgid{25, 5})
|
|
})
|
|
}
|
|
|
|
func TestFreelistHashmap_allocate(t *testing.T) {
|
|
f := NewHashMapFreelist()
|
|
|
|
ids := []common.Pgid{3, 4, 5, 6, 7, 9, 12, 13, 18}
|
|
f.Init(ids)
|
|
|
|
f.Allocate(1, 3)
|
|
if x := f.FreeCount(); x != 6 {
|
|
t.Fatalf("exp=6; got=%v", x)
|
|
}
|
|
|
|
f.Allocate(1, 2)
|
|
if x := f.FreeCount(); x != 4 {
|
|
t.Fatalf("exp=4; got=%v", x)
|
|
}
|
|
f.Allocate(1, 1)
|
|
if x := f.FreeCount(); x != 3 {
|
|
t.Fatalf("exp=3; got=%v", x)
|
|
}
|
|
|
|
f.Allocate(1, 0)
|
|
if x := f.FreeCount(); x != 3 {
|
|
t.Fatalf("exp=3; got=%v", x)
|
|
}
|
|
}
|
|
|
|
func TestFreelistHashmap_mergeWithExist(t *testing.T) {
|
|
bm1 := pidSet{1: struct{}{}}
|
|
|
|
bm2 := pidSet{5: struct{}{}}
|
|
tests := []struct {
|
|
name string
|
|
ids common.Pgids
|
|
pgid common.Pgid
|
|
want common.Pgids
|
|
wantForwardmap map[common.Pgid]uint64
|
|
wantBackwardmap map[common.Pgid]uint64
|
|
wantfreemap map[uint64]pidSet
|
|
}{
|
|
{
|
|
name: "test1",
|
|
ids: []common.Pgid{1, 2, 4, 5, 6},
|
|
pgid: 3,
|
|
want: []common.Pgid{1, 2, 3, 4, 5, 6},
|
|
wantForwardmap: map[common.Pgid]uint64{1: 6},
|
|
wantBackwardmap: map[common.Pgid]uint64{6: 6},
|
|
wantfreemap: map[uint64]pidSet{6: bm1},
|
|
},
|
|
{
|
|
name: "test2",
|
|
ids: []common.Pgid{1, 2, 5, 6},
|
|
pgid: 3,
|
|
want: []common.Pgid{1, 2, 3, 5, 6},
|
|
wantForwardmap: map[common.Pgid]uint64{1: 3, 5: 2},
|
|
wantBackwardmap: map[common.Pgid]uint64{6: 2, 3: 3},
|
|
wantfreemap: map[uint64]pidSet{3: bm1, 2: bm2},
|
|
},
|
|
{
|
|
name: "test3",
|
|
ids: []common.Pgid{1, 2},
|
|
pgid: 3,
|
|
want: []common.Pgid{1, 2, 3},
|
|
wantForwardmap: map[common.Pgid]uint64{1: 3},
|
|
wantBackwardmap: map[common.Pgid]uint64{3: 3},
|
|
wantfreemap: map[uint64]pidSet{3: bm1},
|
|
},
|
|
{
|
|
name: "test4",
|
|
ids: []common.Pgid{2, 3},
|
|
pgid: 1,
|
|
want: []common.Pgid{1, 2, 3},
|
|
wantForwardmap: map[common.Pgid]uint64{1: 3},
|
|
wantBackwardmap: map[common.Pgid]uint64{3: 3},
|
|
wantfreemap: map[uint64]pidSet{3: bm1},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
f := newTestHashMapFreelist()
|
|
f.Init(tt.ids)
|
|
|
|
f.mergeWithExistingSpan(tt.pgid)
|
|
|
|
if got := f.freePageIds(); !reflect.DeepEqual(tt.want, got) {
|
|
t.Fatalf("name %s; exp=%v; got=%v", tt.name, tt.want, got)
|
|
}
|
|
if got := f.forwardMap; !reflect.DeepEqual(tt.wantForwardmap, got) {
|
|
t.Fatalf("name %s; exp=%v; got=%v", tt.name, tt.wantForwardmap, got)
|
|
}
|
|
if got := f.backwardMap; !reflect.DeepEqual(tt.wantBackwardmap, got) {
|
|
t.Fatalf("name %s; exp=%v; got=%v", tt.name, tt.wantBackwardmap, got)
|
|
}
|
|
if got := f.freemaps; !reflect.DeepEqual(tt.wantfreemap, got) {
|
|
t.Fatalf("name %s; exp=%v; got=%v", tt.name, tt.wantfreemap, got)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestFreelistHashmap_GetFreePageIDs(t *testing.T) {
|
|
f := newTestHashMapFreelist()
|
|
|
|
N := int32(100000)
|
|
fm := make(map[common.Pgid]uint64)
|
|
i := int32(0)
|
|
val := int32(0)
|
|
for i = 0; i < N; {
|
|
val = rand.Int31n(1000)
|
|
fm[common.Pgid(i)] = uint64(val)
|
|
i += val
|
|
f.freePagesCount += uint64(val)
|
|
}
|
|
|
|
f.forwardMap = fm
|
|
res := f.freePageIds()
|
|
|
|
if !sort.SliceIsSorted(res, func(i, j int) bool { return res[i] < res[j] }) {
|
|
t.Fatalf("pgids not sorted")
|
|
}
|
|
}
|
|
|
|
func Test_Freelist_Hashmap_Rollback(t *testing.T) {
|
|
f := newTestHashMapFreelist()
|
|
|
|
f.Init([]common.Pgid{3, 5, 6, 7, 12, 13})
|
|
|
|
f.Free(100, common.NewPage(20, 0, 0, 1))
|
|
f.Allocate(100, 3)
|
|
f.Free(100, common.NewPage(25, 0, 0, 0))
|
|
f.Allocate(100, 2)
|
|
|
|
require.Equal(t, map[common.Pgid]common.Txid{5: 100, 12: 100}, f.allocs)
|
|
require.Equal(t, map[common.Txid]*txPending{100: {
|
|
ids: []common.Pgid{20, 21, 25},
|
|
alloctx: []common.Txid{0, 0, 0},
|
|
}}, f.pending)
|
|
|
|
f.Rollback(100)
|
|
|
|
require.Equal(t, map[common.Pgid]common.Txid{}, f.allocs)
|
|
require.Equal(t, map[common.Txid]*txPending{}, f.pending)
|
|
}
|
|
|
|
func Benchmark_freelist_hashmapGetFreePageIDs(b *testing.B) {
|
|
f := newTestHashMapFreelist()
|
|
N := int32(100000)
|
|
fm := make(map[common.Pgid]uint64)
|
|
i := int32(0)
|
|
val := int32(0)
|
|
for i = 0; i < N; {
|
|
val = rand.Int31n(1000)
|
|
fm[common.Pgid(i)] = uint64(val)
|
|
i += val
|
|
}
|
|
|
|
f.forwardMap = fm
|
|
|
|
b.ReportAllocs()
|
|
b.ResetTimer()
|
|
for n := 0; n < b.N; n++ {
|
|
f.freePageIds()
|
|
}
|
|
}
|
|
|
|
func newTestHashMapFreelist() *hashMap {
|
|
f := NewHashMapFreelist()
|
|
return f.(*hashMap)
|
|
}
|