blob: 04f0369be69070cf05c4e0d4a897042e3e5015cd [file] [log] [blame]
Hal Canaryac7f23c2018-11-26 14:07:41 -05001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "tools/skqp/src/skqp.h"
Hal Canaryac7f23c2018-11-26 14:07:41 -05009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "gm/gm.h"
11#include "include/core/SkFontStyle.h"
12#include "include/core/SkGraphics.h"
13#include "include/core/SkStream.h"
14#include "include/core/SkSurface.h"
15#include "include/encode/SkPngEncoder.h"
16#include "include/gpu/GrContext.h"
17#include "include/gpu/GrContextOptions.h"
18#include "include/private/SkImageInfoPriv.h"
19#include "src/core/SkFontMgrPriv.h"
20#include "src/core/SkOSFile.h"
21#include "src/core/SkStreamPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "src/utils/SkOSPath.h"
23#include "tests/Test.h"
24#include "tools/fonts/TestFontMgr.h"
25#include "tools/gpu/gl/GLTestContext.h"
26#include "tools/gpu/vk/VkTestContext.h"
Hal Canaryac7f23c2018-11-26 14:07:41 -050027
Ben Wagner0b9b1f12019-04-04 18:00:05 -040028#include <limits.h>
Hal Canaryac7f23c2018-11-26 14:07:41 -050029#include <algorithm>
30#include <cinttypes>
31#include <sstream>
32
Mike Kleinc0bd9f92019-04-23 12:05:21 -050033#include "tools/skqp/src/skqp_model.h"
Hal Canaryac7f23c2018-11-26 14:07:41 -050034
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
42static constexpr char kRenderTestCSVReport[] = "out.csv";
43static constexpr char kRenderTestReportPath[] = "report.html";
44static constexpr char kRenderTestsPath[] = "skqp/rendertests.txt";
45static constexpr char kUnitTestReportPath[] = "unit_tests.txt";
46static 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
51template <typename F>
52static 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
64static 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
87static 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
135static 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) {
Hal Canarya67523a2019-08-28 09:10:32 -0400138// TODO(halcanary): Fuchsia will have SK_SUPPORT_GPU and SK_VULKAN, but *not* SK_GL.
139#ifdef SK_GL
Hal Canaryac7f23c2018-11-26 14:07:41 -0500140 case SkQP::SkiaBackend::kGL:
141 return U(sk_gpu_test::CreatePlatformGLTestContext(kGL_GrGLStandard, nullptr));
142 case SkQP::SkiaBackend::kGLES:
143 return U(sk_gpu_test::CreatePlatformGLTestContext(kGLES_GrGLStandard, nullptr));
Hal Canarya67523a2019-08-28 09:10:32 -0400144#endif
Hal Canaryac7f23c2018-11-26 14:07:41 -0500145#ifdef SK_VULKAN
146 case SkQP::SkiaBackend::kVulkan:
147 return U(sk_gpu_test::CreatePlatformVkTestContext(nullptr));
148#endif
149 default:
150 return nullptr;
151 }
152}
153
154static GrContextOptions context_options(skiagm::GM* gm = nullptr) {
155 GrContextOptions grContextOptions;
156 grContextOptions.fAllowPathMaskCaching = true;
Hal Canaryac7f23c2018-11-26 14:07:41 -0500157 grContextOptions.fDisableDriverCorrectnessWorkarounds = true;
158 if (gm) {
159 gm->modifyGrContextOptions(&grContextOptions);
160 }
161 return grContextOptions;
162}
163
164static std::vector<SkQP::SkiaBackend> get_backends() {
165 std::vector<SkQP::SkiaBackend> result;
166 SkQP::SkiaBackend backends[] = {
Hal Canarya67523a2019-08-28 09:10:32 -0400167 #ifdef SK_GL
Hal Canaryac7f23c2018-11-26 14:07:41 -0500168 #ifndef SK_BUILD_FOR_ANDROID
169 SkQP::SkiaBackend::kGL, // Used for testing on desktop machines.
170 #endif
171 SkQP::SkiaBackend::kGLES,
Hal Canarya67523a2019-08-28 09:10:32 -0400172 #endif // SK_GL
Hal Canaryac7f23c2018-11-26 14:07:41 -0500173 #ifdef SK_VULKAN
174 SkQP::SkiaBackend::kVulkan,
175 #endif
176 };
177 for (SkQP::SkiaBackend backend : backends) {
178 std::unique_ptr<sk_gpu_test::TestContext> testCtx = make_test_context(backend);
179 if (testCtx) {
180 testCtx->makeCurrent();
181 if (nullptr != testCtx->makeGrContext(context_options())) {
182 result.push_back(backend);
183 }
184 }
185 }
186 SkASSERT_RELEASE(result.size() > 0);
187 return result;
188}
189
190static void print_backend_info(const char* dstPath,
191 const std::vector<SkQP::SkiaBackend>& backends) {
192#ifdef SK_ENABLE_DUMP_GPU
193 SkFILEWStream out(dstPath);
194 out.writeText("[\n");
195 for (SkQP::SkiaBackend backend : backends) {
196 if (std::unique_ptr<sk_gpu_test::TestContext> testCtx = make_test_context(backend)) {
197 testCtx->makeCurrent();
198 if (sk_sp<GrContext> ctx = testCtx->makeGrContext(context_options())) {
Brian Salomonec22b1a2019-08-09 09:41:48 -0400199 SkString info = ctx->dump();
Hal Canaryac7f23c2018-11-26 14:07:41 -0500200 // remove null
201 out.write(info.c_str(), info.size());
202 out.writeText(",\n");
203 }
204 }
205 }
206 out.writeText("]\n");
207#endif
208}
209
210static void encode_png(const SkBitmap& src, const std::string& dst) {
211 SkFILEWStream wStream(dst.c_str());
212 SkPngEncoder::Options options;
213 bool success = wStream.isValid() && SkPngEncoder::Encode(&wStream, src.pixmap(), options);
214 SkASSERT_RELEASE(success);
215}
216
217static void write_to_file(const sk_sp<SkData>& src, const std::string& dst) {
218 SkFILEWStream wStream(dst.c_str());
219 bool success = wStream.isValid() && wStream.write(src->data(), src->size());
220 SkASSERT_RELEASE(success);
221}
222
223////////////////////////////////////////////////////////////////////////////////
224
225const char* SkQP::GetBackendName(SkQP::SkiaBackend b) {
226 switch (b) {
227 case SkQP::SkiaBackend::kGL: return "gl";
228 case SkQP::SkiaBackend::kGLES: return "gles";
229 case SkQP::SkiaBackend::kVulkan: return "vk";
230 }
231 return "";
232}
233
234std::string SkQP::GetGMName(SkQP::GMFactory f) {
Hal Canaryedda5652019-08-05 10:28:09 -0400235 std::unique_ptr<skiagm::GM> gm(f ? f() : nullptr);
Hal Canaryac7f23c2018-11-26 14:07:41 -0500236 return std::string(gm ? gm->getName() : "");
237}
238
239const char* SkQP::GetUnitTestName(SkQP::UnitTest t) { return t->name; }
240
241SkQP::SkQP() {}
242
243SkQP::~SkQP() {}
244
245void SkQP::init(SkQPAssetManager* am, const char* reportDirectory) {
246 SkASSERT_RELEASE(!fAssetManager);
247 SkASSERT_RELEASE(am);
248 fAssetManager = am;
249 fReportDirectory = reportDirectory;
250
251 SkGraphics::Init();
Mike Kleinea3f0142019-03-20 11:12:10 -0500252 gSkFontMgr_DefaultFactory = &ToolUtils::MakePortableFontMgr;
Hal Canaryac7f23c2018-11-26 14:07:41 -0500253
254 /* If the file "skqp/rendertests.txt" does not exist or is empty, run all
255 render tests. Otherwise only run tests mentioned in that file. */
256 get_render_tests(fAssetManager, &fGMs, &fGMThresholds);
257 /* If the file "skqp/unittests.txt" does not exist or is empty, run all gpu
258 unit tests. Otherwise only run tests mentioned in that file. */
259 get_unit_tests(fAssetManager, &fUnitTests);
260 fSupportedBackends = get_backends();
261
262 print_backend_info((fReportDirectory + "/grdump.txt").c_str(), fSupportedBackends);
263}
264
265std::tuple<SkQP::RenderOutcome, std::string> SkQP::evaluateGM(SkQP::SkiaBackend backend,
266 SkQP::GMFactory gmFact) {
267 SkASSERT_RELEASE(fAssetManager);
268 static constexpr SkQP::RenderOutcome kError = {INT_MAX, INT_MAX, INT64_MAX};
269 static constexpr SkQP::RenderOutcome kPass = {0, 0, 0};
270
Hal Canary70441fa2019-02-08 13:43:26 -0500271 std::unique_ptr<sk_gpu_test::TestContext> testCtx = make_test_context(backend);
272 if (!testCtx) {
273 return std::make_tuple(kError, "Skia Failure: test context");
274 }
275 testCtx->makeCurrent();
276
Hal Canaryac7f23c2018-11-26 14:07:41 -0500277 SkASSERT(gmFact);
Hal Canaryedda5652019-08-05 10:28:09 -0400278 std::unique_ptr<skiagm::GM> gm(gmFact());
Hal Canaryac7f23c2018-11-26 14:07:41 -0500279 SkASSERT(gm);
280 const char* const name = gm->getName();
281 const SkISize size = gm->getISize();
282 const int w = size.width();
283 const int h = size.height();
284 const SkImageInfo info =
285 SkImageInfo::Make(w, h, skqp::kColorType, kPremul_SkAlphaType, nullptr);
286 const SkSurfaceProps props(0, SkSurfaceProps::kLegacyFontHost_InitType);
287
Hal Canaryac7f23c2018-11-26 14:07:41 -0500288 sk_sp<SkSurface> surf = SkSurface::MakeRenderTarget(
289 testCtx->makeGrContext(context_options(gm.get())).get(),
290 SkBudgeted::kNo, info, 0, &props);
291 if (!surf) {
292 return std::make_tuple(kError, "Skia Failure: gr-context");
293 }
294 gm->draw(surf->getCanvas());
295
296 SkBitmap image;
297 image.allocPixels(SkImageInfo::Make(w, h, skqp::kColorType, skqp::kAlphaType));
298
299 // SkColorTypeBytesPerPixel should be constexpr, but is not.
300 SkASSERT(SkColorTypeBytesPerPixel(skqp::kColorType) == sizeof(uint32_t));
301 // Call readPixels because we need to compare pixels.
302 if (!surf->readPixels(image.pixmap(), 0, 0)) {
303 return std::make_tuple(kError, "Skia Failure: read pixels");
304 }
305 int64_t passingThreshold = fGMThresholds.empty() ? -1 : fGMThresholds[std::string(name)];
306
307 if (-1 == passingThreshold) {
308 return std::make_tuple(kPass, "");
309 }
310 skqp::ModelResult modelResult =
311 skqp::CheckAgainstModel(name, image.pixmap(), fAssetManager);
312
313 if (!modelResult.fErrorString.empty()) {
314 return std::make_tuple(kError, std::move(modelResult.fErrorString));
315 }
316 fRenderResults.push_back(SkQP::RenderResult{backend, gmFact, modelResult.fOutcome});
317 if (modelResult.fOutcome.fMaxError <= passingThreshold) {
318 return std::make_tuple(kPass, "");
319 }
320 std::string imagesDirectory = fReportDirectory + "/" IMAGES_DIRECTORY_PATH;
321 if (!sk_mkdir(imagesDirectory.c_str())) {
322 SkDebugf("ERROR: sk_mkdir('%s');\n", imagesDirectory.c_str());
323 return std::make_tuple(modelResult.fOutcome, "");
324 }
325 std::ostringstream tmp;
326 tmp << imagesDirectory << '/' << SkQP::GetBackendName(backend) << '_' << name << '_';
327 std::string imagesPathPrefix1 = tmp.str();
328 tmp = std::ostringstream();
329 tmp << imagesDirectory << '/' << PATH_MODEL << '_' << name << '_';
330 std::string imagesPathPrefix2 = tmp.str();
331 encode_png(image, imagesPathPrefix1 + PATH_IMG_PNG);
332 encode_png(modelResult.fErrors, imagesPathPrefix1 + PATH_ERR_PNG);
333 write_to_file(modelResult.fMaxPng, imagesPathPrefix2 + PATH_MAX_PNG);
334 write_to_file(modelResult.fMinPng, imagesPathPrefix2 + PATH_MIN_PNG);
335 return std::make_tuple(modelResult.fOutcome, "");
336}
337
338std::vector<std::string> SkQP::executeTest(SkQP::UnitTest test) {
339 SkASSERT_RELEASE(fAssetManager);
340 struct : public skiatest::Reporter {
341 std::vector<std::string> fErrors;
342 void reportFailed(const skiatest::Failure& failure) override {
343 SkString desc = failure.toString();
344 fErrors.push_back(std::string(desc.c_str(), desc.size()));
345 }
346 } r;
347 GrContextOptions options;
348 options.fDisableDriverCorrectnessWorkarounds = true;
349 if (test->fContextOptionsProc) {
350 test->fContextOptionsProc(&options);
351 }
352 test->proc(&r, options);
353 fUnitTestResults.push_back(UnitTestResult{test, r.fErrors});
354 return r.fErrors;
355}
356
357////////////////////////////////////////////////////////////////////////////////
358
359static constexpr char kDocHead[] =
360 "<!doctype html>\n"
361 "<html lang=\"en\">\n"
362 "<head>\n"
363 "<meta charset=\"UTF-8\">\n"
364 "<title>SkQP Report</title>\n"
365 "<style>\n"
366 "img { max-width:48%; border:1px green solid;\n"
367 " image-rendering: pixelated;\n"
368 " background-image:url('"
369 "AAANSUhEUgAAABAAAAAQCAAAAAA6mKC9AAAAAXNSR0IArs4c6QAAAAJiS0dEAP+H"
370 "j8y/AAAACXBIWXMAAA7DAAAOwwHHb6hkAAAAB3RJTUUH3gUBEi4DGRAQYgAAAB1J"
371 "REFUGNNjfMoAAVJQmokBDdBHgPE/lPFsYN0BABdaAwN6tehMAAAAAElFTkSuQmCC"
372 "'); }\n"
373 "</style>\n"
374 "<script>\n"
375 "function ce(t) { return document.createElement(t); }\n"
376 "function ct(n) { return document.createTextNode(n); }\n"
377 "function ac(u,v) { return u.appendChild(v); }\n"
378 "function br(u) { ac(u, ce(\"br\")); }\n"
379 "function ma(s, c) { var a = ce(\"a\"); a.href = s; ac(a, c); return a; }\n"
380 "function f(backend, gm, e1, e2, e3) {\n"
381 " var b = ce(\"div\");\n"
382 " var x = ce(\"h2\");\n"
383 " var t = backend + \"_\" + gm;\n"
384 " ac(x, ct(t));\n"
385 " ac(b, x);\n"
386 " ac(b, ct(\"backend: \" + backend));\n"
387 " br(b);\n"
388 " ac(b, ct(\"gm name: \" + gm));\n"
389 " br(b);\n"
390 " ac(b, ct(\"maximum error: \" + e1));\n"
391 " br(b);\n"
392 " ac(b, ct(\"bad pixel counts: \" + e2));\n"
393 " br(b);\n"
394 " ac(b, ct(\"total error: \" + e3));\n"
395 " br(b);\n"
396 " var q = \"" IMAGES_DIRECTORY_PATH "/\" + backend + \"_\" + gm + \"_\";\n"
397 " var p = \"" IMAGES_DIRECTORY_PATH "/" PATH_MODEL "_\" + gm + \"_\";\n"
398 " var i = ce(\"img\");\n"
399 " i.src = q + \"" PATH_IMG_PNG "\";\n"
400 " i.alt = \"img\";\n"
401 " ac(b, ma(i.src, i));\n"
402 " i = ce(\"img\");\n"
403 " i.src = q + \"" PATH_ERR_PNG "\";\n"
404 " i.alt = \"err\";\n"
405 " ac(b, ma(i.src, i));\n"
406 " br(b);\n"
407 " ac(b, ct(\"Expectation: \"));\n"
408 " ac(b, ma(p + \"" PATH_MAX_PNG "\", ct(\"max\")));\n"
409 " ac(b, ct(\" | \"));\n"
410 " ac(b, ma(p + \"" PATH_MIN_PNG "\", ct(\"min\")));\n"
411 " ac(b, ce(\"hr\"));\n"
412 " b.id = backend + \":\" + gm;\n"
413 " ac(document.body, b);\n"
414 " l = ce(\"li\");\n"
415 " ac(l, ct(\"[\" + e3 + \"] \"));\n"
416 " ac(l, ma(\"#\" + backend +\":\"+ gm , ct(t)));\n"
417 " ac(document.getElementById(\"toc\"), l);\n"
418 "}\n"
419 "function main() {\n";
420
421static constexpr char kDocMiddle[] =
422 "}\n"
423 "</script>\n"
424 "</head>\n"
425 "<body onload=\"main()\">\n"
426 "<h1>SkQP Report</h1>\n";
427
428static constexpr char kDocTail[] =
429 "<ul id=\"toc\"></ul>\n"
430 "<hr>\n"
431 "<p>Left image: test result<br>\n"
432 "Right image: errors (white = no error, black = smallest error, red = biggest error; "
433 "other errors are a color between black and red.)</p>\n"
434 "<hr>\n"
435 "</body>\n"
436 "</html>\n";
437
438template <typename T>
439inline void write(SkWStream* wStream, const T& text) {
440 wStream->write(text.c_str(), text.size());
441}
442
443void SkQP::makeReport() {
444 SkASSERT_RELEASE(fAssetManager);
445 int glesErrorCount = 0, vkErrorCount = 0, gles = 0, vk = 0;
446
447 if (!sk_isdir(fReportDirectory.c_str())) {
448 SkDebugf("Report destination does not exist: '%s'\n", fReportDirectory.c_str());
449 return;
450 }
451 SkFILEWStream csvOut(SkOSPath::Join(fReportDirectory.c_str(), kRenderTestCSVReport).c_str());
452 SkFILEWStream htmOut(SkOSPath::Join(fReportDirectory.c_str(), kRenderTestReportPath).c_str());
453 SkASSERT_RELEASE(csvOut.isValid() && htmOut.isValid());
454 htmOut.writeText(kDocHead);
455 for (const SkQP::RenderResult& run : fRenderResults) {
456 switch (run.fBackend) {
457 case SkQP::SkiaBackend::kGLES: ++gles; break;
458 case SkQP::SkiaBackend::kVulkan: ++vk; break;
459 default: break;
460 }
461 const char* backendName = SkQP::GetBackendName(run.fBackend);
462 std::string gmName = SkQP::GetGMName(run.fGM);
Hal Canary44861f92019-03-01 10:57:04 -0500463 const SkQP::RenderOutcome& outcome = run.fOutcome;
Hal Canaryac7f23c2018-11-26 14:07:41 -0500464 auto str = SkStringPrintf("\"%s\",\"%s\",%d,%d,%" PRId64, backendName, gmName.c_str(),
465 outcome.fMaxError, outcome.fBadPixelCount, outcome.fTotalError);
466 write(&csvOut, SkStringPrintf("%s\n", str.c_str()));
467
468 int64_t passingThreshold = fGMThresholds.empty() ? 0 : fGMThresholds[gmName];
469 if (passingThreshold == -1 || outcome.fMaxError <= passingThreshold) {
470 continue;
471 }
472 write(&htmOut, SkStringPrintf(" f(%s);\n", str.c_str()));
473 switch (run.fBackend) {
474 case SkQP::SkiaBackend::kGLES: ++glesErrorCount; break;
475 case SkQP::SkiaBackend::kVulkan: ++vkErrorCount; break;
476 default: break;
477 }
478 }
479 htmOut.writeText(kDocMiddle);
480 write(&htmOut, SkStringPrintf("<p>gles errors: %d (of %d)</br>\n"
481 "vk errors: %d (of %d)</p>\n",
482 glesErrorCount, gles, vkErrorCount, vk));
483 htmOut.writeText(kDocTail);
484 SkFILEWStream unitOut(SkOSPath::Join(fReportDirectory.c_str(), kUnitTestReportPath).c_str());
485 SkASSERT_RELEASE(unitOut.isValid());
486 for (const SkQP::UnitTestResult& result : fUnitTestResults) {
487 unitOut.writeText(GetUnitTestName(result.fUnitTest));
488 if (result.fErrors.empty()) {
489 unitOut.writeText(" PASSED\n* * *\n");
490 } else {
491 write(&unitOut, SkStringPrintf(" FAILED (%u errors)\n", result.fErrors.size()));
492 for (const std::string& err : result.fErrors) {
493 write(&unitOut, err);
494 unitOut.newline();
495 }
496 unitOut.writeText("* * *\n");
497 }
498 }
499}