utc: increase coverage SwEngine
authorMichal Maciola <m.maciola@samsung.com>
Thu, 19 Aug 2021 09:53:54 +0000 (11:53 +0200)
committerJunsuChoi <jsuya.choi@samsung.com>
Mon, 23 Aug 2021 01:21:39 +0000 (10:21 +0900)
test/meson.build
test/testSwEngine.cpp [new file with mode: 0644]

index 11ed926..0cd4d45 100644 (file)
@@ -8,6 +8,7 @@ test_file = [
     'testShape.cpp',
     'testSwCanvas.cpp',
     'testSwCanvasBase.cpp',
+    'testSwEngine.cpp',
 ]
 
 tests = executable('tvgUnitTests',
diff --git a/test/testSwEngine.cpp b/test/testSwEngine.cpp
new file mode 100644 (file)
index 0000000..c010847
--- /dev/null
@@ -0,0 +1,725 @@
+/*
+ * Copyright (c) 2021 Samsung Electronics Co., Ltd. All rights reserved.
+
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+
+ * The above copyright notice and this permission notice shall be included in all
+ * copies or substantial portions of the Software.
+
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+ * SOFTWARE.
+ */
+
+#include <thorvg.h>
+#include "catch.hpp"
+
+using namespace tvg;
+
+
+TEST_CASE("Basic draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Arc Line
+    auto shape1 = tvg::Shape::gen();
+    REQUIRE(shape1);
+
+    REQUIRE(shape1->appendArc(150, 150, 80, 10, 180, false) == Result::Success);
+    REQUIRE(shape1->stroke(255, 255, 255, 255) == Result::Success);
+    REQUIRE(shape1->stroke(2) == Result::Success);
+    REQUIRE(canvas->push(move(shape1)) == Result::Success);
+
+    //Cubic Line
+    auto shape2 = tvg::Shape::gen();
+    REQUIRE(shape2);
+
+    REQUIRE(shape2->moveTo(50, 25) == Result::Success);
+    REQUIRE(shape2->cubicTo(62, 25, 75, 38, 75, 50) == Result::Success);
+    REQUIRE(shape2->close() == Result::Success);
+    REQUIRE(shape2->stroke(1) == Result::Success);
+    REQUIRE(canvas->push(move(shape2)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("Rect Draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Basic shapes and masking
+    auto basicShape = tvg::Shape::gen();
+    REQUIRE(basicShape);
+    auto basicMask = tvg::Shape::gen();
+    REQUIRE(basicMask);
+    REQUIRE(basicShape->appendRect(0, 0, 50, 50, 0, 0) == Result::Success);
+    REQUIRE(basicMask->appendRect(10, 10, 30, 30, 0, 0) == Result::Success);
+    REQUIRE(basicShape->fill(255, 255, 255, 255) == Result::Success);
+
+    auto basicShape2 = std::unique_ptr<Shape>(static_cast<Shape*>(basicShape->duplicate()));
+    REQUIRE(basicShape2);
+    auto basicMask2 = std::unique_ptr<Shape>(static_cast<Shape*>(basicMask->duplicate()));
+    REQUIRE(basicMask2);
+
+    auto basicShape3 = std::unique_ptr<Shape>(static_cast<Shape*>(basicShape->duplicate()));
+    REQUIRE(basicShape3);
+    auto basicMask3 = std::unique_ptr<Shape>(static_cast<Shape*>(basicMask->duplicate()));
+    REQUIRE(basicMask3);
+
+    auto basicShape4 = std::unique_ptr<Shape>(static_cast<Shape*>(basicShape->duplicate()));
+    REQUIRE(basicShape4);
+
+    REQUIRE(basicShape->composite(move(basicMask), tvg::CompositeMethod::AlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(basicShape)) == Result::Success);
+
+    REQUIRE(basicShape2->composite(move(basicMask2), tvg::CompositeMethod::InvAlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(basicShape2)) == Result::Success);
+
+    REQUIRE(basicShape3->composite(move(basicMask3), tvg::CompositeMethod::ClipPath) == Result::Success);
+    REQUIRE(canvas->push(move(basicShape3)) == Result::Success);
+
+    REQUIRE(canvas->push(move(basicShape4)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("RLE Rect Draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Basic shapes and masking
+    auto basicShape = tvg::Shape::gen();
+    REQUIRE(basicShape);
+    auto basicMask = tvg::Shape::gen();
+    REQUIRE(basicMask);
+    REQUIRE(basicShape->appendRect(0, 0, 50, 50, 50, 50) == Result::Success);
+    REQUIRE(basicMask->appendRect(10, 10, 30, 30, 0, 0) == Result::Success);
+    REQUIRE(basicShape->fill(255, 255, 255, 100) == Result::Success);
+
+    auto basicShape2 = std::unique_ptr<Shape>(static_cast<Shape*>(basicShape->duplicate()));
+    REQUIRE(basicShape2);
+    auto basicMask2 = std::unique_ptr<Shape>(static_cast<Shape*>(basicMask->duplicate()));
+    REQUIRE(basicMask2);
+
+    auto basicShape3 = std::unique_ptr<Shape>(static_cast<Shape*>(basicShape->duplicate()));
+    REQUIRE(basicShape3);
+    auto basicMask3 = std::unique_ptr<Shape>(static_cast<Shape*>(basicMask->duplicate()));
+    REQUIRE(basicMask3);
+
+    auto basicShape4 = std::unique_ptr<Shape>(static_cast<Shape*>(basicShape->duplicate()));
+    REQUIRE(basicShape4);
+
+    REQUIRE(basicShape->composite(move(basicMask), tvg::CompositeMethod::AlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(basicShape)) == Result::Success);
+
+    REQUIRE(basicShape2->composite(move(basicMask2), tvg::CompositeMethod::InvAlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(basicShape2)) == Result::Success);
+
+    REQUIRE(basicShape3->composite(move(basicMask3), tvg::CompositeMethod::ClipPath) == Result::Success);
+    REQUIRE(canvas->push(move(basicShape3)) == Result::Success);
+
+    REQUIRE(canvas->push(move(basicShape4)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("Filling Draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 0, 0) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 0, 0) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    REQUIRE(canvas->push(move(shape3)) == Result::Success);
+    REQUIRE(canvas->push(move(shape4)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("Filling Opaque Draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 255},
+        {0.2f, 50, 25, 50, 255},
+        {0.5f, 100, 100, 100, 255},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 0, 0) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 0, 0) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    REQUIRE(canvas->push(move(shape3)) == Result::Success);
+    REQUIRE(canvas->push(move(shape4)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("Filling AlphaMask", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Mask
+    auto mask = tvg::Shape::gen();
+    REQUIRE(mask);
+    REQUIRE(mask->appendCircle(50, 50, 50, 50) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 0, 0) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 0, 0) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    //Scene
+    auto scene = tvg::Scene::gen();
+    REQUIRE(scene);
+    REQUIRE(scene->push(move(shape3)) == Result::Success);
+    REQUIRE(scene->push(move(shape4)) == Result::Success);
+    REQUIRE(scene->composite(move(mask), tvg::CompositeMethod::AlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(scene)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("Filling InvAlphaMask", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Mask
+    auto mask = tvg::Shape::gen();
+    REQUIRE(mask);
+    REQUIRE(mask->appendCircle(50, 50, 50, 50) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 0, 0) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 0, 0) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    //Scene
+    auto scene = tvg::Scene::gen();
+    REQUIRE(scene);
+    REQUIRE(scene->push(move(shape3)) == Result::Success);
+    REQUIRE(scene->push(move(shape4)) == Result::Success);
+    REQUIRE(scene->composite(move(mask), tvg::CompositeMethod::InvAlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(scene)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("Filling ClipPath", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Mask
+    auto mask = tvg::Shape::gen();
+    REQUIRE(mask);
+    REQUIRE(mask->appendCircle(50, 50, 50, 50) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 0, 0) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 0, 0) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    //Scene
+    auto scene = tvg::Scene::gen();
+    REQUIRE(scene);
+    REQUIRE(scene->push(move(shape3)) == Result::Success);
+    REQUIRE(scene->push(move(shape4)) == Result::Success);
+    REQUIRE(scene->composite(move(mask), tvg::CompositeMethod::ClipPath) == Result::Success);
+    REQUIRE(canvas->push(move(scene)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("RLE Filling Draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 10, 10) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 10, 10) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    REQUIRE(canvas->push(move(shape3)) == Result::Success);
+    REQUIRE(canvas->push(move(shape4)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("RLE Filling Opaque Draw", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 255},
+        {0.2f, 50, 25, 50, 255},
+        {0.5f, 100, 100, 100, 255},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 10, 10) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 10, 10) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    REQUIRE(canvas->push(move(shape3)) == Result::Success);
+    REQUIRE(canvas->push(move(shape4)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("RLE Filling AlphaMask", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Mask
+    auto mask = tvg::Shape::gen();
+    REQUIRE(mask);
+    REQUIRE(mask->appendCircle(50, 50, 50, 50) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 10, 10) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 10, 10) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    //Scene
+    auto scene = tvg::Scene::gen();
+    REQUIRE(scene);
+    REQUIRE(scene->push(move(shape3)) == Result::Success);
+    REQUIRE(scene->push(move(shape4)) == Result::Success);
+    REQUIRE(scene->composite(move(mask), tvg::CompositeMethod::AlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(scene)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("RLE Filling InvAlphaMask", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Mask
+    auto mask = tvg::Shape::gen();
+    REQUIRE(mask);
+    REQUIRE(mask->appendCircle(50, 50, 50, 50) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 10, 10) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 10, 10) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    //Scene
+    auto scene = tvg::Scene::gen();
+    REQUIRE(scene);
+    REQUIRE(scene->push(move(shape3)) == Result::Success);
+    REQUIRE(scene->push(move(shape4)) == Result::Success);
+    REQUIRE(scene->composite(move(mask), tvg::CompositeMethod::InvAlphaMask) == Result::Success);
+    REQUIRE(canvas->push(move(scene)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}
+
+
+TEST_CASE("RLE Filling ClipPath", "[tvgSwEngine]")
+{
+    REQUIRE(Initializer::init(CanvasEngine::Sw, 0) == Result::Success);
+
+    auto canvas = SwCanvas::gen();
+    REQUIRE(canvas);
+
+    uint32_t buffer[100*100];
+    REQUIRE(canvas->target(buffer, 100, 100, 100, SwCanvas::Colorspace::ABGR8888) == Result::Success);
+
+    //Fill
+    auto linearFill = LinearGradient::gen();
+    REQUIRE(linearFill);
+
+    auto radialFill = RadialGradient::gen();
+    REQUIRE(radialFill);
+
+    Fill::ColorStop cs[4] = {
+        {0.0f, 0, 0, 0, 0},
+        {0.2f, 50, 25, 50, 25},
+        {0.5f, 100, 100, 100, 125},
+        {1.0f, 255, 255, 255, 255}
+    };
+    REQUIRE(linearFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(radialFill->colorStops(cs, 4) == Result::Success);
+    REQUIRE(linearFill->linear(0.0f, 0.0f, 100.0f, 120.0f) == Result::Success);
+    REQUIRE(radialFill->radial(50.0f, 50.0f, 50.0f) == Result::Success);
+
+    //Mask
+    auto mask = tvg::Shape::gen();
+    REQUIRE(mask);
+    REQUIRE(mask->appendCircle(50, 50, 50, 50) == Result::Success);
+
+    //Filled Shapes
+    auto shape3 = tvg::Shape::gen();
+    REQUIRE(shape3);
+    auto shape4 = tvg::Shape::gen();
+    REQUIRE(shape4);
+    REQUIRE(shape3->appendRect(0, 0, 50, 50, 10, 10) == Result::Success);
+    REQUIRE(shape4->appendRect(50, 0, 50, 50, 10, 10) == Result::Success);
+
+    REQUIRE(shape3->fill(move(linearFill)) == Result::Success);
+    REQUIRE(shape4->fill(move(radialFill)) == Result::Success);
+
+    //Scene
+    auto scene = tvg::Scene::gen();
+    REQUIRE(scene);
+    REQUIRE(scene->push(move(shape3)) == Result::Success);
+    REQUIRE(scene->push(move(shape4)) == Result::Success);
+    REQUIRE(scene->composite(move(mask), tvg::CompositeMethod::ClipPath) == Result::Success);
+    REQUIRE(canvas->push(move(scene)) == Result::Success);
+
+    //Draw
+    REQUIRE(canvas->draw() == Result::Success);
+    REQUIRE(canvas->sync() == Result::Success);
+
+    REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
+}