summaryrefslogtreecommitdiffstats
path: root/thirdparty/thorvg/src/renderer
diff options
context:
space:
mode:
Diffstat (limited to 'thirdparty/thorvg/src/renderer')
-rw-r--r--thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h164
-rw-r--r--thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h142
-rw-r--r--thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h7
-rw-r--r--thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp21
-rw-r--r--thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp4
-rw-r--r--thirdparty/thorvg/src/renderer/tvgCanvas.h22
-rw-r--r--thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp11
-rw-r--r--thirdparty/thorvg/src/renderer/tvgLoader.cpp24
-rw-r--r--thirdparty/thorvg/src/renderer/tvgLoader.h1
-rw-r--r--thirdparty/thorvg/src/renderer/tvgShape.cpp15
-rw-r--r--thirdparty/thorvg/src/renderer/tvgShape.h10
-rw-r--r--thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp14
-rw-r--r--thirdparty/thorvg/src/renderer/tvgText.cpp15
-rw-r--r--thirdparty/thorvg/src/renderer/tvgText.h2
-rw-r--r--thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp16
15 files changed, 285 insertions, 183 deletions
diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h
index cbaec28fa3..a072a88819 100644
--- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h
+++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h
@@ -101,47 +101,57 @@ static void avxRasterPixel32(uint32_t *dst, uint32_t val, uint32_t offset, int32
static bool avxRasterTranslucentRect(SwSurface* surface, const SwBBox& region, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
- if (surface->channelSize != sizeof(uint32_t)) {
- TVGERR("SW_ENGINE", "Unsupported Channel Size = %d", surface->channelSize);
- return false;
- }
-
- auto color = surface->join(r, g, b, a);
- auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x;
auto h = static_cast<uint32_t>(region.max.y - region.min.y);
auto w = static_cast<uint32_t>(region.max.x - region.min.x);
- uint32_t ialpha = 255 - a;
+ //32bits channels
+ if (surface->channelSize == sizeof(uint32_t)) {
+ auto color = surface->join(r, g, b, a);
+ auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x;
- auto avxColor = _mm_set1_epi32(color);
- auto avxIalpha = _mm_set1_epi8(ialpha);
+ uint32_t ialpha = 255 - a;
- for (uint32_t y = 0; y < h; ++y) {
- auto dst = &buffer[y * surface->stride];
+ auto avxColor = _mm_set1_epi32(color);
+ auto avxIalpha = _mm_set1_epi8(ialpha);
- //1. fill the not aligned memory (for 128-bit registers a 16-bytes alignment is required)
- auto notAligned = ((uintptr_t)dst & 0xf) / 4;
- if (notAligned) {
- notAligned = (N_32BITS_IN_128REG - notAligned > w ? w : N_32BITS_IN_128REG - notAligned);
- for (uint32_t x = 0; x < notAligned; ++x, ++dst) {
- *dst = color + ALPHA_BLEND(*dst, ialpha);
+ for (uint32_t y = 0; y < h; ++y) {
+ auto dst = &buffer[y * surface->stride];
+
+ //1. fill the not aligned memory (for 128-bit registers a 16-bytes alignment is required)
+ auto notAligned = ((uintptr_t)dst & 0xf) / 4;
+ if (notAligned) {
+ notAligned = (N_32BITS_IN_128REG - notAligned > w ? w : N_32BITS_IN_128REG - notAligned);
+ for (uint32_t x = 0; x < notAligned; ++x, ++dst) {
+ *dst = color + ALPHA_BLEND(*dst, ialpha);
+ }
}
- }
- //2. fill the aligned memory - N_32BITS_IN_128REG pixels processed at once
- uint32_t iterations = (w - notAligned) / N_32BITS_IN_128REG;
- uint32_t avxFilled = iterations * N_32BITS_IN_128REG;
- auto avxDst = (__m128i*)dst;
- for (uint32_t x = 0; x < iterations; ++x, ++avxDst) {
- *avxDst = _mm_add_epi32(avxColor, ALPHA_BLEND(*avxDst, avxIalpha));
- }
+ //2. fill the aligned memory - N_32BITS_IN_128REG pixels processed at once
+ uint32_t iterations = (w - notAligned) / N_32BITS_IN_128REG;
+ uint32_t avxFilled = iterations * N_32BITS_IN_128REG;
+ auto avxDst = (__m128i*)dst;
+ for (uint32_t x = 0; x < iterations; ++x, ++avxDst) {
+ *avxDst = _mm_add_epi32(avxColor, ALPHA_BLEND(*avxDst, avxIalpha));
+ }
- //3. fill the remaining pixels
- int32_t leftovers = w - notAligned - avxFilled;
- dst += avxFilled;
- while (leftovers--) {
- *dst = color + ALPHA_BLEND(*dst, ialpha);
- dst++;
+ //3. fill the remaining pixels
+ int32_t leftovers = w - notAligned - avxFilled;
+ dst += avxFilled;
+ while (leftovers--) {
+ *dst = color + ALPHA_BLEND(*dst, ialpha);
+ dst++;
+ }
+ }
+ //8bit grayscale
+ } else if (surface->channelSize == sizeof(uint8_t)) {
+ TVGLOG("SW_ENGINE", "Require AVX Optimization, Channel Size = %d", surface->channelSize);
+ auto buffer = surface->buf8 + (region.min.y * surface->stride) + region.min.x;
+ auto ialpha = ~a;
+ for (uint32_t y = 0; y < h; ++y) {
+ auto dst = &buffer[y * surface->stride];
+ for (uint32_t x = 0; x < w; ++x, ++dst) {
+ *dst = a + MULTIPLY(*dst, ialpha);
+ }
}
}
return true;
@@ -150,56 +160,68 @@ static bool avxRasterTranslucentRect(SwSurface* surface, const SwBBox& region, u
static bool avxRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
- if (surface->channelSize != sizeof(uint32_t)) {
- TVGERR("SW_ENGINE", "Unsupported Channel Size = %d", surface->channelSize);
- return false;
- }
-
- auto color = surface->join(r, g, b, a);
auto span = rle->spans;
- uint32_t src;
- for (uint32_t i = 0; i < rle->size; ++i) {
- auto dst = &surface->buf32[span->y * surface->stride + span->x];
+ //32bit channels
+ if (surface->channelSize == sizeof(uint32_t)) {
+ auto color = surface->join(r, g, b, a);
+ uint32_t src;
- if (span->coverage < 255) src = ALPHA_BLEND(color, span->coverage);
- else src = color;
+ for (uint32_t i = 0; i < rle->size; ++i) {
+ auto dst = &surface->buf32[span->y * surface->stride + span->x];
- auto ialpha = IA(src);
+ if (span->coverage < 255) src = ALPHA_BLEND(color, span->coverage);
+ else src = color;
- //1. fill the not aligned memory (for 128-bit registers a 16-bytes alignment is required)
- auto notAligned = ((uintptr_t)dst & 0xf) / 4;
- if (notAligned) {
- notAligned = (N_32BITS_IN_128REG - notAligned > span->len ? span->len : N_32BITS_IN_128REG - notAligned);
- for (uint32_t x = 0; x < notAligned; ++x, ++dst) {
- *dst = src + ALPHA_BLEND(*dst, ialpha);
+ auto ialpha = IA(src);
+
+ //1. fill the not aligned memory (for 128-bit registers a 16-bytes alignment is required)
+ auto notAligned = ((uintptr_t)dst & 0xf) / 4;
+ if (notAligned) {
+ notAligned = (N_32BITS_IN_128REG - notAligned > span->len ? span->len : N_32BITS_IN_128REG - notAligned);
+ for (uint32_t x = 0; x < notAligned; ++x, ++dst) {
+ *dst = src + ALPHA_BLEND(*dst, ialpha);
+ }
}
- }
- //2. fill the aligned memory using avx - N_32BITS_IN_128REG pixels processed at once
- //In order to avoid unneccessary avx variables declarations a check is made whether there are any iterations at all
- uint32_t iterations = (span->len - notAligned) / N_32BITS_IN_128REG;
- uint32_t avxFilled = 0;
- if (iterations > 0) {
- auto avxSrc = _mm_set1_epi32(src);
- auto avxIalpha = _mm_set1_epi8(ialpha);
+ //2. fill the aligned memory using avx - N_32BITS_IN_128REG pixels processed at once
+ //In order to avoid unneccessary avx variables declarations a check is made whether there are any iterations at all
+ uint32_t iterations = (span->len - notAligned) / N_32BITS_IN_128REG;
+ uint32_t avxFilled = 0;
+ if (iterations > 0) {
+ auto avxSrc = _mm_set1_epi32(src);
+ auto avxIalpha = _mm_set1_epi8(ialpha);
+
+ avxFilled = iterations * N_32BITS_IN_128REG;
+ auto avxDst = (__m128i*)dst;
+ for (uint32_t x = 0; x < iterations; ++x, ++avxDst) {
+ *avxDst = _mm_add_epi32(avxSrc, ALPHA_BLEND(*avxDst, avxIalpha));
+ }
+ }
- avxFilled = iterations * N_32BITS_IN_128REG;
- auto avxDst = (__m128i*)dst;
- for (uint32_t x = 0; x < iterations; ++x, ++avxDst) {
- *avxDst = _mm_add_epi32(avxSrc, ALPHA_BLEND(*avxDst, avxIalpha));
+ //3. fill the remaining pixels
+ int32_t leftovers = span->len - notAligned - avxFilled;
+ dst += avxFilled;
+ while (leftovers--) {
+ *dst = src + ALPHA_BLEND(*dst, ialpha);
+ dst++;
}
- }
- //3. fill the remaining pixels
- int32_t leftovers = span->len - notAligned - avxFilled;
- dst += avxFilled;
- while (leftovers--) {
- *dst = src + ALPHA_BLEND(*dst, ialpha);
- dst++;
+ ++span;
+ }
+ //8bit grayscale
+ } else if (surface->channelSize == sizeof(uint8_t)) {
+ TVGLOG("SW_ENGINE", "Require AVX Optimization, Channel Size = %d", surface->channelSize);
+ uint8_t src;
+ for (uint32_t i = 0; i < rle->size; ++i, ++span) {
+ auto dst = &surface->buf8[span->y * surface->stride + span->x];
+ if (span->coverage < 255) src = MULTIPLY(span->coverage, a);
+ else src = a;
+ auto ialpha = ~a;
+ for (uint32_t x = 0; x < span->len; ++x, ++dst) {
+ *dst = src + MULTIPLY(*dst, ialpha);
+ }
}
-
- ++span;
}
return true;
}
diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h
index 1ea6cd96cf..91cf7743c1 100644
--- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h
+++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h
@@ -91,44 +91,56 @@ static void neonRasterPixel32(uint32_t *dst, uint32_t val, uint32_t offset, int3
static bool neonRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
- if (surface->channelSize != sizeof(uint32_t)) {
- TVGERR("SW_ENGINE", "Unsupported Channel Size = %d", surface->channelSize);
- return false;
- }
-
- auto color = surface->join(r, g, b, a);
auto span = rle->spans;
- uint32_t src;
- uint8x8_t *vDst = nullptr;
- uint16_t align;
-
- for (uint32_t i = 0; i < rle->size; ++i) {
- if (span->coverage < 255) src = ALPHA_BLEND(color, span->coverage);
- else src = color;
-
- auto dst = &surface->buf32[span->y * surface->stride + span->x];
- auto ialpha = IA(src);
-
- if ((((uintptr_t) dst) & 0x7) != 0) {
- //fill not aligned byte
- *dst = src + ALPHA_BLEND(*dst, ialpha);
- vDst = (uint8x8_t*)(dst + 1);
- align = 1;
- } else {
- vDst = (uint8x8_t*) dst;
- align = 0;
- }
- uint8x8_t vSrc = (uint8x8_t) vdup_n_u32(src);
- uint8x8_t vIalpha = vdup_n_u8((uint8_t) ialpha);
+ //32bit channels
+ if (surface->channelSize == sizeof(uint32_t)) {
+ auto color = surface->join(r, g, b, a);
+ uint32_t src;
+ uint8x8_t *vDst = nullptr;
+ uint16_t align;
+
+ for (uint32_t i = 0; i < rle->size; ++i) {
+ if (span->coverage < 255) src = ALPHA_BLEND(color, span->coverage);
+ else src = color;
- for (uint32_t x = 0; x < (span->len - align) / 2; ++x)
- vDst[x] = vadd_u8(vSrc, ALPHA_BLEND(vDst[x], vIalpha));
+ auto dst = &surface->buf32[span->y * surface->stride + span->x];
+ auto ialpha = IA(src);
- auto leftovers = (span->len - align) % 2;
- if (leftovers > 0) dst[span->len - 1] = src + ALPHA_BLEND(dst[span->len - 1], ialpha);
+ if ((((uintptr_t) dst) & 0x7) != 0) {
+ //fill not aligned byte
+ *dst = src + ALPHA_BLEND(*dst, ialpha);
+ vDst = (uint8x8_t*)(dst + 1);
+ align = 1;
+ } else {
+ vDst = (uint8x8_t*) dst;
+ align = 0;
+ }
- ++span;
+ uint8x8_t vSrc = (uint8x8_t) vdup_n_u32(src);
+ uint8x8_t vIalpha = vdup_n_u8((uint8_t) ialpha);
+
+ for (uint32_t x = 0; x < (span->len - align) / 2; ++x)
+ vDst[x] = vadd_u8(vSrc, ALPHA_BLEND(vDst[x], vIalpha));
+
+ auto leftovers = (span->len - align) % 2;
+ if (leftovers > 0) dst[span->len - 1] = src + ALPHA_BLEND(dst[span->len - 1], ialpha);
+
+ ++span;
+ }
+ //8bit grayscale
+ } else if (surface->channelSize == sizeof(uint8_t)) {
+ TVGLOG("SW_ENGINE", "Require Neon Optimization, Channel Size = %d", surface->channelSize);
+ uint8_t src;
+ for (uint32_t i = 0; i < rle->size; ++i, ++span) {
+ auto dst = &surface->buf8[span->y * surface->stride + span->x];
+ if (span->coverage < 255) src = MULTIPLY(span->coverage, a);
+ else src = a;
+ auto ialpha = ~a;
+ for (uint32_t x = 0; x < span->len; ++x, ++dst) {
+ *dst = src + MULTIPLY(*dst, ialpha);
+ }
+ }
}
return true;
}
@@ -136,41 +148,51 @@ static bool neonRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, u
static bool neonRasterTranslucentRect(SwSurface* surface, const SwBBox& region, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
- if (surface->channelSize != sizeof(uint32_t)) {
- TVGERR("SW_ENGINE", "Unsupported Channel Size = %d", surface->channelSize);
- return false;
- }
-
- auto color = surface->join(r, g, b, a);
- auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x;
auto h = static_cast<uint32_t>(region.max.y - region.min.y);
auto w = static_cast<uint32_t>(region.max.x - region.min.x);
- auto ialpha = 255 - a;
- auto vColor = vdup_n_u32(color);
- auto vIalpha = vdup_n_u8((uint8_t) ialpha);
+ //32bits channels
+ if (surface->channelSize == sizeof(uint32_t)) {
+ auto color = surface->join(r, g, b, a);
+ auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x;
+ auto ialpha = 255 - a;
- uint8x8_t* vDst = nullptr;
- uint32_t align;
+ auto vColor = vdup_n_u32(color);
+ auto vIalpha = vdup_n_u8((uint8_t) ialpha);
- for (uint32_t y = 0; y < h; ++y) {
- auto dst = &buffer[y * surface->stride];
+ uint8x8_t* vDst = nullptr;
+ uint32_t align;
- if ((((uintptr_t) dst) & 0x7) != 0) {
- //fill not aligned byte
- *dst = color + ALPHA_BLEND(*dst, ialpha);
- vDst = (uint8x8_t*) (dst + 1);
- align = 1;
- } else {
- vDst = (uint8x8_t*) dst;
- align = 0;
- }
+ for (uint32_t y = 0; y < h; ++y) {
+ auto dst = &buffer[y * surface->stride];
- for (uint32_t x = 0; x < (w - align) / 2; ++x)
- vDst[x] = vadd_u8((uint8x8_t)vColor, ALPHA_BLEND(vDst[x], vIalpha));
+ if ((((uintptr_t) dst) & 0x7) != 0) {
+ //fill not aligned byte
+ *dst = color + ALPHA_BLEND(*dst, ialpha);
+ vDst = (uint8x8_t*) (dst + 1);
+ align = 1;
+ } else {
+ vDst = (uint8x8_t*) dst;
+ align = 0;
+ }
- auto leftovers = (w - align) % 2;
- if (leftovers > 0) dst[w - 1] = color + ALPHA_BLEND(dst[w - 1], ialpha);
+ for (uint32_t x = 0; x < (w - align) / 2; ++x)
+ vDst[x] = vadd_u8((uint8x8_t)vColor, ALPHA_BLEND(vDst[x], vIalpha));
+
+ auto leftovers = (w - align) % 2;
+ if (leftovers > 0) dst[w - 1] = color + ALPHA_BLEND(dst[w - 1], ialpha);
+ }
+ //8bit grayscale
+ } else if (surface->channelSize == sizeof(uint8_t)) {
+ TVGLOG("SW_ENGINE", "Require Neon Optimization, Channel Size = %d", surface->channelSize);
+ auto buffer = surface->buf8 + (region.min.y * surface->stride) + region.min.x;
+ auto ialpha = ~a;
+ for (uint32_t y = 0; y < h; ++y) {
+ auto dst = &buffer[y * surface->stride];
+ for (uint32_t x = 0; x < w; ++x, ++dst) {
+ *dst = a + MULTIPLY(*dst, ialpha);
+ }
+ }
}
return true;
}
diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h
index bab534bba2..cfce7785c7 100644
--- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h
+++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h
@@ -824,7 +824,7 @@ static AASpans* _AASpans(float ymin, float ymax, const SwImage* image, const SwB
//Initialize X range
auto height = yEnd - yStart;
- aaSpans->lines = static_cast<AALine*>(calloc(height, sizeof(AALine)));
+ aaSpans->lines = static_cast<AALine*>(malloc(height * sizeof(AALine)));
for (int32_t i = 0; i < height; i++) {
aaSpans->lines[i].x[0] = INT32_MAX;
@@ -878,7 +878,7 @@ static void _calcHorizCoverage(AALine *lines, int32_t eidx, int32_t y, int32_t x
/*
* This Anti-Aliasing mechanism is originated from Hermet Park's idea.
* To understand this AA logic, you can refer this page:
- * www.hermet.pe.kr/122 (hermetpark@gmail.com)
+ * https://uigraphics.tistory.com/1
*/
static void _calcAAEdge(AASpans *aaSpans, int32_t eidx)
{
@@ -924,6 +924,9 @@ static void _calcAAEdge(AASpans *aaSpans, int32_t eidx)
//Calculates AA Edges
for (y++; y < yEnd; y++) {
+
+ if (lines[y].x[0] == INT32_MAX) continue;
+
//Ready tx
if (eidx == 0) {
tx[0] = pEdge.x;
diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp
index 78108af095..0a3f5ef7e7 100644
--- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp
+++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp
@@ -147,7 +147,7 @@ struct SwShapeTask : SwTask
}
}
//Fill
- if (flags & (RenderUpdateFlag::Gradient | RenderUpdateFlag::Transform | RenderUpdateFlag::Color)) {
+ if (flags & (RenderUpdateFlag::Path |RenderUpdateFlag::Gradient | RenderUpdateFlag::Transform | RenderUpdateFlag::Color)) {
if (visibleFill || clipper) {
if (!shapeGenRle(&shape, rshape, antialiasing(strokeWidth))) goto err;
}
@@ -160,7 +160,7 @@ struct SwShapeTask : SwTask
}
}
//Stroke
- if (flags & (RenderUpdateFlag::Stroke | RenderUpdateFlag::Transform)) {
+ if (flags & (RenderUpdateFlag::Path | RenderUpdateFlag::Stroke | RenderUpdateFlag::Transform)) {
if (strokeWidth > 0.0f) {
shapeResetStroke(&shape, rshape, transform);
if (!shapeGenStrokeRle(&shape, rshape, transform, clipRegion, bbox, mpool, tid)) goto err;
@@ -718,9 +718,6 @@ void* SwRenderer::prepareCommon(SwTask* task, const RenderTransform* transform,
if (!surface) return task;
if (flags == RenderUpdateFlag::None) return task;
- //Finish previous task if it has duplicated request.
- task->done();
-
//TODO: Failed threading them. It would be better if it's possible.
//See: https://github.com/thorvg/thorvg/issues/1409
//Guarantee composition targets get ready.
@@ -769,8 +766,11 @@ RenderData SwRenderer::prepare(Surface* surface, const RenderMesh* mesh, RenderD
//prepare task
auto task = static_cast<SwImageTask*>(data);
if (!task) task = new SwImageTask;
+ else task->done();
+
task->source = surface;
task->mesh = mesh;
+
return prepareCommon(task, transform, clips, opacity, flags);
}
@@ -780,6 +780,8 @@ RenderData SwRenderer::prepare(const Array<RenderData>& scene, RenderData data,
//prepare task
auto task = static_cast<SwSceneTask*>(data);
if (!task) task = new SwSceneTask;
+ else task->done();
+
task->scene = scene;
//TODO: Failed threading them. It would be better if it's possible.
@@ -788,6 +790,7 @@ RenderData SwRenderer::prepare(const Array<RenderData>& scene, RenderData data,
for (auto task = scene.begin(); task < scene.end(); ++task) {
static_cast<SwTask*>(*task)->done();
}
+
return prepareCommon(task, transform, clips, opacity, flags);
}
@@ -796,10 +799,10 @@ RenderData SwRenderer::prepare(const RenderShape& rshape, RenderData data, const
{
//prepare task
auto task = static_cast<SwShapeTask*>(data);
- if (!task) {
- task = new SwShapeTask;
- task->rshape = &rshape;
- }
+ if (!task) task = new SwShapeTask;
+ else task->done();
+
+ task->rshape = &rshape;
task->clipper = clipper;
return prepareCommon(task, transform, clips, opacity, flags);
diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp
index 9ec4bd78a5..18f5f3eca8 100644
--- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp
+++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp
@@ -238,7 +238,7 @@ static void _outside(SwStroke& stroke, int32_t side, SwFixed lineLength)
} else {
//this is a mitered (pointed) or beveled (truncated) corner
auto rotate = SIDE_TO_ROTATE(side);
- auto bevel = (stroke.join == StrokeJoin::Bevel) ? true : false;
+ auto bevel = stroke.join == StrokeJoin::Bevel;
SwFixed phi = 0;
SwFixed thcos = 0;
@@ -816,7 +816,7 @@ void strokeReset(SwStroke* stroke, const RenderShape* rshape, const Matrix* tran
stroke->width = HALF_STROKE(rshape->strokeWidth());
stroke->cap = rshape->strokeCap();
- stroke->miterlimit = static_cast<SwFixed>(rshape->strokeMiterlimit()) << 16;
+ stroke->miterlimit = static_cast<SwFixed>(rshape->strokeMiterlimit() * 65536.0f);
//Save line join: it can be temporarily changed when stroking curves...
stroke->joinSaved = stroke->join = rshape->strokeJoin();
diff --git a/thirdparty/thorvg/src/renderer/tvgCanvas.h b/thirdparty/thorvg/src/renderer/tvgCanvas.h
index a4b866eacb..9d216e2f30 100644
--- a/thirdparty/thorvg/src/renderer/tvgCanvas.h
+++ b/thirdparty/thorvg/src/renderer/tvgCanvas.h
@@ -26,17 +26,15 @@
#include "tvgPaint.h"
+enum Status : uint8_t {Synced = 0, Updating, Drawing, Damanged};
+
struct Canvas::Impl
{
- enum Status : uint8_t {Synced = 0, Updating, Drawing};
-
list<Paint*> paints;
RenderMethod* renderer;
RenderRegion vport = {0, 0, INT32_MAX, INT32_MAX};
Status status = Status::Synced;
- bool refresh = false; //if all paints should be updated by force.
-
Impl(RenderMethod* pRenderer) : renderer(pRenderer)
{
renderer->ref();
@@ -87,18 +85,13 @@ struct Canvas::Impl
return Result::Success;
}
- void needRefresh()
- {
- refresh = true;
- }
-
Result update(Paint* paint, bool force)
{
if (paints.empty() || status == Status::Drawing) return Result::InsufficientCondition;
Array<RenderData> clips;
auto flag = RenderUpdateFlag::None;
- if (refresh || force) flag = RenderUpdateFlag::All;
+ if (status == Status::Damanged || force) flag = RenderUpdateFlag::All;
if (paint) {
paint->pImpl->update(renderer, nullptr, clips, 255, flag);
@@ -106,7 +99,6 @@ struct Canvas::Impl
for (auto paint : paints) {
paint->pImpl->update(renderer, nullptr, clips, 255, flag);
}
- refresh = false;
}
status = Status::Updating;
return Result::Success;
@@ -114,6 +106,7 @@ struct Canvas::Impl
Result draw()
{
+ if (status == Status::Damanged) update(nullptr, false);
if (status == Status::Drawing || paints.empty() || !renderer->preRender()) return Result::InsufficientCondition;
bool rendered = false;
@@ -129,7 +122,7 @@ struct Canvas::Impl
Result sync()
{
- if (status == Status::Synced) return Result::InsufficientCondition;
+ if (status == Status::Synced || status == Status::Damanged) return Result::InsufficientCondition;
if (renderer->sync()) {
status = Status::Synced;
@@ -141,7 +134,8 @@ struct Canvas::Impl
Result viewport(int32_t x, int32_t y, int32_t w, int32_t h)
{
- if (status != Status::Synced) return Result::InsufficientCondition;
+ if (status != Status::Damanged && status != Status::Synced) return Result::InsufficientCondition;
+
RenderRegion val = {x, y, w, h};
//intersect if the target buffer is already set.
auto surface = renderer->mainSurface();
@@ -151,7 +145,7 @@ struct Canvas::Impl
if (vport == val) return Result::Success;
renderer->viewport(val);
vport = val;
- needRefresh();
+ status = Status::Damanged;
return Result::Success;
}
};
diff --git a/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp b/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp
index 211dbb589c..82666b7ae3 100644
--- a/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp
+++ b/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp
@@ -45,15 +45,14 @@ struct GlCanvas::Impl
/************************************************************************/
#ifdef THORVG_GL_RASTER_SUPPORT
-GlCanvas::GlCanvas() : Canvas(GlRenderer::gen()), pImpl(new Impl)
+GlCanvas::GlCanvas() : Canvas(GlRenderer::gen()), pImpl(nullptr)
#else
-GlCanvas::GlCanvas() : Canvas(nullptr), pImpl(new Impl)
+GlCanvas::GlCanvas() : Canvas(nullptr), pImpl(nullptr)
#endif
{
}
-
GlCanvas::~GlCanvas()
{
delete(pImpl);
@@ -63,6 +62,10 @@ GlCanvas::~GlCanvas()
Result GlCanvas::target(int32_t id, uint32_t w, uint32_t h) noexcept
{
#ifdef THORVG_GL_RASTER_SUPPORT
+ if (Canvas::pImpl->status != Status::Damanged && Canvas::pImpl->status != Status::Synced) {
+ return Result::InsufficientCondition;
+ }
+
//We know renderer type, avoid dynamic_cast for performance.
auto renderer = static_cast<GlRenderer*>(Canvas::pImpl->renderer);
if (!renderer) return Result::MemoryCorruption;
@@ -72,7 +75,7 @@ Result GlCanvas::target(int32_t id, uint32_t w, uint32_t h) noexcept
renderer->viewport(Canvas::pImpl->vport);
//Paints must be updated again with this new target.
- Canvas::pImpl->needRefresh();
+ Canvas::pImpl->status = Status::Damanged;
return Result::Success;
#endif
diff --git a/thirdparty/thorvg/src/renderer/tvgLoader.cpp b/thirdparty/thorvg/src/renderer/tvgLoader.cpp
index 4b8d3256a9..6a81ddcdbb 100644
--- a/thirdparty/thorvg/src/renderer/tvgLoader.cpp
+++ b/thirdparty/thorvg/src/renderer/tvgLoader.cpp
@@ -433,3 +433,27 @@ LoadModule* LoaderMgr::loader(const uint32_t *data, uint32_t w, uint32_t h, bool
delete(loader);
return nullptr;
}
+
+
+//loads fonts from memory - loader is cached (regardless of copy value) in order to access it while setting font
+LoadModule* LoaderMgr::loader(const char* name, const char* data, uint32_t size, TVG_UNUSED const string& mimeType, bool copy)
+{
+#ifdef THORVG_TTF_LOADER_SUPPORT
+ //TODO: add check for mimetype ?
+ if (auto loader = _findFromCache(name)) return loader;
+
+ //function is dedicated for ttf loader (the only supported font loader)
+ auto loader = new TtfLoader;
+ if (loader->open(data, size, copy)) {
+ loader->hashpath = strdup(name);
+ loader->pathcache = true;
+ ScopedLock lock(key);
+ _activeLoaders.back(loader);
+ return loader;
+ }
+
+ TVGLOG("LOADER", "The font data \"%s\" could not be loaded.", name);
+ delete(loader);
+#endif
+ return nullptr;
+}
diff --git a/thirdparty/thorvg/src/renderer/tvgLoader.h b/thirdparty/thorvg/src/renderer/tvgLoader.h
index b15032df27..74c4f43964 100644
--- a/thirdparty/thorvg/src/renderer/tvgLoader.h
+++ b/thirdparty/thorvg/src/renderer/tvgLoader.h
@@ -32,6 +32,7 @@ struct LoaderMgr
static LoadModule* loader(const string& path, bool* invalid);
static LoadModule* loader(const char* data, uint32_t size, const string& mimeType, bool copy);
static LoadModule* loader(const uint32_t* data, uint32_t w, uint32_t h, bool copy);
+ static LoadModule* loader(const char* name, const char* data, uint32_t size, const string& mimeType, bool copy);
static LoadModule* loader(const char* key);
static bool retrieve(const string& path);
static bool retrieve(LoadModule* loader);
diff --git a/thirdparty/thorvg/src/renderer/tvgShape.cpp b/thirdparty/thorvg/src/renderer/tvgShape.cpp
index c010aa7bbf..3b9293a00e 100644
--- a/thirdparty/thorvg/src/renderer/tvgShape.cpp
+++ b/thirdparty/thorvg/src/renderer/tvgShape.cpp
@@ -88,6 +88,8 @@ Result Shape::appendPath(const PathCommand *cmds, uint32_t cmdCnt, const Point*
pImpl->grow(cmdCnt, ptsCnt);
pImpl->append(cmds, cmdCnt, pts, ptsCnt);
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
@@ -104,6 +106,8 @@ Result Shape::lineTo(float x, float y) noexcept
{
pImpl->lineTo(x, y);
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
@@ -112,6 +116,8 @@ Result Shape::cubicTo(float cx1, float cy1, float cx2, float cy2, float x, float
{
pImpl->cubicTo(cx1, cy1, cx2, cy2, x, y);
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
@@ -120,6 +126,8 @@ Result Shape::close() noexcept
{
pImpl->close();
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
@@ -137,9 +145,12 @@ Result Shape::appendCircle(float cx, float cy, float rx, float ry) noexcept
pImpl->cubicTo(cx + rxKappa, cy - ry, cx + rx, cy - ryKappa, cx + rx, cy);
pImpl->close();
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
+
Result Shape::appendArc(float cx, float cy, float radius, float startAngle, float sweep, bool pie) noexcept
{
//just circle
@@ -196,6 +207,8 @@ Result Shape::appendArc(float cx, float cy, float radius, float startAngle, floa
if (pie) pImpl->close();
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
@@ -234,6 +247,8 @@ Result Shape::appendRect(float x, float y, float w, float h, float rx, float ry)
pImpl->close();
}
+ pImpl->flag |= RenderUpdateFlag::Path;
+
return Result::Success;
}
diff --git a/thirdparty/thorvg/src/renderer/tvgShape.h b/thirdparty/thorvg/src/renderer/tvgShape.h
index 4e85db37d0..c45995a64d 100644
--- a/thirdparty/thorvg/src/renderer/tvgShape.h
+++ b/thirdparty/thorvg/src/renderer/tvgShape.h
@@ -167,24 +167,18 @@ struct Shape::Impl
memcpy(rs.path.pts.end(), pts, sizeof(Point) * ptsCnt);
rs.path.cmds.count += cmdCnt;
rs.path.pts.count += ptsCnt;
-
- flag |= RenderUpdateFlag::Path;
}
void moveTo(float x, float y)
{
rs.path.cmds.push(PathCommand::MoveTo);
rs.path.pts.push({x, y});
-
- flag |= RenderUpdateFlag::Path;
}
void lineTo(float x, float y)
{
rs.path.cmds.push(PathCommand::LineTo);
rs.path.pts.push({x, y});
-
- flag |= RenderUpdateFlag::Path;
}
void cubicTo(float cx1, float cy1, float cx2, float cy2, float x, float y)
@@ -193,8 +187,6 @@ struct Shape::Impl
rs.path.pts.push({cx1, cy1});
rs.path.pts.push({cx2, cy2});
rs.path.pts.push({x, y});
-
- flag |= RenderUpdateFlag::Path;
}
void close()
@@ -203,8 +195,6 @@ struct Shape::Impl
if (rs.path.cmds.count > 0 && rs.path.cmds.last() == PathCommand::Close) return;
rs.path.cmds.push(PathCommand::Close);
-
- flag |= RenderUpdateFlag::Path;
}
void strokeWidth(float width)
diff --git a/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp b/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp
index 52d85d8320..d762492f22 100644
--- a/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp
+++ b/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp
@@ -46,9 +46,9 @@ struct SwCanvas::Impl
/************************************************************************/
#ifdef THORVG_SW_RASTER_SUPPORT
-SwCanvas::SwCanvas() : Canvas(SwRenderer::gen()), pImpl(new Impl)
+SwCanvas::SwCanvas() : Canvas(SwRenderer::gen()), pImpl(nullptr)
#else
-SwCanvas::SwCanvas() : Canvas(nullptr), pImpl(new Impl)
+SwCanvas::SwCanvas() : Canvas(nullptr), pImpl(nullptr)
#endif
{
}
@@ -82,6 +82,10 @@ Result SwCanvas::mempool(MempoolPolicy policy) noexcept
Result SwCanvas::target(uint32_t* buffer, uint32_t stride, uint32_t w, uint32_t h, Colorspace cs) noexcept
{
#ifdef THORVG_SW_RASTER_SUPPORT
+ if (Canvas::pImpl->status != Status::Damanged && Canvas::pImpl->status != Status::Synced) {
+ return Result::InsufficientCondition;
+ }
+
//We know renderer type, avoid dynamic_cast for performance.
auto renderer = static_cast<SwRenderer*>(Canvas::pImpl->renderer);
if (!renderer) return Result::MemoryCorruption;
@@ -90,12 +94,12 @@ Result SwCanvas::target(uint32_t* buffer, uint32_t stride, uint32_t w, uint32_t
Canvas::pImpl->vport = {0, 0, (int32_t)w, (int32_t)h};
renderer->viewport(Canvas::pImpl->vport);
- //Paints must be updated again with this new target.
- Canvas::pImpl->needRefresh();
-
//FIXME: The value must be associated with an individual canvas instance.
ImageLoader::cs = static_cast<ColorSpace>(cs);
+ //Paints must be updated again with this new target.
+ Canvas::pImpl->status = Status::Damanged;
+
return Result::Success;
#endif
return Result::NonSupport;
diff --git a/thirdparty/thorvg/src/renderer/tvgText.cpp b/thirdparty/thorvg/src/renderer/tvgText.cpp
index 1fe244c11d..4b5eb35ce5 100644
--- a/thirdparty/thorvg/src/renderer/tvgText.cpp
+++ b/thirdparty/thorvg/src/renderer/tvgText.cpp
@@ -71,6 +71,21 @@ Result Text::load(const std::string& path) noexcept
}
+Result Text::load(const char* name, const char* data, uint32_t size, const string& mimeType, bool copy) noexcept
+{
+ if (!name || (size == 0 && data)) return Result::InvalidArguments;
+
+ //unload font
+ if (!data) {
+ if (LoaderMgr::retrieve(name)) return Result::Success;
+ return Result::InsufficientCondition;
+ }
+
+ if (!LoaderMgr::loader(name, data, size, mimeType, copy)) return Result::NonSupport;
+ return Result::Success;
+}
+
+
Result Text::unload(const std::string& path) noexcept
{
if (LoaderMgr::retrieve(path)) return Result::Success;
diff --git a/thirdparty/thorvg/src/renderer/tvgText.h b/thirdparty/thorvg/src/renderer/tvgText.h
index f4fb12259a..c56ce8b878 100644
--- a/thirdparty/thorvg/src/renderer/tvgText.h
+++ b/thirdparty/thorvg/src/renderer/tvgText.h
@@ -100,7 +100,7 @@ struct Text::Impl
bool render(RenderMethod* renderer)
{
if (paint) return PP(paint)->render(renderer);
- return false;
+ return true;
}
bool load()
diff --git a/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp b/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp
index 7db77f6d0c..067e35b1f0 100644
--- a/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp
+++ b/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp
@@ -47,33 +47,39 @@ WgCanvas::WgCanvas() : Canvas(nullptr), pImpl(nullptr)
{
}
+
WgCanvas::~WgCanvas()
{
delete pImpl;
}
-Result WgCanvas::target(void* window, uint32_t w, uint32_t h) noexcept
+
+Result WgCanvas::target(void* instance, void* surface, uint32_t w, uint32_t h) noexcept
{
#ifdef THORVG_WG_RASTER_SUPPORT
- if (!window) return Result::InvalidArguments;
- if ((w == 0) || (h == 0)) return Result::InvalidArguments;
+ if (Canvas::pImpl->status != Status::Damanged && Canvas::pImpl->status != Status::Synced) {
+ return Result::InsufficientCondition;
+ }
+
+ if (!instance || !surface || (w == 0) || (h == 0)) return Result::InvalidArguments;
//We know renderer type, avoid dynamic_cast for performance.
auto renderer = static_cast<WgRenderer*>(Canvas::pImpl->renderer);
if (!renderer) return Result::MemoryCorruption;
- if (!renderer->target(window, w, h)) return Result::Unknown;
+ if (!renderer->target((WGPUInstance)instance, (WGPUSurface)surface, w, h)) return Result::Unknown;
Canvas::pImpl->vport = {0, 0, (int32_t)w, (int32_t)h};
renderer->viewport(Canvas::pImpl->vport);
//Paints must be updated again with this new target.
- Canvas::pImpl->needRefresh();
+ Canvas::pImpl->status = Status::Damanged;
return Result::Success;
#endif
return Result::NonSupport;
}
+
unique_ptr<WgCanvas> WgCanvas::gen() noexcept
{
#ifdef THORVG_WG_RASTER_SUPPORT