This commit is contained in:
Ryan Oldenburg 2021-11-21 16:54:39 -06:00
parent 56a9981f91
commit 27baf16e9b

View file

@ -1120,38 +1120,38 @@ proc maxEntryCount(partitioning: Partitioning): int =
result = max(result, partitioning.partitions[i].len) result = max(result, partitioning.partitions[i].len)
proc insertionSort( proc insertionSort(
a: var seq[(float32, int16)], lo, hi: int hits: var seq[(float32, int16)], lo, hi: int
) {.inline.} = ) {.inline.} =
for i in lo + 1 .. hi: for i in lo + 1 .. hi:
var var
j = i - 1 j = i - 1
k = i k = i
while j >= 0 and a[j][0] > a[k][0]: while j >= 0 and hits[j][0] > hits[k][0]:
swap(a[j + 1], a[j]) swap(hits[j + 1], hits[j])
dec j dec j
dec k dec k
proc sort(a: var seq[(float32, int16)], inl, inr: int) = proc sort(hits: var seq[(float32, int16)], inl, inr: int) =
## Quicksort + insertion sort, in-place and faster than standard lib sort. ## Quicksort + insertion sort, in-place and faster than standard lib sort.
let n = inr - inl + 1 let n = inr - inl + 1
if n < 32: if n < 32:
insertionSort(a, inl, inr) insertionSort(hits, inl, inr)
return return
var var
l = inl l = inl
r = inr r = inr
let p = a[l + n div 2][0] let p = hits[l + n div 2][0]
while l <= r: while l <= r:
if a[l][0] < p: if hits[l][0] < p:
inc l inc l
elif a[r][0] > p: elif hits[r][0] > p:
dec r dec r
else: else:
swap(a[l], a[r]) swap(hits[l], hits[r])
inc l inc l
dec r dec r
sort(a, inl, r) sort(hits, inl, r)
sort(a, l, inr) sort(hits, l, inr)
proc shouldFill( proc shouldFill(
windingRule: WindingRule, count: int windingRule: WindingRule, count: int
@ -1175,26 +1175,23 @@ iterator walk(
prevAt: float32 prevAt: float32
while i < numHits: while i < numHits:
let (at, winding) = hits[i] let (at, winding) = hits[i]
if windingRule == wrNonZero and
count != 0 and
count + winding != 0 and
i < numHits - 1:
# Shortcut: if nonzero rule, we only care about when the count changes
# between zero and nonzero (or the last hit)
count += winding
inc i
continue
if at > 0: if at > 0:
if shouldFill(windingRule, count): if shouldFill(windingRule, count):
# Look ahead to see if the next hit is in the same spot as this hit.
# If it is, see if this and the next hit's windings cancel out.
# If they do, skip the hits and do not yield yet. It will be yielded
# later in a larger chunk.
if i < numHits - 1: if i < numHits - 1:
# Look ahead to see if the next hit is in the same spot as this hit.
# If it is, see if this hit and the next hit's windings cancel out.
# If they do, skip the hits. It will be yielded later in a
# larger chunk.
let (nextAt, nextWinding) = hits[i + 1] let (nextAt, nextWinding) = hits[i + 1]
if nextAt == at and winding + nextWinding == 0: if nextAt == at and winding + nextWinding == 0:
i += 2 i += 2
continue continue
# Shortcut: we only care about when we stop filling (or the last hit).
# If we continue filling, move to next hit.
if windingRule.shouldFill(count + winding):
count += winding
inc i
continue
yield (prevAt, at, count) yield (prevAt, at, count)
prevAt = at prevAt = at
count += winding count += winding