Home | History | Annotate | Download | only in src
      1 /*
      2  * Copyright 2018 Google Inc.
      3  *
      4  * Use of this source code is governed by a BSD-style license that can be
      5  * found in the LICENSE file.
      6  */
      7 
      8 #include "skqp.h"
      9 
     10 #include "../../../src/core/SkStreamPriv.h"
     11 #include "../../tools/fonts/SkTestFontMgr.h"
     12 #include "GrContext.h"
     13 #include "GrContextOptions.h"
     14 #include "GrContextPriv.h"
     15 #include "SkFontMgrPriv.h"
     16 #include "SkFontStyle.h"
     17 #include "SkGraphics.h"
     18 #include "SkImageInfoPriv.h"
     19 #include "SkOSFile.h"
     20 #include "SkOSPath.h"
     21 #include "SkPngEncoder.h"
     22 #include "SkStream.h"
     23 #include "SkSurface.h"
     24 #include "Test.h"
     25 #include "gl/GLTestContext.h"
     26 #include "gm.h"
     27 #include "vk/VkTestContext.h"
     28 
     29 #include <algorithm>
     30 #include <cinttypes>
     31 #include <sstream>
     32 
     33 #include "skqp_model.h"
     34 
     35 #define IMAGES_DIRECTORY_PATH "images"
     36 #define PATH_MAX_PNG "max.png"
     37 #define PATH_MIN_PNG "min.png"
     38 #define PATH_IMG_PNG "image.png"
     39 #define PATH_ERR_PNG "errors.png"
     40 #define PATH_MODEL "model"
     41 
     42 static constexpr char kRenderTestCSVReport[] = "out.csv";
     43 static constexpr char kRenderTestReportPath[] = "report.html";
     44 static constexpr char kRenderTestsPath[] = "skqp/rendertests.txt";
     45 static constexpr char kUnitTestReportPath[] = "unit_tests.txt";
     46 static constexpr char kUnitTestsPath[]   = "skqp/unittests.txt";
     47 
     48 // Kind of like Python's readlines(), but without any allocation.
     49 // Calls f() on each line.
     50 // F is [](const char*, size_t) -> void
     51 template <typename F>
     52 static void readlines(const void* data, size_t size, F f) {
     53     const char* start = (const char*)data;
     54     const char* end = start + size;
     55     const char* ptr = start;
     56     while (ptr < end) {
     57         while (*ptr++ != '\n' && ptr < end) {}
     58         size_t len = ptr - start;
     59         f(start, len);
     60         start = ptr;
     61     }
     62 }
     63 
     64 static void get_unit_tests(SkQPAssetManager* mgr, std::vector<SkQP::UnitTest>* unitTests) {
     65     std::unordered_set<std::string> testset;
     66     auto insert = [&testset](const char* s, size_t l) {
     67         SkASSERT(l > 1) ;
     68         if (l > 0 && s[l - 1] == '\n') {  // strip line endings.
     69             --l;
     70         }
     71         if (l > 0) {  // only add non-empty strings.
     72             testset.insert(std::string(s, l));
     73         }
     74     };
     75     if (sk_sp<SkData> dat = mgr->open(kUnitTestsPath)) {
     76         readlines(dat->data(), dat->size(), insert);
     77     }
     78     for (const skiatest::Test& test : skiatest::TestRegistry::Range()) {
     79         if ((testset.empty() || testset.count(std::string(test.name)) > 0) && test.needsGpu) {
     80             unitTests->push_back(&test);
     81         }
     82     }
     83     auto lt = [](SkQP::UnitTest u, SkQP::UnitTest v) { return strcmp(u->name, v->name) < 0; };
     84     std::sort(unitTests->begin(), unitTests->end(), lt);
     85 }
     86 
     87 static void get_render_tests(SkQPAssetManager* mgr,
     88                              std::vector<SkQP::GMFactory>* gmlist,
     89                              std::unordered_map<std::string, int64_t>* gmThresholds) {
     90     auto insert = [gmThresholds](const char* s, size_t l) {
     91         SkASSERT(l > 1) ;
     92         if (l > 0 && s[l - 1] == '\n') {  // strip line endings.
     93             --l;
     94         }
     95         if (l == 0) {
     96             return;
     97         }
     98         const char* end = s + l;
     99         const char* ptr = s;
    100         constexpr char kDelimeter = ',';
    101         while (ptr < end && *ptr != kDelimeter) { ++ptr; }
    102         if (ptr + 1 >= end) {
    103             SkASSERT(false);  // missing delimeter
    104             return;
    105         }
    106         std::string key(s, ptr - s);
    107         ++ptr;  // skip delimeter
    108         std::string number(ptr, end - ptr);  // null-terminated copy.
    109         int64_t value = 0;
    110         if (1 != sscanf(number.c_str(), "%" SCNd64 , &value)) {
    111             SkASSERT(false);  // Not a number
    112             return;
    113         }
    114         gmThresholds->insert({std::move(key), value});  // (*gmThresholds)[s] = value;
    115     };
    116     if (sk_sp<SkData> dat = mgr->open(kRenderTestsPath)) {
    117         readlines(dat->data(), dat->size(), insert);
    118     }
    119     using GmAndName = std::pair<SkQP::GMFactory, std::string>;
    120     std::vector<GmAndName> gmsWithNames;
    121     for (skiagm::GMFactory f : skiagm::GMRegistry::Range()) {
    122         std::string name = SkQP::GetGMName(f);
    123         if ((gmThresholds->empty() || gmThresholds->count(name) > 0)) {
    124             gmsWithNames.push_back(std::make_pair(f, std::move(name)));
    125         }
    126     }
    127     std::sort(gmsWithNames.begin(), gmsWithNames.end(),
    128               [](GmAndName u, GmAndName v) { return u.second < v.second; });
    129     gmlist->reserve(gmsWithNames.size());
    130     for (const GmAndName& gmn : gmsWithNames) {
    131         gmlist->push_back(gmn.first);
    132     }
    133 }
    134 
    135 static std::unique_ptr<sk_gpu_test::TestContext> make_test_context(SkQP::SkiaBackend backend) {
    136     using U = std::unique_ptr<sk_gpu_test::TestContext>;
    137     switch (backend) {
    138         case SkQP::SkiaBackend::kGL:
    139             return U(sk_gpu_test::CreatePlatformGLTestContext(kGL_GrGLStandard, nullptr));
    140         case SkQP::SkiaBackend::kGLES:
    141             return U(sk_gpu_test::CreatePlatformGLTestContext(kGLES_GrGLStandard, nullptr));
    142 #ifdef SK_VULKAN
    143         case SkQP::SkiaBackend::kVulkan:
    144             return U(sk_gpu_test::CreatePlatformVkTestContext(nullptr));
    145 #endif
    146         default:
    147             return nullptr;
    148     }
    149 }
    150 
    151 static GrContextOptions context_options(skiagm::GM* gm = nullptr) {
    152     GrContextOptions grContextOptions;
    153     grContextOptions.fAllowPathMaskCaching = true;
    154     grContextOptions.fSuppressPathRendering = true;
    155     grContextOptions.fDisableDriverCorrectnessWorkarounds = true;
    156     if (gm) {
    157         gm->modifyGrContextOptions(&grContextOptions);
    158     }
    159     return grContextOptions;
    160 }
    161 
    162 static std::vector<SkQP::SkiaBackend> get_backends() {
    163     std::vector<SkQP::SkiaBackend> result;
    164     SkQP::SkiaBackend backends[] = {
    165         #ifndef SK_BUILD_FOR_ANDROID
    166         SkQP::SkiaBackend::kGL,  // Used for testing on desktop machines.
    167         #endif
    168         SkQP::SkiaBackend::kGLES,
    169         #ifdef SK_VULKAN
    170         SkQP::SkiaBackend::kVulkan,
    171         #endif
    172     };
    173     for (SkQP::SkiaBackend backend : backends) {
    174         std::unique_ptr<sk_gpu_test::TestContext> testCtx = make_test_context(backend);
    175         if (testCtx) {
    176             testCtx->makeCurrent();
    177             if (nullptr != testCtx->makeGrContext(context_options())) {
    178                 result.push_back(backend);
    179             }
    180         }
    181     }
    182     SkASSERT_RELEASE(result.size() > 0);
    183     return result;
    184 }
    185 
    186 static void print_backend_info(const char* dstPath,
    187                                const std::vector<SkQP::SkiaBackend>& backends) {
    188 #ifdef SK_ENABLE_DUMP_GPU
    189     SkFILEWStream out(dstPath);
    190     out.writeText("[\n");
    191     for (SkQP::SkiaBackend backend : backends) {
    192         if (std::unique_ptr<sk_gpu_test::TestContext> testCtx = make_test_context(backend)) {
    193             testCtx->makeCurrent();
    194             if (sk_sp<GrContext> ctx = testCtx->makeGrContext(context_options())) {
    195                 SkString info = ctx->priv().dump();
    196                 // remove null
    197                 out.write(info.c_str(), info.size());
    198                 out.writeText(",\n");
    199             }
    200         }
    201     }
    202     out.writeText("]\n");
    203 #endif
    204 }
    205 
    206 static void encode_png(const SkBitmap& src, const std::string& dst) {
    207     SkFILEWStream wStream(dst.c_str());
    208     SkPngEncoder::Options options;
    209     bool success = wStream.isValid() && SkPngEncoder::Encode(&wStream, src.pixmap(), options);
    210     SkASSERT_RELEASE(success);
    211 }
    212 
    213 static void write_to_file(const sk_sp<SkData>& src, const std::string& dst) {
    214     SkFILEWStream wStream(dst.c_str());
    215     bool success = wStream.isValid() && wStream.write(src->data(), src->size());
    216     SkASSERT_RELEASE(success);
    217 }
    218 
    219 ////////////////////////////////////////////////////////////////////////////////
    220 
    221 const char* SkQP::GetBackendName(SkQP::SkiaBackend b) {
    222     switch (b) {
    223         case SkQP::SkiaBackend::kGL:     return "gl";
    224         case SkQP::SkiaBackend::kGLES:   return "gles";
    225         case SkQP::SkiaBackend::kVulkan: return "vk";
    226     }
    227     return "";
    228 }
    229 
    230 std::string SkQP::GetGMName(SkQP::GMFactory f) {
    231     std::unique_ptr<skiagm::GM> gm(f ? f(nullptr) : nullptr);
    232     return std::string(gm ? gm->getName() : "");
    233 }
    234 
    235 const char* SkQP::GetUnitTestName(SkQP::UnitTest t) { return t->name; }
    236 
    237 SkQP::SkQP() {}
    238 
    239 SkQP::~SkQP() {}
    240 
    241 void SkQP::init(SkQPAssetManager* am, const char* reportDirectory) {
    242     SkASSERT_RELEASE(!fAssetManager);
    243     SkASSERT_RELEASE(am);
    244     fAssetManager = am;
    245     fReportDirectory = reportDirectory;
    246 
    247     SkGraphics::Init();
    248     gSkFontMgr_DefaultFactory = &sk_tool_utils::MakePortableFontMgr;
    249 
    250     /* If the file "skqp/rendertests.txt" does not exist or is empty, run all
    251        render tests.  Otherwise only run tests mentioned in that file.  */
    252     get_render_tests(fAssetManager, &fGMs, &fGMThresholds);
    253     /* If the file "skqp/unittests.txt" does not exist or is empty, run all gpu
    254        unit tests.  Otherwise only run tests mentioned in that file.  */
    255     get_unit_tests(fAssetManager, &fUnitTests);
    256     fSupportedBackends = get_backends();
    257 
    258     print_backend_info((fReportDirectory + "/grdump.txt").c_str(), fSupportedBackends);
    259 }
    260 
    261 std::tuple<SkQP::RenderOutcome, std::string> SkQP::evaluateGM(SkQP::SkiaBackend backend,
    262                                                               SkQP::GMFactory gmFact) {
    263     SkASSERT_RELEASE(fAssetManager);
    264     static constexpr SkQP::RenderOutcome kError = {INT_MAX, INT_MAX, INT64_MAX};
    265     static constexpr SkQP::RenderOutcome kPass = {0, 0, 0};
    266 
    267     std::unique_ptr<sk_gpu_test::TestContext> testCtx = make_test_context(backend);
    268     if (!testCtx) {
    269         return std::make_tuple(kError, "Skia Failure: test context");
    270     }
    271     testCtx->makeCurrent();
    272 
    273     SkASSERT(gmFact);
    274     std::unique_ptr<skiagm::GM> gm(gmFact(nullptr));
    275     SkASSERT(gm);
    276     const char* const name = gm->getName();
    277     const SkISize size = gm->getISize();
    278     const int w = size.width();
    279     const int h = size.height();
    280     const SkImageInfo info =
    281         SkImageInfo::Make(w, h, skqp::kColorType, kPremul_SkAlphaType, nullptr);
    282     const SkSurfaceProps props(0, SkSurfaceProps::kLegacyFontHost_InitType);
    283 
    284     sk_sp<SkSurface> surf = SkSurface::MakeRenderTarget(
    285             testCtx->makeGrContext(context_options(gm.get())).get(),
    286             SkBudgeted::kNo, info, 0, &props);
    287     if (!surf) {
    288         return std::make_tuple(kError, "Skia Failure: gr-context");
    289     }
    290     gm->draw(surf->getCanvas());
    291 
    292     SkBitmap image;
    293     image.allocPixels(SkImageInfo::Make(w, h, skqp::kColorType, skqp::kAlphaType));
    294 
    295     // SkColorTypeBytesPerPixel should be constexpr, but is not.
    296     SkASSERT(SkColorTypeBytesPerPixel(skqp::kColorType) == sizeof(uint32_t));
    297     // Call readPixels because we need to compare pixels.
    298     if (!surf->readPixels(image.pixmap(), 0, 0)) {
    299         return std::make_tuple(kError, "Skia Failure: read pixels");
    300     }
    301     int64_t passingThreshold = fGMThresholds.empty() ? -1 : fGMThresholds[std::string(name)];
    302 
    303     if (-1 == passingThreshold) {
    304         return std::make_tuple(kPass, "");
    305     }
    306     skqp::ModelResult modelResult =
    307         skqp::CheckAgainstModel(name, image.pixmap(), fAssetManager);
    308 
    309     if (!modelResult.fErrorString.empty()) {
    310         return std::make_tuple(kError, std::move(modelResult.fErrorString));
    311     }
    312     fRenderResults.push_back(SkQP::RenderResult{backend, gmFact, modelResult.fOutcome});
    313     if (modelResult.fOutcome.fMaxError <= passingThreshold) {
    314         return std::make_tuple(kPass, "");
    315     }
    316     std::string imagesDirectory = fReportDirectory + "/" IMAGES_DIRECTORY_PATH;
    317     if (!sk_mkdir(imagesDirectory.c_str())) {
    318         SkDebugf("ERROR: sk_mkdir('%s');\n", imagesDirectory.c_str());
    319         return std::make_tuple(modelResult.fOutcome, "");
    320     }
    321     std::ostringstream tmp;
    322     tmp << imagesDirectory << '/' << SkQP::GetBackendName(backend) << '_' << name << '_';
    323     std::string imagesPathPrefix1 = tmp.str();
    324     tmp = std::ostringstream();
    325     tmp << imagesDirectory << '/' << PATH_MODEL << '_' << name << '_';
    326     std::string imagesPathPrefix2 = tmp.str();
    327     encode_png(image,                  imagesPathPrefix1 + PATH_IMG_PNG);
    328     encode_png(modelResult.fErrors,    imagesPathPrefix1 + PATH_ERR_PNG);
    329     write_to_file(modelResult.fMaxPng, imagesPathPrefix2 + PATH_MAX_PNG);
    330     write_to_file(modelResult.fMinPng, imagesPathPrefix2 + PATH_MIN_PNG);
    331     return std::make_tuple(modelResult.fOutcome, "");
    332 }
    333 
    334 std::vector<std::string> SkQP::executeTest(SkQP::UnitTest test) {
    335     SkASSERT_RELEASE(fAssetManager);
    336     struct : public skiatest::Reporter {
    337         std::vector<std::string> fErrors;
    338         void reportFailed(const skiatest::Failure& failure) override {
    339             SkString desc = failure.toString();
    340             fErrors.push_back(std::string(desc.c_str(), desc.size()));
    341         }
    342     } r;
    343     GrContextOptions options;
    344     options.fDisableDriverCorrectnessWorkarounds = true;
    345     if (test->fContextOptionsProc) {
    346         test->fContextOptionsProc(&options);
    347     }
    348     test->proc(&r, options);
    349     fUnitTestResults.push_back(UnitTestResult{test, r.fErrors});
    350     return r.fErrors;
    351 }
    352 
    353 ////////////////////////////////////////////////////////////////////////////////
    354 
    355 static constexpr char kDocHead[] =
    356     "<!doctype html>\n"
    357     "<html lang=\"en\">\n"
    358     "<head>\n"
    359     "<meta charset=\"UTF-8\">\n"
    360     "<title>SkQP Report</title>\n"
    361     "<style>\n"
    362     "img { max-width:48%; border:1px green solid;\n"
    363     "      image-rendering: pixelated;\n"
    364     "      background-image:url('data:image/png;base64,iVBORw0KGgoA"
    365     "AAANSUhEUgAAABAAAAAQCAAAAAA6mKC9AAAAAXNSR0IArs4c6QAAAAJiS0dEAP+H"
    366     "j8y/AAAACXBIWXMAAA7DAAAOwwHHb6hkAAAAB3RJTUUH3gUBEi4DGRAQYgAAAB1J"
    367     "REFUGNNjfMoAAVJQmokBDdBHgPE/lPFsYN0BABdaAwN6tehMAAAAAElFTkSuQmCC"
    368     "'); }\n"
    369     "</style>\n"
    370     "<script>\n"
    371     "function ce(t) { return document.createElement(t); }\n"
    372     "function ct(n) { return document.createTextNode(n); }\n"
    373     "function ac(u,v) { return u.appendChild(v); }\n"
    374     "function br(u) { ac(u, ce(\"br\")); }\n"
    375     "function ma(s, c) { var a = ce(\"a\"); a.href = s; ac(a, c); return a; }\n"
    376     "function f(backend, gm, e1, e2, e3) {\n"
    377     "  var b = ce(\"div\");\n"
    378     "  var x = ce(\"h2\");\n"
    379     "  var t = backend + \"_\" + gm;\n"
    380     "  ac(x, ct(t));\n"
    381     "  ac(b, x);\n"
    382     "  ac(b, ct(\"backend: \" + backend));\n"
    383     "  br(b);\n"
    384     "  ac(b, ct(\"gm name: \" + gm));\n"
    385     "  br(b);\n"
    386     "  ac(b, ct(\"maximum error: \" + e1));\n"
    387     "  br(b);\n"
    388     "  ac(b, ct(\"bad pixel counts: \" + e2));\n"
    389     "  br(b);\n"
    390     "  ac(b, ct(\"total error: \" + e3));\n"
    391     "  br(b);\n"
    392     "  var q = \"" IMAGES_DIRECTORY_PATH "/\" + backend + \"_\" + gm + \"_\";\n"
    393     "  var p = \"" IMAGES_DIRECTORY_PATH "/"   PATH_MODEL  "_\" + gm + \"_\";\n"
    394     "  var i = ce(\"img\");\n"
    395     "  i.src = q + \"" PATH_IMG_PNG "\";\n"
    396     "  i.alt = \"img\";\n"
    397     "  ac(b, ma(i.src, i));\n"
    398     "  i = ce(\"img\");\n"
    399     "  i.src = q + \"" PATH_ERR_PNG "\";\n"
    400     "  i.alt = \"err\";\n"
    401     "  ac(b, ma(i.src, i));\n"
    402     "  br(b);\n"
    403     "  ac(b, ct(\"Expectation: \"));\n"
    404     "  ac(b, ma(p + \"" PATH_MAX_PNG "\", ct(\"max\")));\n"
    405     "  ac(b, ct(\" | \"));\n"
    406     "  ac(b, ma(p + \"" PATH_MIN_PNG "\", ct(\"min\")));\n"
    407     "  ac(b, ce(\"hr\"));\n"
    408     "  b.id = backend + \":\" + gm;\n"
    409     "  ac(document.body, b);\n"
    410     "  l = ce(\"li\");\n"
    411     "  ac(l, ct(\"[\" + e3 + \"] \"));\n"
    412     "  ac(l, ma(\"#\" + backend +\":\"+ gm , ct(t)));\n"
    413     "  ac(document.getElementById(\"toc\"), l);\n"
    414     "}\n"
    415     "function main() {\n";
    416 
    417 static constexpr char kDocMiddle[] =
    418     "}\n"
    419     "</script>\n"
    420     "</head>\n"
    421     "<body onload=\"main()\">\n"
    422     "<h1>SkQP Report</h1>\n";
    423 
    424 static constexpr char kDocTail[] =
    425     "<ul id=\"toc\"></ul>\n"
    426     "<hr>\n"
    427     "<p>Left image: test result<br>\n"
    428     "Right image: errors (white = no error, black = smallest error, red = biggest error; "
    429     "other errors are a color between black and red.)</p>\n"
    430     "<hr>\n"
    431     "</body>\n"
    432     "</html>\n";
    433 
    434 template <typename T>
    435 inline void write(SkWStream* wStream, const T& text) {
    436     wStream->write(text.c_str(), text.size());
    437 }
    438 
    439 void SkQP::makeReport() {
    440     SkASSERT_RELEASE(fAssetManager);
    441     int glesErrorCount = 0, vkErrorCount = 0, gles = 0, vk = 0;
    442 
    443     if (!sk_isdir(fReportDirectory.c_str())) {
    444         SkDebugf("Report destination does not exist: '%s'\n", fReportDirectory.c_str());
    445         return;
    446     }
    447     SkFILEWStream csvOut(SkOSPath::Join(fReportDirectory.c_str(), kRenderTestCSVReport).c_str());
    448     SkFILEWStream htmOut(SkOSPath::Join(fReportDirectory.c_str(), kRenderTestReportPath).c_str());
    449     SkASSERT_RELEASE(csvOut.isValid() && htmOut.isValid());
    450     htmOut.writeText(kDocHead);
    451     for (const SkQP::RenderResult& run : fRenderResults) {
    452         switch (run.fBackend) {
    453             case SkQP::SkiaBackend::kGLES: ++gles; break;
    454             case SkQP::SkiaBackend::kVulkan: ++vk; break;
    455             default: break;
    456         }
    457         const char* backendName = SkQP::GetBackendName(run.fBackend);
    458         std::string gmName = SkQP::GetGMName(run.fGM);
    459         const SkQP::RenderOutcome& outcome = run.fOutcome;
    460         auto str = SkStringPrintf("\"%s\",\"%s\",%d,%d,%" PRId64, backendName, gmName.c_str(),
    461                                   outcome.fMaxError, outcome.fBadPixelCount, outcome.fTotalError);
    462         write(&csvOut, SkStringPrintf("%s\n", str.c_str()));
    463 
    464         int64_t passingThreshold = fGMThresholds.empty() ? 0 : fGMThresholds[gmName];
    465         if (passingThreshold == -1 || outcome.fMaxError <= passingThreshold) {
    466             continue;
    467         }
    468         write(&htmOut, SkStringPrintf("  f(%s);\n", str.c_str()));
    469         switch (run.fBackend) {
    470             case SkQP::SkiaBackend::kGLES: ++glesErrorCount; break;
    471             case SkQP::SkiaBackend::kVulkan: ++vkErrorCount; break;
    472             default: break;
    473         }
    474     }
    475     htmOut.writeText(kDocMiddle);
    476     write(&htmOut, SkStringPrintf("<p>gles errors: %d (of %d)</br>\n"
    477                                   "vk errors: %d (of %d)</p>\n",
    478                                   glesErrorCount, gles, vkErrorCount, vk));
    479     htmOut.writeText(kDocTail);
    480     SkFILEWStream unitOut(SkOSPath::Join(fReportDirectory.c_str(), kUnitTestReportPath).c_str());
    481     SkASSERT_RELEASE(unitOut.isValid());
    482     for (const SkQP::UnitTestResult& result : fUnitTestResults) {
    483         unitOut.writeText(GetUnitTestName(result.fUnitTest));
    484         if (result.fErrors.empty()) {
    485             unitOut.writeText(" PASSED\n* * *\n");
    486         } else {
    487             write(&unitOut, SkStringPrintf(" FAILED (%u errors)\n", result.fErrors.size()));
    488             for (const std::string& err : result.fErrors) {
    489                 write(&unitOut, err);
    490                 unitOut.newline();
    491             }
    492             unitOut.writeText("* * *\n");
    493         }
    494     }
    495 }
    496