better
This commit is contained in:
parent
56a9981f91
commit
27baf16e9b
1 changed files with 21 additions and 24 deletions
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue