1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
|
// Copyright 2020 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 ld
import (
"cmd/link/internal/loader"
"testing"
)
func TestHeap(t *testing.T) {
tests := [][]loader.Sym{
{10, 20, 30, 40, 50, 60, 70, 80, 90, 100},
{100, 90, 80, 70, 60, 50, 40, 30, 20, 10},
{30, 50, 80, 20, 60, 70, 10, 100, 90, 40},
}
for _, s := range tests {
h := heap{}
for _, i := range s {
h.push(i)
if !verify(&h, 0) {
t.Errorf("heap invariant violated: %v", h)
}
}
for j := 0; j < len(s); j++ {
x := h.pop()
if !verify(&h, 0) {
t.Errorf("heap invariant violated: %v", h)
}
// pop should return elements in ascending order.
if want := loader.Sym((j + 1) * 10); x != want {
t.Errorf("pop returns wrong element: want %d, got %d", want, x)
}
}
if !h.empty() {
t.Errorf("heap is not empty after all pops")
}
}
// Also check that mixed pushes and pops work correctly.
for _, s := range tests {
h := heap{}
for i := 0; i < len(s)/2; i++ {
// two pushes, one pop
h.push(s[2*i])
if !verify(&h, 0) {
t.Errorf("heap invariant violated: %v", h)
}
h.push(s[2*i+1])
if !verify(&h, 0) {
t.Errorf("heap invariant violated: %v", h)
}
h.pop()
if !verify(&h, 0) {
t.Errorf("heap invariant violated: %v", h)
}
}
for !h.empty() { // pop remaining elements
h.pop()
if !verify(&h, 0) {
t.Errorf("heap invariant violated: %v", h)
}
}
}
}
// recursively verify heap-ness, starting at element i.
func verify(h *heap, i int) bool {
n := len(*h)
c1 := 2*i + 1 // left child
c2 := 2*i + 2 // right child
if c1 < n {
if (*h)[c1] < (*h)[i] {
return false
}
if !verify(h, c1) {
return false
}
}
if c2 < n {
if (*h)[c2] < (*h)[i] {
return false
}
if !verify(h, c2) {
return false
}
}
return true
}
|