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