blob: 6cb0620f7b883c7c27eeec9cedf73d6027dc8c7b [file] [log] [blame] [edit]
// Copyright 2019 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package runtime_test
import (
"internal/goos"
"math/rand"
. "runtime"
"testing"
)
func checkPageCache(t *testing.T, got, want PageCache) {
if got.Base() != want.Base() {
t.Errorf("bad pageCache base: got 0x%x, want 0x%x", got.Base(), want.Base())
}
if got.Cache() != want.Cache() {
t.Errorf("bad pageCache bits: got %016x, want %016x", got.Base(), want.Base())
}
if got.Scav() != want.Scav() {
t.Errorf("bad pageCache scav: got %016x, want %016x", got.Scav(), want.Scav())
}
}
func TestPageCacheAlloc(t *testing.T) {
base := PageBase(BaseChunkIdx, 0)
type hit struct {
npages uintptr
base uintptr
scav uintptr
}
tests := map[string]struct {
cache PageCache
hits []hit
}{
"Empty": {
cache: NewPageCache(base, 0, 0),
hits: []hit{
{1, 0, 0},
{2, 0, 0},
{3, 0, 0},
{4, 0, 0},
{5, 0, 0},
{11, 0, 0},
{12, 0, 0},
{16, 0, 0},
{27, 0, 0},
{32, 0, 0},
{43, 0, 0},
{57, 0, 0},
{64, 0, 0},
{121, 0, 0},
},
},
"Lo1": {
cache: NewPageCache(base, 0x1, 0x1),
hits: []hit{
{1, base, PageSize},
{1, 0, 0},
{10, 0, 0},
},
},
"Hi1": {
cache: NewPageCache(base, 0x1<<63, 0x1),
hits: []hit{
{1, base + 63*PageSize, 0},
{1, 0, 0},
{10, 0, 0},
},
},
"Swiss1": {
cache: NewPageCache(base, 0x20005555, 0x5505),
hits: []hit{
{2, 0, 0},
{1, base, PageSize},
{1, base + 2*PageSize, PageSize},
{1, base + 4*PageSize, 0},
{1, base + 6*PageSize, 0},
{1, base + 8*PageSize, PageSize},
{1, base + 10*PageSize, PageSize},
{1, base + 12*PageSize, PageSize},
{1, base + 14*PageSize, PageSize},
{1, base + 29*PageSize, 0},
{1, 0, 0},
{10, 0, 0},
},
},
"Lo2": {
cache: NewPageCache(base, 0x3, 0x2<<62),
hits: []hit{
{2, base, 0},
{2, 0, 0},
{1, 0, 0},
},
},
"Hi2": {
cache: NewPageCache(base, 0x3<<62, 0x3<<62),
hits: []hit{
{2, base + 62*PageSize, 2 * PageSize},
{2, 0, 0},
{1, 0, 0},
},
},
"Swiss2": {
cache: NewPageCache(base, 0x3333<<31, 0x3030<<31),
hits: []hit{
{2, base + 31*PageSize, 0},
{2, base + 35*PageSize, 2 * PageSize},
{2, base + 39*PageSize, 0},
{2, base + 43*PageSize, 2 * PageSize},
{2, 0, 0},
},
},
"Hi53": {
cache: NewPageCache(base, ((uint64(1)<<53)-1)<<10, ((uint64(1)<<16)-1)<<10),
hits: []hit{
{53, base + 10*PageSize, 16 * PageSize},
{53, 0, 0},
{1, 0, 0},
},
},
"Full53": {
cache: NewPageCache(base, ^uint64(0), ((uint64(1)<<16)-1)<<10),
hits: []hit{
{53, base, 16 * PageSize},
{53, 0, 0},
{1, base + 53*PageSize, 0},
},
},
"Full64": {
cache: NewPageCache(base, ^uint64(0), ^uint64(0)),
hits: []hit{
{64, base, 64 * PageSize},
{64, 0, 0},
{1, 0, 0},
},
},
"FullMixed": {
cache: NewPageCache(base, ^uint64(0), ^uint64(0)),
hits: []hit{
{5, base, 5 * PageSize},
{7, base + 5*PageSize, 7 * PageSize},
{1, base + 12*PageSize, 1 * PageSize},
{23, base + 13*PageSize, 23 * PageSize},
{63, 0, 0},
{3, base + 36*PageSize, 3 * PageSize},
{3, base + 39*PageSize, 3 * PageSize},
{3, base + 42*PageSize, 3 * PageSize},
{12, base + 45*PageSize, 12 * PageSize},
{11, 0, 0},
{4, base + 57*PageSize, 4 * PageSize},
{4, 0, 0},
{6, 0, 0},
{36, 0, 0},
{2, base + 61*PageSize, 2 * PageSize},
{3, 0, 0},
{1, base + 63*PageSize, 1 * PageSize},
{4, 0, 0},
{2, 0, 0},
{62, 0, 0},
{1, 0, 0},
},
},
}
for name, test := range tests {
test := test
t.Run(name, func(t *testing.T) {
c := test.cache
for i, h := range test.hits {
b, s := c.Alloc(h.npages)
if b != h.base {
t.Fatalf("bad alloc base #%d: got 0x%x, want 0x%x", i, b, h.base)
}
if s != h.scav {
t.Fatalf("bad alloc scav #%d: got %d, want %d", i, s, h.scav)
}
}
})
}
}
func TestPageCacheFlush(t *testing.T) {
if GOOS == "openbsd" && testing.Short() {
t.Skip("skipping because virtual memory is limited; see #36210")
}
bits64ToBitRanges := func(bits uint64, base uint) []BitRange {
var ranges []BitRange
start, size := uint(0), uint(0)
for i := 0; i < 64; i++ {
if bits&(1<<i) != 0 {
if size == 0 {
start = uint(i) + base
}
size++
} else {
if size != 0 {
ranges = append(ranges, BitRange{start, size})
size = 0
}
}
}
if size != 0 {
ranges = append(ranges, BitRange{start, size})
}
return ranges
}
runTest := func(t *testing.T, base uint, cache, scav uint64) {
// Set up the before state.
beforeAlloc := map[ChunkIdx][]BitRange{
BaseChunkIdx: {{base, 64}},
}
beforeScav := map[ChunkIdx][]BitRange{
BaseChunkIdx: {},
}
b := NewPageAlloc(beforeAlloc, beforeScav)
defer FreePageAlloc(b)
// Create and flush the cache.
c := NewPageCache(PageBase(BaseChunkIdx, base), cache, scav)
c.Flush(b)
if !c.Empty() {
t.Errorf("pageCache flush did not clear cache")
}
// Set up the expected after state.
afterAlloc := map[ChunkIdx][]BitRange{
BaseChunkIdx: bits64ToBitRanges(^cache, base),
}
afterScav := map[ChunkIdx][]BitRange{
BaseChunkIdx: bits64ToBitRanges(scav, base),
}
want := NewPageAlloc(afterAlloc, afterScav)
defer FreePageAlloc(want)
// Check to see if it worked.
checkPageAlloc(t, want, b)
}
// Empty.
runTest(t, 0, 0, 0)
// Full.
runTest(t, 0, ^uint64(0), ^uint64(0))
// Random.
for i := 0; i < 100; i++ {
// Generate random valid base within a chunk.
base := uint(rand.Intn(PallocChunkPages/64)) * 64
// Generate random cache.
cache := rand.Uint64()
scav := rand.Uint64() & cache
// Run the test.
runTest(t, base, cache, scav)
}
}
func TestPageAllocAllocToCache(t *testing.T) {
if GOOS == "openbsd" && testing.Short() {
t.Skip("skipping because virtual memory is limited; see #36210")
}
type test struct {
beforeAlloc map[ChunkIdx][]BitRange
beforeScav map[ChunkIdx][]BitRange
hits []PageCache // expected base addresses and patterns
afterAlloc map[ChunkIdx][]BitRange
afterScav map[ChunkIdx][]BitRange
}
tests := map[string]test{
"AllFree": {
beforeAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {},
},
beforeScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{1, 1}, {64, 64}},
},
hits: []PageCache{
NewPageCache(PageBase(BaseChunkIdx, 0), ^uint64(0), 0x2),
NewPageCache(PageBase(BaseChunkIdx, 64), ^uint64(0), ^uint64(0)),
NewPageCache(PageBase(BaseChunkIdx, 128), ^uint64(0), 0),
NewPageCache(PageBase(BaseChunkIdx, 192), ^uint64(0), 0),
},
afterAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 256}},
},
},
"ManyArena": {
beforeAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 1: {{0, PallocChunkPages}},
BaseChunkIdx + 2: {{0, PallocChunkPages - 64}},
},
beforeScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 1: {{0, PallocChunkPages}},
BaseChunkIdx + 2: {},
},
hits: []PageCache{
NewPageCache(PageBase(BaseChunkIdx+2, PallocChunkPages-64), ^uint64(0), 0),
},
afterAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 1: {{0, PallocChunkPages}},
BaseChunkIdx + 2: {{0, PallocChunkPages}},
},
},
"NotContiguous": {
beforeAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 0xff: {{0, 0}},
},
beforeScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 0xff: {{31, 67}},
},
hits: []PageCache{
NewPageCache(PageBase(BaseChunkIdx+0xff, 0), ^uint64(0), ((uint64(1)<<33)-1)<<31),
},
afterAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 0xff: {{0, 64}},
},
afterScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
BaseChunkIdx + 0xff: {{64, 34}},
},
},
"First": {
beforeAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 32}, {33, 31}, {96, 32}},
},
beforeScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{1, 4}, {31, 5}, {66, 2}},
},
hits: []PageCache{
NewPageCache(PageBase(BaseChunkIdx, 0), 1<<32, 1<<32),
NewPageCache(PageBase(BaseChunkIdx, 64), (uint64(1)<<32)-1, 0x3<<2),
},
afterAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 128}},
},
},
"Fail": {
beforeAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
},
hits: []PageCache{
NewPageCache(0, 0, 0),
NewPageCache(0, 0, 0),
NewPageCache(0, 0, 0),
},
afterAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, PallocChunkPages}},
},
},
"RetainScavBits": {
beforeAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 1}, {10, 2}},
},
beforeScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 4}, {11, 1}},
},
hits: []PageCache{
NewPageCache(PageBase(BaseChunkIdx, 0), ^uint64(0x1|(0x3<<10)), 0x7<<1),
},
afterAlloc: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 64}},
},
afterScav: map[ChunkIdx][]BitRange{
BaseChunkIdx: {{0, 1}, {11, 1}},
},
},
}
// Disable these tests on iOS since we have a small address space.
// See #46860.
if PageAlloc64Bit != 0 && goos.IsIos == 0 {
const chunkIdxBigJump = 0x100000 // chunk index offset which translates to O(TiB)
// This test is similar to the one with the same name for
// pageAlloc.alloc and serves the same purpose.
// See mpagealloc_test.go for details.
sumsPerPhysPage := ChunkIdx(PhysPageSize / PallocSumBytes)
baseChunkIdx := BaseChunkIdx &^ (sumsPerPhysPage - 1)
tests["DiscontiguousMappedSumBoundary"] = test{
beforeAlloc: map[ChunkIdx][]BitRange{
baseChunkIdx + sumsPerPhysPage - 1: {{0, PallocChunkPages - 1}},
baseChunkIdx + chunkIdxBigJump: {{1, PallocChunkPages - 1}},
},
beforeScav: map[ChunkIdx][]BitRange{
baseChunkIdx + sumsPerPhysPage - 1: {},
baseChunkIdx + chunkIdxBigJump: {},
},
hits: []PageCache{
NewPageCache(PageBase(baseChunkIdx+sumsPerPhysPage-1, PallocChunkPages-64), 1<<63, 0),
NewPageCache(PageBase(baseChunkIdx+chunkIdxBigJump, 0), 1, 0),
NewPageCache(0, 0, 0),
},
afterAlloc: map[ChunkIdx][]BitRange{
baseChunkIdx + sumsPerPhysPage - 1: {{0, PallocChunkPages}},
baseChunkIdx + chunkIdxBigJump: {{0, PallocChunkPages}},
},
}
}
for name, v := range tests {
v := v
t.Run(name, func(t *testing.T) {
b := NewPageAlloc(v.beforeAlloc, v.beforeScav)
defer FreePageAlloc(b)
for _, expect := range v.hits {
checkPageCache(t, b.AllocToCache(), expect)
if t.Failed() {
return
}
}
want := NewPageAlloc(v.afterAlloc, v.afterScav)
defer FreePageAlloc(want)
checkPageAlloc(t, want, b)
})
}
}