Merge pull request #446 from guzba/master
implement the last 2 mask:mask blends simd
This commit is contained in:
commit
460d97c520
3 changed files with 123 additions and 75 deletions
|
@ -11,8 +11,8 @@ when defined(amd64) and allowSimd:
|
||||||
type
|
type
|
||||||
Blender* = proc(backdrop, source: ColorRGBX): ColorRGBX {.gcsafe, raises: [].}
|
Blender* = proc(backdrop, source: ColorRGBX): ColorRGBX {.gcsafe, raises: [].}
|
||||||
## Function signature returned by blender.
|
## Function signature returned by blender.
|
||||||
Masker* = proc(backdrop, source: uint8): uint8 {.gcsafe, raises: [].}
|
MaskBlender* = proc(backdrop, source: uint8): uint8 {.gcsafe, raises: [].}
|
||||||
## Function signature returned by masker.
|
## Function signature returned by maskBlender.
|
||||||
|
|
||||||
when defined(release):
|
when defined(release):
|
||||||
{.push checks: off.}
|
{.push checks: off.}
|
||||||
|
@ -484,29 +484,29 @@ proc maskBlendExclude*(backdrop, source: uint8): uint8 {.inline.} =
|
||||||
## Exclude blend masks
|
## Exclude blend masks
|
||||||
max(backdrop, source) - min(backdrop, source)
|
max(backdrop, source) - min(backdrop, source)
|
||||||
|
|
||||||
proc maskBlendNormalMasker(backdrop, source: uint8): uint8 =
|
proc maskBlendNormalMaskBlender(backdrop, source: uint8): uint8 =
|
||||||
maskBlendNormal(backdrop, source)
|
maskBlendNormal(backdrop, source)
|
||||||
|
|
||||||
proc maskBlendMaskMasker(backdrop, source: uint8): uint8 =
|
proc maskBlendMaskMaskBlender(backdrop, source: uint8): uint8 =
|
||||||
maskBlendMask(backdrop, source)
|
maskBlendMask(backdrop, source)
|
||||||
|
|
||||||
proc maskBlendSubtractMasker(backdrop, source: uint8): uint8 =
|
proc maskBlendSubtractMaskBlender(backdrop, source: uint8): uint8 =
|
||||||
maskBlendSubtract(backdrop, source)
|
maskBlendSubtract(backdrop, source)
|
||||||
|
|
||||||
proc maskBlendExcludeMasker(backdrop, source: uint8): uint8 =
|
proc maskBlendExcludeMaskBlender(backdrop, source: uint8): uint8 =
|
||||||
maskBlendExclude(backdrop, source)
|
maskBlendExclude(backdrop, source)
|
||||||
|
|
||||||
proc maskBlendOverwriteMasker(backdrop, source: uint8): uint8 =
|
proc maskBlendOverwriteMaskBlender(backdrop, source: uint8): uint8 =
|
||||||
source
|
source
|
||||||
|
|
||||||
proc masker*(blendMode: BlendMode): Masker {.raises: [PixieError].} =
|
proc maskBlender*(blendMode: BlendMode): MaskBlender {.raises: [PixieError].} =
|
||||||
## Returns a blend masking function for a given blend masking mode.
|
## Returns a blend masking function for a given blend masking mode.
|
||||||
case blendMode:
|
case blendMode:
|
||||||
of NormalBlend: maskBlendNormalMasker
|
of NormalBlend: maskBlendNormalMaskBlender
|
||||||
of MaskBlend: maskBlendMaskMasker
|
of MaskBlend: maskBlendMaskMaskBlender
|
||||||
of OverwriteBlend: maskBlendOverwriteMasker
|
of OverwriteBlend: maskBlendOverwriteMaskBlender
|
||||||
of SubtractMaskBlend: maskBlendSubtractMasker
|
of SubtractMaskBlend: maskBlendSubtractMaskBlender
|
||||||
of ExcludeMaskBlend: maskBlendExcludeMasker
|
of ExcludeMaskBlend: maskBlendExcludeMaskBlender
|
||||||
else:
|
else:
|
||||||
raise newException(PixieError, "No masker for " & $blendMode)
|
raise newException(PixieError, "No masker for " & $blendMode)
|
||||||
|
|
||||||
|
@ -647,24 +647,63 @@ when defined(amd64) and allowSimd:
|
||||||
|
|
||||||
mm_or_si128(backdropEven, mm_slli_epi16(backdropOdd, 8))
|
mm_or_si128(backdropEven, mm_slli_epi16(backdropOdd, 8))
|
||||||
|
|
||||||
proc maskBlendNormalSimdMasker(backdrop, source: M128i): M128i =
|
proc maskBlendSubtractSimd*(backdrop, source: M128i): M128i {.inline.} =
|
||||||
|
let
|
||||||
|
oddMask = mm_set1_epi16(cast[int16](0xff00))
|
||||||
|
vec255 = mm_set1_epi8(255)
|
||||||
|
div255 = mm_set1_epi16(cast[int16](0x8081))
|
||||||
|
|
||||||
|
let sourceMinus255 = mm_sub_epi8(vec255, source)
|
||||||
|
|
||||||
|
var
|
||||||
|
multiplierEven = mm_slli_epi16(sourceMinus255, 8)
|
||||||
|
multiplierOdd = mm_and_si128(sourceMinus255, oddMask)
|
||||||
|
backdropEven = mm_slli_epi16(backdrop, 8)
|
||||||
|
backdropOdd = mm_and_si128(backdrop, oddMask)
|
||||||
|
|
||||||
|
backdropEven = mm_mulhi_epu16(backdropEven, multiplierEven)
|
||||||
|
backdropOdd = mm_mulhi_epu16(backdropOdd, multiplierOdd)
|
||||||
|
|
||||||
|
backdropEven = mm_srli_epi16(mm_mulhi_epu16(backdropEven, div255), 7)
|
||||||
|
backdropOdd = mm_srli_epi16(mm_mulhi_epu16(backdropOdd, div255), 7)
|
||||||
|
|
||||||
|
mm_or_si128(backdropEven, mm_slli_epi16(backdropOdd, 8))
|
||||||
|
|
||||||
|
proc maskBlendExcludeSimd*(backdrop, source: M128i): M128i {.inline.} =
|
||||||
|
mm_sub_epi8(mm_max_epu8(backdrop, source), mm_min_epu8(backdrop, source))
|
||||||
|
|
||||||
|
proc maskBlendNormalSimdMaskBlender(backdrop, source: M128i): M128i =
|
||||||
maskBlendNormalSimd(backdrop, source)
|
maskBlendNormalSimd(backdrop, source)
|
||||||
|
|
||||||
proc maskBlendMaskSimdMasker(backdrop, source: M128i): M128i =
|
proc maskBlendMaskSimdMaskBlender(backdrop, source: M128i): M128i =
|
||||||
maskBlendMaskSimd(backdrop, source)
|
maskBlendMaskSimd(backdrop, source)
|
||||||
|
|
||||||
proc maskerSimd*(blendMode: BlendMode): MaskerSimd {.raises: [PixieError].} =
|
proc maskBlendExcludeSimdMaskBlender(backdrop, source: M128i): M128i =
|
||||||
|
maskBlendExcludeSimd(backdrop, source)
|
||||||
|
|
||||||
|
proc maskBlendSubtractSimdMaskBlender(backdrop, source: M128i): M128i =
|
||||||
|
maskBlendSubtractSimd(backdrop, source)
|
||||||
|
|
||||||
|
proc maskBlenderSimd*(blendMode: BlendMode): MaskerSimd {.raises: [PixieError].} =
|
||||||
## Returns a blend masking function with SIMD support.
|
## Returns a blend masking function with SIMD support.
|
||||||
case blendMode:
|
case blendMode:
|
||||||
of NormalBlend: maskBlendNormalSimdMasker
|
of NormalBlend: maskBlendNormalSimdMaskBlender
|
||||||
of MaskBlend: maskBlendMaskSimdMasker
|
of MaskBlend: maskBlendMaskSimdMaskBlender
|
||||||
of OverwriteBlend: overwriteSimdBlender
|
of OverwriteBlend: overwriteSimdBlender
|
||||||
|
of SubtractMaskBlend: maskBlendSubtractSimdMaskBlender
|
||||||
|
of ExcludeMaskBlend: maskBlendExcludeSimdMaskBlender
|
||||||
else:
|
else:
|
||||||
raise newException(PixieError, "No SIMD masker for " & $blendMode)
|
raise newException(PixieError, "No SIMD masker for " & $blendMode)
|
||||||
|
|
||||||
proc hasSimdMasker*(blendMode: BlendMode): bool {.inline, raises: [].} =
|
proc hasSimdMaskBlender*(blendMode: BlendMode): bool {.inline, raises: [].} =
|
||||||
## Is there a blend masking function with SIMD support?
|
## Is there a blend masking function with SIMD support?
|
||||||
blendMode in {NormalBlend, MaskBlend, OverwriteBlend}
|
blendMode in {
|
||||||
|
NormalBlend,
|
||||||
|
MaskBlend,
|
||||||
|
OverwriteBlend,
|
||||||
|
SubtractMaskBlend,
|
||||||
|
ExcludeMaskBlend
|
||||||
|
}
|
||||||
|
|
||||||
when defined(release):
|
when defined(release):
|
||||||
{.pop.}
|
{.pop.}
|
||||||
|
|
|
@ -714,7 +714,7 @@ proc drawUber(
|
||||||
when type(a) is Image:
|
when type(a) is Image:
|
||||||
let blender = blendMode.blender()
|
let blender = blendMode.blender()
|
||||||
else: # a is a Mask
|
else: # a is a Mask
|
||||||
let masker = blendMode.masker()
|
let maskBlender = blendMode.maskBlender()
|
||||||
|
|
||||||
if blendMode == MaskBlend:
|
if blendMode == MaskBlend:
|
||||||
if yMin > 0:
|
if yMin > 0:
|
||||||
|
@ -777,7 +777,7 @@ proc drawUber(
|
||||||
let sample = b.getRgbaSmooth(srcPos.x, srcPos.y).a
|
let sample = b.getRgbaSmooth(srcPos.x, srcPos.y).a
|
||||||
else: # b is a Mask
|
else: # b is a Mask
|
||||||
let sample = b.getValueSmooth(srcPos.x, srcPos.y)
|
let sample = b.getValueSmooth(srcPos.x, srcPos.y)
|
||||||
a.unsafe[x, y] = masker(backdrop, sample)
|
a.unsafe[x, y] = maskBlender(backdrop, sample)
|
||||||
|
|
||||||
srcPos += dx
|
srcPos += dx
|
||||||
|
|
||||||
|
@ -972,8 +972,8 @@ proc drawUber(
|
||||||
x += 16
|
x += 16
|
||||||
sx += 16
|
sx += 16
|
||||||
else: # is a Mask
|
else: # is a Mask
|
||||||
if blendMode.hasSimdMasker():
|
if blendMode.hasSimdMaskBlender():
|
||||||
let maskerSimd = blendMode.maskerSimd()
|
let maskerSimd = blendMode.maskBlenderSimd()
|
||||||
for _ in 0 ..< (xStop - xStart) div 16:
|
for _ in 0 ..< (xStop - xStart) div 16:
|
||||||
let backdrop = mm_loadu_si128(a.data[a.dataIndex(x, y)].addr)
|
let backdrop = mm_loadu_si128(a.data[a.dataIndex(x, y)].addr)
|
||||||
when type(b) is Image:
|
when type(b) is Image:
|
||||||
|
@ -1089,7 +1089,7 @@ proc drawUber(
|
||||||
let sample = b.unsafe[samplePos.x, samplePos.y].a
|
let sample = b.unsafe[samplePos.x, samplePos.y].a
|
||||||
else: # b is a Mask
|
else: # b is a Mask
|
||||||
let sample = b.unsafe[samplePos.x, samplePos.y]
|
let sample = b.unsafe[samplePos.x, samplePos.y]
|
||||||
a.unsafe[x, y] = masker(backdrop, sample)
|
a.unsafe[x, y] = maskBlender(backdrop, sample)
|
||||||
srcPos += dx
|
srcPos += dx
|
||||||
|
|
||||||
if blendMode == MaskBlend:
|
if blendMode == MaskBlend:
|
||||||
|
|
|
@ -1223,6 +1223,20 @@ iterator walk(
|
||||||
if prevAt != width.float32.fixed32 and count != 0:
|
if prevAt != width.float32.fixed32 and count != 0:
|
||||||
echo "Leak detected: ", count, " @ (", prevAt, ", ", y, ")"
|
echo "Leak detected: ", count, " @ (", prevAt, ", ", y, ")"
|
||||||
|
|
||||||
|
iterator walkInteger(
|
||||||
|
hits: seq[(int32, int16)],
|
||||||
|
numHits: int,
|
||||||
|
windingRule: WindingRule,
|
||||||
|
y, width: int
|
||||||
|
): (int, int) =
|
||||||
|
for (prevAt, at, count) in hits.walk(numHits, windingRule, y, width):
|
||||||
|
let
|
||||||
|
fillStart = prevAt.integer
|
||||||
|
fillLen = at.integer - fillStart
|
||||||
|
if fillLen <= 0:
|
||||||
|
continue
|
||||||
|
yield (fillStart, fillLen)
|
||||||
|
|
||||||
proc computeCoverage(
|
proc computeCoverage(
|
||||||
coverages: ptr UncheckedArray[uint8],
|
coverages: ptr UncheckedArray[uint8],
|
||||||
hits: var seq[(Fixed32, int16)],
|
hits: var seq[(Fixed32, int16)],
|
||||||
|
@ -1443,9 +1457,9 @@ proc fillCoverage(
|
||||||
) =
|
) =
|
||||||
var x = startX
|
var x = startX
|
||||||
when defined(amd64) and allowSimd:
|
when defined(amd64) and allowSimd:
|
||||||
if blendMode.hasSimdMasker():
|
if blendMode.hasSimdMaskBlender():
|
||||||
let
|
let
|
||||||
maskerSimd = blendMode.maskerSimd()
|
maskerSimd = blendMode.maskBlenderSimd()
|
||||||
vecZero = mm_setzero_si128()
|
vecZero = mm_setzero_si128()
|
||||||
for _ in 0 ..< coverages.len div 16:
|
for _ in 0 ..< coverages.len div 16:
|
||||||
let
|
let
|
||||||
|
@ -1465,7 +1479,7 @@ proc fillCoverage(
|
||||||
mm_storeu_si128(mask.data[index].addr, vecZero)
|
mm_storeu_si128(mask.data[index].addr, vecZero)
|
||||||
x += 16
|
x += 16
|
||||||
|
|
||||||
let masker = blendMode.masker()
|
let maskBlender = blendMode.maskBlender()
|
||||||
for x in x ..< startX + coverages.len:
|
for x in x ..< startX + coverages.len:
|
||||||
let coverage = coverages[x - startX]
|
let coverage = coverages[x - startX]
|
||||||
if coverage != 0 or blendMode == ExcludeMaskBlend:
|
if coverage != 0 or blendMode == ExcludeMaskBlend:
|
||||||
|
@ -1473,7 +1487,7 @@ proc fillCoverage(
|
||||||
mask.unsafe[x, y] = coverage
|
mask.unsafe[x, y] = coverage
|
||||||
else:
|
else:
|
||||||
let backdrop = mask.unsafe[x, y]
|
let backdrop = mask.unsafe[x, y]
|
||||||
mask.unsafe[x, y] = masker(backdrop, coverage)
|
mask.unsafe[x, y] = maskBlender(backdrop, coverage)
|
||||||
elif blendMode == MaskBlend:
|
elif blendMode == MaskBlend:
|
||||||
mask.unsafe[x, y] = 0
|
mask.unsafe[x, y] = 0
|
||||||
|
|
||||||
|
@ -1481,22 +1495,6 @@ proc fillCoverage(
|
||||||
mask.clearUnsafe(0, y, startX, y)
|
mask.clearUnsafe(0, y, startX, y)
|
||||||
mask.clearUnsafe(startX + coverages.len, y, mask.width, y)
|
mask.clearUnsafe(startX + coverages.len, y, mask.width, y)
|
||||||
|
|
||||||
template walkHits(
|
|
||||||
hits: seq[(int32, int16)],
|
|
||||||
numHits: int,
|
|
||||||
windingRule: WindingRule,
|
|
||||||
y, width: int,
|
|
||||||
inner: untyped
|
|
||||||
) =
|
|
||||||
for (prevAt, at, count) in hits.walk(numHits, windingRule, y, width):
|
|
||||||
let
|
|
||||||
fillStart {.inject.} = prevAt.integer
|
|
||||||
fillLen {.inject.} = at.integer - fillStart
|
|
||||||
if fillLen <= 0:
|
|
||||||
continue
|
|
||||||
|
|
||||||
inner
|
|
||||||
|
|
||||||
proc fillHits(
|
proc fillHits(
|
||||||
image: Image,
|
image: Image,
|
||||||
rgbx: ColorRGBX,
|
rgbx: ColorRGBX,
|
||||||
|
@ -1506,38 +1504,36 @@ proc fillHits(
|
||||||
windingRule: WindingRule,
|
windingRule: WindingRule,
|
||||||
blendMode: BlendMode
|
blendMode: BlendMode
|
||||||
) =
|
) =
|
||||||
template simdBlob(image: Image, x: var int, blendProc: untyped) =
|
template simdBlob(image: Image, x: var int, len: int, blendProc: untyped) =
|
||||||
when allowSimd:
|
when allowSimd:
|
||||||
when defined(amd64):
|
when defined(amd64):
|
||||||
let colorVec = mm_set1_epi32(cast[int32](rgbx))
|
let colorVec = mm_set1_epi32(cast[int32](rgbx))
|
||||||
for _ in 0 ..< fillLen div 4:
|
for _ in 0 ..< len div 4:
|
||||||
let
|
let backdrop = mm_loadu_si128(image.unsafe[x, y].addr)
|
||||||
index = image.dataIndex(x, y)
|
mm_storeu_si128(image.unsafe[x, y].addr, blendProc(backdrop, colorVec))
|
||||||
backdrop = mm_loadu_si128(image.data[index].addr)
|
|
||||||
mm_storeu_si128(image.data[index].addr, blendProc(backdrop, colorVec))
|
|
||||||
x += 4
|
x += 4
|
||||||
|
|
||||||
case blendMode:
|
case blendMode:
|
||||||
of OverwriteBlend:
|
of OverwriteBlend:
|
||||||
walkHits hits, numHits, windingRule, y, image.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
|
||||||
fillUnsafe(image.data, rgbx, image.dataIndex(fillStart, y), fillLen)
|
fillUnsafe(image.data, rgbx, image.dataIndex(start, y), len)
|
||||||
|
|
||||||
of NormalBlend:
|
of NormalBlend:
|
||||||
walkHits hits, numHits, windingRule, y, image.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
|
||||||
if rgbx.a == 255:
|
if rgbx.a == 255:
|
||||||
fillUnsafe(image.data, rgbx, image.dataIndex(fillStart, y), fillLen)
|
fillUnsafe(image.data, rgbx, image.dataIndex(start, y), len)
|
||||||
else:
|
else:
|
||||||
var x = fillStart
|
var x = start
|
||||||
simdBlob(image, x, blendNormalSimd)
|
simdBlob(image, x, len, blendNormalSimd)
|
||||||
for x in x ..< fillStart + fillLen:
|
for x in x ..< start + len:
|
||||||
let backdrop = image.unsafe[x, y]
|
let backdrop = image.unsafe[x, y]
|
||||||
image.unsafe[x, y] = blendNormal(backdrop, rgbx)
|
image.unsafe[x, y] = blendNormal(backdrop, rgbx)
|
||||||
|
|
||||||
of MaskBlend:
|
of MaskBlend:
|
||||||
var filledTo = startX
|
var filledTo = startX
|
||||||
walkHits hits, numHits, windingRule, y, image.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
|
||||||
block: # Clear any gap between this fill and the previous fill
|
block: # Clear any gap between this fill and the previous fill
|
||||||
let gapBetween = fillStart - filledTo
|
let gapBetween = start - filledTo
|
||||||
if gapBetween > 0:
|
if gapBetween > 0:
|
||||||
fillUnsafe(
|
fillUnsafe(
|
||||||
image.data,
|
image.data,
|
||||||
|
@ -1545,12 +1541,12 @@ proc fillHits(
|
||||||
image.dataIndex(filledTo, y),
|
image.dataIndex(filledTo, y),
|
||||||
gapBetween
|
gapBetween
|
||||||
)
|
)
|
||||||
filledTo = fillStart + fillLen
|
filledTo = start + len
|
||||||
block: # Handle this fill
|
block: # Handle this fill
|
||||||
if rgbx.a != 255:
|
if rgbx.a != 255:
|
||||||
var x = fillStart
|
var x = start
|
||||||
simdBlob(image, x, blendMaskSimd)
|
simdBlob(image, x, len, blendMaskSimd)
|
||||||
for x in x ..< fillStart + fillLen:
|
for x in x ..< start + len:
|
||||||
let backdrop = image.unsafe[x, y]
|
let backdrop = image.unsafe[x, y]
|
||||||
image.unsafe[x, y] = blendMask(backdrop, rgbx)
|
image.unsafe[x, y] = blendMask(backdrop, rgbx)
|
||||||
|
|
||||||
|
@ -1559,8 +1555,8 @@ proc fillHits(
|
||||||
|
|
||||||
else:
|
else:
|
||||||
let blender = blendMode.blender()
|
let blender = blendMode.blender()
|
||||||
walkHits hits, numHits, windingRule, y, image.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
|
||||||
for x in fillStart ..< fillStart + fillLen:
|
for x in start ..< start + len:
|
||||||
let backdrop = image.unsafe[x, y]
|
let backdrop = image.unsafe[x, y]
|
||||||
image.unsafe[x, y] = blender(backdrop, rgbx)
|
image.unsafe[x, y] = blender(backdrop, rgbx)
|
||||||
|
|
||||||
|
@ -1572,31 +1568,44 @@ proc fillHits(
|
||||||
windingRule: WindingRule,
|
windingRule: WindingRule,
|
||||||
blendMode: BlendMode
|
blendMode: BlendMode
|
||||||
) =
|
) =
|
||||||
|
template simdBlob(mask: Mask, x: var int, len: int, blendProc: untyped) =
|
||||||
|
when allowSimd:
|
||||||
|
when defined(amd64):
|
||||||
|
let vec255 = mm_set1_epi8(255)
|
||||||
|
for _ in 0 ..< len div 16:
|
||||||
|
let backdrop = mm_loadu_si128(mask.unsafe[x, y].addr)
|
||||||
|
mm_storeu_si128(mask.unsafe[x, y].addr, blendProc(backdrop, vec255))
|
||||||
|
x += 16
|
||||||
|
|
||||||
case blendMode:
|
case blendMode:
|
||||||
of NormalBlend, OverwriteBlend:
|
of NormalBlend, OverwriteBlend:
|
||||||
walkHits hits, numHits, windingRule, y, mask.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
|
||||||
fillUnsafe(mask.data, 255, mask.dataIndex(fillStart, y), fillLen)
|
fillUnsafe(mask.data, 255, mask.dataIndex(start, y), len)
|
||||||
|
|
||||||
of MaskBlend:
|
of MaskBlend:
|
||||||
var filledTo = startX
|
var filledTo = startX
|
||||||
walkHits hits, numHits, windingRule,y, mask.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
|
||||||
let gapBetween = fillStart - filledTo
|
let gapBetween = start - filledTo
|
||||||
if gapBetween > 0:
|
if gapBetween > 0:
|
||||||
fillUnsafe(mask.data, 0, mask.dataIndex(filledTo, y), gapBetween)
|
fillUnsafe(mask.data, 0, mask.dataIndex(filledTo, y), gapBetween)
|
||||||
filledTo = fillStart + fillLen
|
filledTo = start + len
|
||||||
|
|
||||||
mask.clearUnsafe(0, y, startX, y)
|
mask.clearUnsafe(0, y, startX, y)
|
||||||
mask.clearUnsafe(filledTo, y, mask.width, y)
|
mask.clearUnsafe(filledTo, y, mask.width, y)
|
||||||
|
|
||||||
of SubtractMaskBlend:
|
of SubtractMaskBlend:
|
||||||
walkHits hits, numHits, windingRule, y, mask.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
|
||||||
for x in fillStart ..< fillStart + fillLen:
|
var x = start
|
||||||
|
simdBlob(mask, x, len, maskBlendSubtractSimd)
|
||||||
|
for x in x ..< start + len:
|
||||||
let backdrop = mask.unsafe[x, y]
|
let backdrop = mask.unsafe[x, y]
|
||||||
mask.unsafe[x, y] = maskBlendSubtract(backdrop, 255)
|
mask.unsafe[x, y] = maskBlendSubtract(backdrop, 255)
|
||||||
|
|
||||||
of ExcludeMaskBlend:
|
of ExcludeMaskBlend:
|
||||||
walkHits hits, numHits, windingRule, y, mask.width:
|
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
|
||||||
for x in fillStart ..< fillStart + fillLen:
|
var x = start
|
||||||
|
simdBlob(mask, x, len, maskBlendExcludeSimd)
|
||||||
|
for x in x ..< start + len:
|
||||||
let backdrop = mask.unsafe[x, y]
|
let backdrop = mask.unsafe[x, y]
|
||||||
mask.unsafe[x, y] = maskBlendExclude(backdrop, 255)
|
mask.unsafe[x, y] = maskBlendExclude(backdrop, 255)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue