| // 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 |
| } |