From fcf072ff5a571606ffe63d9ed5c316ca523f7bac Mon Sep 17 00:00:00 2001 From: Ryan Oldenburg Date: Thu, 3 Dec 2020 19:58:44 -0600 Subject: [PATCH 1/5] inplace blur, faster fill --- src/pixie/images.nim | 54 ++++++++++++++++++++------------------ tests/benchmark_images.nim | 12 +++++++++ 2 files changed, 40 insertions(+), 26 deletions(-) diff --git a/src/pixie/images.nim b/src/pixie/images.nim index 53b46bc..e02b040 100644 --- a/src/pixie/images.nim +++ b/src/pixie/images.nim @@ -70,8 +70,11 @@ proc `[]=`*(image: Image, x, y: int, rgba: ColorRGBA) {.inline.} = proc fill*(image: Image, rgba: ColorRgba) = ## Fills the image with a solid color. - for c in image.data.mitems: - c = rgba + if rgba == rgba(0, 0, 0, 0): + zeroMem(image.data[0].addr, image.data.len * 4) + else: + for c in image.data.mitems: + c = rgba proc invert*(image: Image) = ## Inverts all of the colors and alpha. @@ -206,7 +209,7 @@ proc resize*(srcImage: Image, width, height: int): Image = proc blur*(image: Image, radius: float32): Image = ## Applies Gaussian blur to the image given a radius. - let radius = radius.int + let radius = round(radius).int if radius == 0: return image.copy() @@ -264,11 +267,11 @@ proc blur*(image: Image, radius: float32): Image = return blurY -proc blurAlpha*(image: Image, radius: float32): Image = +proc blurAlpha*(image: Image, radius: float32) = ## Applies Gaussian blur to the image given a radius. - let radius = radius.int + let radius = round(radius).int if radius == 0: - return image.copy() + return # Compute lookup table for 1d Gaussian kernel. var @@ -294,8 +297,7 @@ proc blurAlpha*(image: Image, radius: float32): Image = alpha += c2.a.float32 * a blurX.setRgbaUnsafe(x, y, rgba(0, 0, 0, alpha.uint8)) - # Blur in the Y direction. - var blurY = newImage(image.width, image.height) + # Blur in the Y direction and modify image. for y in 0 ..< image.height: for x in 0 ..< image.width: var alpha: float32 @@ -303,32 +305,32 @@ proc blurAlpha*(image: Image, radius: float32): Image = let c2 = blurX[x, y + yb] let a = lookup[yb + radius] alpha += c2.a.float32 * a - blurY.setRgbaUnsafe(x, y, rgba(0, 0, 0, alpha.uint8)) + image.setRgbaUnsafe(x, y, rgba(0, 0, 0, alpha.uint8)) - return blurY - -proc shift*(image: Image, offset: Vec2): Image = +proc shift*(image: Image, offset: Vec2) = ## Shifts the image by offset. - result = newImage(image.width, image.height) - result.draw(image, offset) + let copy = image.copy() # Copy to read from. + image.fill(rgba(0, 0, 0, 0)) # Reset this for being drawn to. + image.draw(copy, offset) # Draw copy into image. -proc spread*(image: Image, spread: float32): Image = +proc spread*(image: Image, spread: float32) = ## Grows the image as a mask by spread. - result = newImage(image.width, image.height) + let + copy = image.copy() + spread = round(spread).int assert spread > 0 - for y in 0 ..< result.height: - for x in 0 ..< result.width: + for y in 0 ..< image.height: + for x in 0 ..< image.width: var maxAlpha = 0.uint8 block blurBox: - for bx in -spread.int .. spread.int: - for by in -spread.int .. spread.int: - # if vec2(bx.float32, by.float32).length < spread: - let alpha = image[x + bx, y + by].a + for bx in -spread .. spread: + for by in -spread .. spread: + let alpha = copy[x + bx, y + by].a if alpha > maxAlpha: maxAlpha = alpha if maxAlpha == 255: break blurBox - result[x, y] = rgba(0, 0, 0, maxAlpha) + image[x, y] = rgba(0, 0, 0, maxAlpha) proc shadow*( mask: Image, @@ -339,11 +341,11 @@ proc shadow*( ## Create a shadow of the image with the offset, spread and blur. var shadow = mask if offset != vec2(0, 0): - shadow = shadow.shift(offset) + shadow.shift(offset) if spread > 0: - shadow = shadow.spread(spread) + shadow.spread(spread) if blur > 0: - shadow = shadow.blurAlpha(blur) + shadow.blurAlpha(blur) result = newImage(mask.width, mask.height) result.fill(color) result.draw(shadow, blendMode = bmMask) diff --git a/tests/benchmark_images.nim b/tests/benchmark_images.nim index 4d6c390..51c23c0 100644 --- a/tests/benchmark_images.nim +++ b/tests/benchmark_images.nim @@ -48,6 +48,18 @@ timeIt "fill": a.fill(rgba(255, 255, 255, 255)) doAssert a[0, 0] == rgba(255, 255, 255, 255) +timeIt "fillOriginal rgba(0, 0, 0, 0)": + var a = newImage(2560, 1440) + for i in 0 ..< iterations: + a.fillOriginal(rgba(0, 0, 0, 0)) + doAssert a[0, 0] == rgba(0, 0, 0, 0) + +timeIt "fill rgba(0, 0, 0, 0)": + var a = newImage(2560, 1440) + for i in 0 ..< iterations: + a.fill(rgba(0, 0, 0, 0)) + doAssert a[0, 0] == rgba(0, 0, 0, 0) + timeIt "invertOriginal": var a = newImage(2560, 1440) for i in 0 ..< iterations: From 9633f29a6b220c6f872eb6e9fd46102bcb75e95f Mon Sep 17 00:00:00 2001 From: Ryan Oldenburg Date: Thu, 3 Dec 2020 20:18:15 -0600 Subject: [PATCH 2/5] simpler SetSat --- src/pixie/blends.nim | 38 +++----------------------------------- 1 file changed, 3 insertions(+), 35 deletions(-) diff --git a/src/pixie/blends.nim b/src/pixie/blends.nim index 8c6c6de..15a296b 100644 --- a/src/pixie/blends.nim +++ b/src/pixie/blends.nim @@ -312,41 +312,9 @@ proc Sat(C: Color): float32 {.inline.} = max([C.r, C.g, C.b]) - min([C.r, C.g, C.b]) proc SetSat(C: Color, s: float32): Color {.inline.} = - var arr = [(C.r, 0), (C.g, 1), (C.b, 2)] - # TODO: Don't rely on sort. - arr.sort() - var - Cmin = arr[0][0] - Cmid = arr[1][0] - Cmax = arr[2][0] - if Cmax > Cmin: - Cmid = (((Cmid - Cmin) * s) / (Cmax - Cmin)) - Cmax = s - else: - Cmid = 0 - Cmax = 0 - Cmin = 0 - - if arr[0][1] == 0: - result.r = Cmin - if arr[1][1] == 0: - result.r = Cmid - if arr[2][1] == 0: - result.r = Cmax - - if arr[0][1] == 1: - result.g = Cmin - if arr[1][1] == 1: - result.g = Cmid - if arr[2][1] == 1: - result.g = Cmax - - if arr[0][1] == 2: - result.b = Cmin - if arr[1][1] == 2: - result.b = Cmid - if arr[2][1] == 2: - result.b = Cmax + let satC = Sat(C) + if satC > 0: + result = (C - min([C.r, C.g, C.b])) * s / satC proc alphaFix(Cb, Cs, mixed: Color): Color {.inline.} = let ab = Cb.a From bed384db6a56bcf28773098fffe91282d13bf632 Mon Sep 17 00:00:00 2001 From: Ryan Oldenburg Date: Thu, 3 Dec 2020 20:21:15 -0600 Subject: [PATCH 3/5] add links to blend formula sources --- src/pixie/blends.nim | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/pixie/blends.nim b/src/pixie/blends.nim index 15a296b..3ac932f 100644 --- a/src/pixie/blends.nim +++ b/src/pixie/blends.nim @@ -1,6 +1,9 @@ ## Blending modes. import chroma, math, algorithm +# See https://www.w3.org/TR/compositing-1/ +# See https://www.khronos.org/registry/OpenGL/extensions/KHR/KHR_blend_equation_advanced.txt + type BlendMode* = enum bmNormal bmDarken From fce9f12147b4d5c0f9a28d292b882cdddd27c00a Mon Sep 17 00:00:00 2001 From: Ryan Oldenburg Date: Thu, 3 Dec 2020 21:45:25 -0600 Subject: [PATCH 4/5] add links to blend formula sources --- src/pixie/images.nim | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/src/pixie/images.nim b/src/pixie/images.nim index e02b040..7c1c3d7 100644 --- a/src/pixie/images.nim +++ b/src/pixie/images.nim @@ -1,4 +1,4 @@ -import chroma, blends, vmath, common +import chroma, blends, vmath, common, system/memory type Image* = ref object @@ -70,11 +70,9 @@ proc `[]=`*(image: Image, x, y: int, rgba: ColorRGBA) {.inline.} = proc fill*(image: Image, rgba: ColorRgba) = ## Fills the image with a solid color. - if rgba == rgba(0, 0, 0, 0): - zeroMem(image.data[0].addr, image.data.len * 4) - else: - for c in image.data.mitems: - c = rgba + nimSetMem(image.data[0].addr, cast[int32](rgba), image.data.len) + # for c in image.data.mitems: + # c = rgba proc invert*(image: Image) = ## Inverts all of the colors and alpha. From 9fbd37bb53f3417166e10c2a1066d624f0ca9c45 Mon Sep 17 00:00:00 2001 From: Ryan Oldenburg Date: Thu, 3 Dec 2020 21:49:11 -0600 Subject: [PATCH 5/5] use memset --- src/pixie/images.nim | 8 +++++--- tests/benchmark_images.nim | 12 ------------ 2 files changed, 5 insertions(+), 15 deletions(-) diff --git a/src/pixie/images.nim b/src/pixie/images.nim index 7c1c3d7..9607323 100644 --- a/src/pixie/images.nim +++ b/src/pixie/images.nim @@ -70,9 +70,11 @@ proc `[]=`*(image: Image, x, y: int, rgba: ColorRGBA) {.inline.} = proc fill*(image: Image, rgba: ColorRgba) = ## Fills the image with a solid color. - nimSetMem(image.data[0].addr, cast[int32](rgba), image.data.len) - # for c in image.data.mitems: - # c = rgba + if rgba.r == rgba.g and rgba.r == rgba.b and rgba.r == rgba.a: + nimSetMem(image.data[0].addr, rgba.r.cint, image.data.len * 4) + else: + for c in image.data.mitems: + c = rgba proc invert*(image: Image) = ## Inverts all of the colors and alpha. diff --git a/tests/benchmark_images.nim b/tests/benchmark_images.nim index 51c23c0..4d6c390 100644 --- a/tests/benchmark_images.nim +++ b/tests/benchmark_images.nim @@ -48,18 +48,6 @@ timeIt "fill": a.fill(rgba(255, 255, 255, 255)) doAssert a[0, 0] == rgba(255, 255, 255, 255) -timeIt "fillOriginal rgba(0, 0, 0, 0)": - var a = newImage(2560, 1440) - for i in 0 ..< iterations: - a.fillOriginal(rgba(0, 0, 0, 0)) - doAssert a[0, 0] == rgba(0, 0, 0, 0) - -timeIt "fill rgba(0, 0, 0, 0)": - var a = newImage(2560, 1440) - for i in 0 ..< iterations: - a.fill(rgba(0, 0, 0, 0)) - doAssert a[0, 0] == rgba(0, 0, 0, 0) - timeIt "invertOriginal": var a = newImage(2560, 1440) for i in 0 ..< iterations: