-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathadjust_heights_heap.go
134 lines (122 loc) · 3.79 KB
/
adjust_heights_heap.go
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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
package incr
import (
"fmt"
"sync"
)
func newAdjustHeightsHeap(maxHeightAllowed int) *adjustHeightsHeap {
return &adjustHeightsHeap{
nodesByHeight: make([]*queue[INode], maxHeightAllowed),
heightLowerBound: maxHeightAllowed + 1,
}
}
// adjustHeightsHeap is a specialized heap that handles setting node heights
// such that we don't recompute heights of nodes more than once.
//
// the way the heap works is as we add nodes, we ensure that the invariant
// that a child's node height is more than it's parents. to do that we
// add the parent to a [nodesByHeight] list, and any children to the
// height above that, recursing through the children adding more as we
// see them, preserving the invariant.
type adjustHeightsHeap struct {
mu sync.Mutex
nodesByHeight []*queue[INode]
numNodes int
maxHeightSeen int
heightLowerBound int
}
func (ah *adjustHeightsHeap) len() int {
return ah.numNodes
}
func (ah *adjustHeightsHeap) maxHeightAllowed() int {
return len(ah.nodesByHeight) - 1
}
func (ah *adjustHeightsHeap) setHeight(node INode, height int) error {
ah.mu.Lock()
defer ah.mu.Unlock()
return ah.setHeightUnsafe(node, height)
}
func (ah *adjustHeightsHeap) adjustHeights(rh *recomputeHeap, originalChild, originalParent INode) error {
ah.mu.Lock()
rh.mu.Lock()
defer ah.mu.Unlock()
defer rh.mu.Unlock()
ah.heightLowerBound = originalChild.Node().height
if err := ah.ensureHeightRequirementUnsafe(originalChild, originalParent, originalChild, originalParent); err != nil {
return err
}
for ah.numNodes > 0 {
parent, _ := ah.removeMinUnsafe()
if parent.Node().heightInRecomputeHeap != HeightUnset {
rh.fixUnsafe(parent)
}
for _, child := range parent.Node().children {
if err := ah.ensureHeightRequirementUnsafe(originalChild, originalParent, child, parent); err != nil {
return err
}
}
if typed, typedOK := parent.(IBindChange); typedOK {
for _, nodeOnRight := range typed.RightScopeNodes() {
if nodeOnRight.Node().isNecessary() {
if err := ah.ensureHeightRequirementUnsafe(originalChild, originalParent, nodeOnRight, parent); err != nil {
return err
}
}
}
}
}
return nil
}
func (ah *adjustHeightsHeap) ensureHeightRequirementUnsafe(originalChild, originalParent, child, parent INode) error {
if originalParent.Node().id == child.Node().id {
return fmt.Errorf("cycle detected at %v to %v", originalChild, originalParent)
}
if parent.Node().height >= child.Node().height {
// we set `child.height` after adding `child` to the heap, so that `child` goes
// in the heap with its pre-adjusted height.
ah.addUnsafe(child)
if err := ah.setHeightUnsafe(child, parent.Node().height+1); err != nil {
return err
}
}
return nil
}
func (ah *adjustHeightsHeap) removeMinUnsafe() (node INode, ok bool) {
if ah.numNodes == 0 {
return
}
for x := ah.heightLowerBound; x <= ah.maxHeightSeen; x++ {
if ah.nodesByHeight[x] != nil && ah.nodesByHeight[x].len() > 0 {
node, ok = ah.nodesByHeight[x].pop()
ah.heightLowerBound = x
node.Node().heightInAdjustHeightsHeap = HeightUnset
ah.numNodes--
return
}
}
return
}
func (ah *adjustHeightsHeap) addUnsafe(node INode) {
if node.Node().heightInAdjustHeightsHeap != HeightUnset {
return
}
height := node.Node().height
node.Node().heightInAdjustHeightsHeap = height
if ah.nodesByHeight[height] == nil {
ah.nodesByHeight[height] = new(queue[INode])
}
if height > ah.maxHeightSeen {
ah.maxHeightSeen = height
}
ah.nodesByHeight[height].push(node)
ah.numNodes++
}
func (ah *adjustHeightsHeap) setHeightUnsafe(node INode, height int) error {
if height > ah.maxHeightAllowed() {
return fmt.Errorf("cannot set node height above %d", ah.maxHeightAllowed())
}
if height > ah.maxHeightSeen {
ah.maxHeightSeen = height
}
node.Node().height = height
return nil
}