1 /*
2  * Copyright 2017 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 <iostream>
9 #include <sys/stat.h>
10 
11 #include "skqp.h"
12 
13 #include "Resources.h"
14 #include "SkData.h"
15 #include "SkOSFile.h"
16 
17 ////////////////////////////////////////////////////////////////////////////////
18 
19 namespace {
20 class StdAssetManager : public SkQPAssetManager {
21 public:
StdAssetManager(const char * p)22     StdAssetManager(const char* p) : fPrefix(p) {
23         SkASSERT(!fPrefix.empty());
24         //TODO(halcanary): does this need to be changed if I run SkQP in Windows?
25         fPrefix += "/";
26     }
open(const char * path)27     sk_sp<SkData> open(const char* path) override {
28         return SkData::MakeFromFileName((fPrefix + path).c_str());
29     }
30 private:
31     std::string fPrefix;
32 };
33 }
34 
35 static constexpr char kSkipUsage[] =
36     " TEST_MATCH_RULES:"
37     "    [~][^]substring[$] [...] of name to run.\n"
38     "    Multiple matches may be separated by spaces.\n"
39     "    ~ causes a matching name to always be skipped\n"
40     "    ^ requires the start of the name to match\n"
41     "    $ requires the end of the name to match\n"
42     "    ^ and $ requires an exact match\n"
43     "    If a name does not match any list entry,\n"
44     "    it is skipped unless some list entry starts with ~\n";
45 
should_skip(const char * const * rules,size_t count,const char * name)46 static bool should_skip(const char* const* rules, size_t count, const char* name) {
47     size_t testLen = strlen(name);
48     bool anyExclude = count == 0;
49     for (size_t i = 0; i < count; ++i) {
50         const char* matchName = rules[i];
51         size_t matchLen = strlen(matchName);
52         bool matchExclude, matchStart, matchEnd;
53         if ((matchExclude = matchName[0] == '~')) {
54             anyExclude = true;
55             matchName++;
56             matchLen--;
57         }
58         if ((matchStart = matchName[0] == '^')) {
59             matchName++;
60             matchLen--;
61         }
62         if ((matchEnd = matchName[matchLen - 1] == '$')) {
63             matchLen--;
64         }
65         if (matchStart ? (!matchEnd || matchLen == testLen)
66                 && strncmp(name, matchName, matchLen) == 0
67                 : matchEnd ? matchLen <= testLen
68                 && strncmp(name + testLen - matchLen, matchName, matchLen) == 0
69                 : strstr(name, matchName) != nullptr) {
70             return matchExclude;
71         }
72     }
73     return !anyExclude;
74 }
75 
main(int argc,char ** argv)76 int main(int argc, char** argv) {
77     if (argc < 3) {
78         std::cerr << "Usage:\n  " << argv[0]
79                   << " ASSET_DIRECTORY_PATH SKQP_REPORT_PATH [TEST_MATCH_RULES]\n"
80                   << kSkipUsage << '\n';
81         return 1;
82     }
83     SetResourcePath((std::string(argv[1]) + "/resources").c_str());
84     if (!sk_mkdir(argv[2])) {
85         std::cerr << "sk_mkdir(" << argv[2] << ") failed.\n";
86         return 2;
87     }
88     StdAssetManager mgr(argv[1]);
89     SkQP skqp;
90     skqp.init(&mgr, argv[2]);
91     int ret = 0;
92 
93     const char* const* matchRules = &argv[3];
94     size_t matchRulesCount = (size_t)(argc - 3);
95 
96     // Rendering Tests
97     std::ostream& out = std::cout;
98     for (auto backend : skqp.getSupportedBackends()) {
99         auto testPrefix = std::string(SkQP::GetBackendName(backend)) + "_";
100         for (auto gmFactory : skqp.getGMs()) {
101             auto testName = testPrefix + SkQP::GetGMName(gmFactory);
102             if (should_skip(matchRules, matchRulesCount, testName.c_str())) {
103                 continue;
104             }
105             out << "Starting: " << testName << std::endl;
106             SkQP::RenderOutcome outcome;
107             std::string except;
108 
109             std::tie(outcome, except) = skqp.evaluateGM(backend, gmFactory);
110             if (!except.empty()) {
111                 out << "ERROR:    " << testName << " (" << except << ")\n";
112                 ret = 1;
113             } else if (outcome.fMaxError != 0) {
114                 out << "FAILED:   " << testName << " (" << outcome.fMaxError << ")\n";
115                 ret = 1;
116             } else {
117                 out << "Passed:   " << testName << "\n";
118             }
119             out.flush();
120         }
121     }
122 
123     // Unit Tests
124     for (auto test : skqp.getUnitTests()) {
125         auto testName = std::string("unitTest_") +  SkQP::GetUnitTestName(test);
126         if (should_skip(matchRules, matchRulesCount, testName.c_str())) {
127             continue;
128         }
129         out << "Starting test: " << testName << std::endl;
130         std::vector<std::string> errors = skqp.executeTest(test);
131         if (!errors.empty()) {
132             out << "TEST FAILED (" << errors.size() << "): " << testName << "\n";
133             for (const std::string& error : errors) {
134                 out << error << "\n";
135             }
136             ret = 1;
137         } else {
138             out << "Test passed:   " << testName << "\n";
139         }
140         out.flush();
141     }
142     skqp.makeReport();
143 
144     return ret;
145 }
146