/* * Copyright 2017 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "include/core/SkCanvas.h" #include "include/core/SkSurface.h" #include "include/core/SkVertices.h" #include "src/core/SkVerticesPriv.h" #include "tests/Test.h" #include "tools/ToolUtils.h" static bool equal(const SkVertices* vert0, const SkVertices* vert1) { SkVertices::Info v0, v1; vert0->getInfo(&v0); vert1->getInfo(&v1); if (v0.fMode != v1.fMode) { return false; } if (v0.fVertexCount != v1.fVertexCount) { return false; } if (v0.fIndexCount != v1.fIndexCount) { return false; } if (v0.fPerVertexDataCount != v1.fPerVertexDataCount) { return false; } if (!!v0.fPerVertexData != !!v1.fPerVertexData) { return false; } if (!!v0.fTexCoords != !!v1.fTexCoords) { return false; } if (!!v0.fColors != !!v1.fColors) { return false; } for (int i = 0; i < v0.fVertexCount; ++i) { if (v0.fPositions[i] != v1.fPositions[i]) { return false; } if (v0.fTexCoords) { if (v0.fTexCoords[i] != v1.fTexCoords[i]) { return false; } } if (v0.fColors) { if (v0.fColors[i] != v1.fColors[i]) { return false; } } } int totalVertexDataCount = v0.fVertexCount * v0.fPerVertexDataCount; for (int i = 0; i < totalVertexDataCount; ++i) { if (v0.fPerVertexData[i] != v1.fPerVertexData[i]) { return false; } } for (int i = 0; i < v0.fIndexCount; ++i) { if (v0.fIndices[i] != v1.fIndices[i]) { return false; } } return true; } static void self_test(sk_sp v0, skiatest::Reporter* reporter) { sk_sp data = v0->encode(); sk_sp v1 = SkVertices::Decode(data->data(), data->size()); REPORTER_ASSERT(reporter, v0->uniqueID() != 0); REPORTER_ASSERT(reporter, v1->uniqueID() != 0); REPORTER_ASSERT(reporter, v0->uniqueID() != v1->uniqueID()); REPORTER_ASSERT(reporter, equal(v0.get(), v1.get())); } DEF_TEST(Vertices, reporter) { int vCount = 5; int iCount = 9; // odd value exercises padding logic in encode() // color-tex tests const uint32_t texFlags[] = { 0, SkVertices::kHasTexCoords_BuilderFlag }; const uint32_t colFlags[] = { 0, SkVertices::kHasColors_BuilderFlag }; for (auto texF : texFlags) { for (auto colF : colFlags) { uint32_t flags = texF | colF; SkVertices::Builder builder(SkVertices::kTriangles_VertexMode, vCount, iCount, flags); REPORTER_ASSERT(reporter, builder.vertexCount() == vCount); REPORTER_ASSERT(reporter, builder.indexCount() == iCount); for (int i = 0; i < vCount; ++i) { float x = (float)i; builder.positions()[i].set(x, 1); if (builder.texCoords()) { builder.texCoords()[i].set(x, 2); } if (builder.colors()) { builder.colors()[i] = SkColorSetARGB(0xFF, i, 0x80, 0); } } for (int i = 0; i < builder.indexCount(); ++i) { builder.indices()[i] = i % vCount; } self_test(builder.detach(), reporter); } } // per-vertex-data tests for (int perVertexDataCount : {0, 1, 2, 3, 4, 7, 32}) { SkVertices::Builder builder(SkVertices::kTriangles_VertexMode, vCount, iCount, SkVertices::CustomLayout{perVertexDataCount}); REPORTER_ASSERT(reporter, builder.vertexCount() == vCount); REPORTER_ASSERT(reporter, builder.indexCount() == iCount); REPORTER_ASSERT(reporter, builder.perVertexDataCount() == perVertexDataCount); for (int i = 0; i < builder.vertexCount(); ++i) { builder.positions()[i].set((float)i, 1); for (int j = 0; j < builder.perVertexDataCount(); ++j) { builder.perVertexData()[i * perVertexDataCount + j] = (float)j; } } for (int i = 0; i < builder.indexCount(); ++i) { builder.indices()[i] = i % vCount; } self_test(builder.detach(), reporter); } { // This has the maximum number of vertices to be rewritten as indexed triangles without // overflowing a 16bit index. SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, UINT16_MAX + 1, 0, SkVertices::kHasColors_BuilderFlag); REPORTER_ASSERT(reporter, builder.isValid()); } { // This has too many to be rewritten. SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, UINT16_MAX + 2, 0, SkVertices::kHasColors_BuilderFlag); REPORTER_ASSERT(reporter, !builder.isValid()); } { // Only two vertices - can't be rewritten. SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, 2, 0, SkVertices::kHasColors_BuilderFlag); REPORTER_ASSERT(reporter, !builder.isValid()); } { // Minimum number of indices to be rewritten. SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, 10, 3, SkVertices::kHasColors_BuilderFlag); REPORTER_ASSERT(reporter, builder.isValid()); } { // Too few indices to be rewritten. SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, 10, 2, SkVertices::kHasColors_BuilderFlag); REPORTER_ASSERT(reporter, !builder.isValid()); } // validity tests for per-vertex-data { // negative count is bad SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, 10, 0, SkVertices::CustomLayout{-1}); REPORTER_ASSERT(reporter, !builder.isValid()); } { // zero-per-vertex-data should be ok SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, 10, 0, SkVertices::CustomLayout{0}); REPORTER_ASSERT(reporter, builder.isValid()); REPORTER_ASSERT(reporter, builder.perVertexDataCount() == 0); REPORTER_ASSERT(reporter, builder.perVertexData() == nullptr); } { // "normal" number of per-vertex-data SkVertices::Builder builder(SkVertices::kTriangleFan_VertexMode, 10, 0, SkVertices::CustomLayout{4}); REPORTER_ASSERT(reporter, builder.isValid()); REPORTER_ASSERT(reporter, builder.perVertexDataCount() == 4); REPORTER_ASSERT(reporter, builder.perVertexData() != nullptr); } } static void fill_triangle(SkCanvas* canvas, const SkPoint pts[], SkColor c) { SkColor colors[] = { c, c, c }; auto verts = SkVertices::MakeCopy(SkVertices::kTriangles_VertexMode, 3, pts, nullptr, colors); canvas->drawVertices(verts, SkBlendMode::kSrc, SkPaint()); } DEF_TEST(Vertices_clipping, reporter) { // A very large triangle has to be geometrically clipped (since its "fast" clipping is // normally done in after building SkFixed coordinates). Check that we handle this. // (and don't assert). auto surf = SkSurface::MakeRasterN32Premul(3, 3); SkPoint pts[] = { { -10, 1 }, { -10, 2 }, { 1e9f, 1.5f } }; fill_triangle(surf->getCanvas(), pts, SK_ColorBLACK); ToolUtils::PixelIter iter(surf.get()); SkIPoint loc; while (void* addr = iter.next(&loc)) { SkPMColor c = *(SkPMColor*)addr; if (loc.fY == 1) { REPORTER_ASSERT(reporter, c == 0xFF000000); } else { REPORTER_ASSERT(reporter, c == 0); } } }