forked from project-machine/disko
-
Notifications
You must be signed in to change notification settings - Fork 1
/
util.go
63 lines (54 loc) · 1.73 KB
/
util.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
package disko
import (
"fmt"
"sort"
)
type uRange struct {
Start, Last uint64
}
func (r *uRange) Size() uint64 {
return r.Last - r.Start + 1
}
type uRanges []uRange
func (r uRanges) Len() int { return len(r) }
func (r uRanges) Less(i, j int) bool { return r[i].Start < r[j].Start }
func (r uRanges) Swap(i, j int) { r[i], r[j] = r[j], r[i] }
// findRangeGaps returns a set of uRange to represent the un-used
// uint64 between min and max that are not included in ranges.
// findRangeGaps({{10, 40}, {50, 100}}, 0, 110}) ==
// {{0, 9}, {41, 49}, {101, 110}}
// Note that input list will be sorted.
func findRangeGaps(ranges uRanges, min, max uint64) uRanges {
// start 'ret' off with full range of min to max, then start cutting it up.
ret := uRanges{{min, max}}
sort.Sort(ranges)
for _, i := range ranges {
for r := 0; r < len(ret); r++ {
// 5 cases:
if i.Start > ret[r].Last || i.Last < ret[r].Start {
// a. i has no overlap
} else if i.Start <= ret[r].Start && i.Last >= ret[r].Last {
// b.) i is complete superset, so remove ret[r]
ret = append(ret[:r], ret[r+1:]...)
r--
} else if i.Start > ret[r].Start && i.Last < ret[r].Last {
// c.) i is strict subset: split ret[r]
ret = append(
append(ret[:r+1], uRange{i.Last + 1, ret[r].Last}),
ret[r+1:]...)
ret[r].Last = i.Start - 1
r++ // added entry is guaranteed to be 'a', so skip it.
} else if i.Start <= ret[r].Start {
// d.) overlap left edge to middle
ret[r].Start = i.Last + 1
} else if i.Start <= ret[r].Last {
// e.) middle to right edge (possibly past).
ret[r].Last = i.Start - 1
} else {
panic(fmt.Sprintf("Error in findRangeGaps: %v, r=%d, ret=%v",
i, r, ret))
}
}
}
return ret
}