1 //
2 // Copyright 2015 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6 // angle_deqp_gtest:
7 //   dEQP and GoogleTest integration logic. Calls through to the random
8 //   order executor.
9 
10 #include <stdint.h>
11 #include <array>
12 #include <fstream>
13 
14 #include <gtest/gtest.h>
15 
16 #include "angle_deqp_libtester.h"
17 #include "common/Optional.h"
18 #include "common/angleutils.h"
19 #include "common/debug.h"
20 #include "common/platform.h"
21 #include "common/string_utils.h"
22 #include "common/system_utils.h"
23 #include "platform/PlatformMethods.h"
24 #include "tests/test_utils/runner/TestSuite.h"
25 #include "util/OSWindow.h"
26 #include "util/test_utils.h"
27 
28 namespace angle
29 {
30 namespace
31 {
32 #if !defined(NDEBUG)
33 constexpr bool kIsDebug = true;
34 #else
35 constexpr bool kIsDebug                = false;
36 #endif  // !defined(NDEBUG)
37 
38 bool gGlobalError = false;
39 bool gExpectError = false;
40 bool gVerbose     = false;
41 
42 // Set this to true temporarily to enable image logging in release. Useful for diagnosing errors.
43 bool gLogImages = kIsDebug;
44 
45 constexpr char kInfoTag[] = "*RESULT";
46 
HandlePlatformError(PlatformMethods * platform,const char * errorMessage)47 void HandlePlatformError(PlatformMethods *platform, const char *errorMessage)
48 {
49     if (!gExpectError)
50     {
51         FAIL() << errorMessage;
52     }
53     gGlobalError = true;
54 }
55 
DrawElementsToGoogleTestName(const std::string & dEQPName)56 std::string DrawElementsToGoogleTestName(const std::string &dEQPName)
57 {
58     std::string gTestName = dEQPName.substr(dEQPName.find('.') + 1);
59     std::replace(gTestName.begin(), gTestName.end(), '.', '_');
60 
61     // Occurs in some luminance tests
62     gTestName.erase(std::remove(gTestName.begin(), gTestName.end(), '-'), gTestName.end());
63     return gTestName;
64 }
65 
66 // Relative to the ANGLE root folder.
67 constexpr char kCTSRootPath[] = "third_party/VK-GL-CTS/src/";
68 constexpr char kSupportPath[] = "src/tests/deqp_support/";
69 
70 #define OPENGL_CTS_DIR(PATH) "external/openglcts/data/mustpass/gles/" PATH
71 
72 const char *gCaseListFiles[] = {
73     OPENGL_CTS_DIR("aosp_mustpass/master/gles2-master.txt"),
74     OPENGL_CTS_DIR("aosp_mustpass/master/gles3-master.txt"),
75     OPENGL_CTS_DIR("aosp_mustpass/master/gles31-master.txt"),
76     "/android/cts/master/egl-master.txt",
77     OPENGL_CTS_DIR("khronos_mustpass/master/gles2-khr-master.txt"),
78     OPENGL_CTS_DIR("khronos_mustpass/master/gles3-khr-master.txt"),
79     OPENGL_CTS_DIR("khronos_mustpass/master/gles31-khr-master.txt"),
80     OPENGL_CTS_DIR("khronos_mustpass/master/gles32-khr-master.txt"),
81     OPENGL_CTS_DIR("aosp_mustpass/master/gles3-rotate-landscape.txt"),
82     OPENGL_CTS_DIR("aosp_mustpass/master/gles3-rotate-reverse-portrait.txt"),
83     OPENGL_CTS_DIR("aosp_mustpass/master/gles3-rotate-reverse-landscape.txt"),
84     OPENGL_CTS_DIR("aosp_mustpass/master/gles31-rotate-landscape.txt"),
85     OPENGL_CTS_DIR("aosp_mustpass/master/gles31-rotate-reverse-portrait.txt"),
86     OPENGL_CTS_DIR("aosp_mustpass/master/gles31-rotate-reverse-landscape.txt"),
87 };
88 
89 #undef OPENGL_CTS_DIR
90 
91 const char *gTestExpectationsFiles[] = {
92     "deqp_gles2_test_expectations.txt",         "deqp_gles3_test_expectations.txt",
93     "deqp_gles31_test_expectations.txt",        "deqp_egl_test_expectations.txt",
94     "deqp_khr_gles2_test_expectations.txt",     "deqp_khr_gles3_test_expectations.txt",
95     "deqp_khr_gles31_test_expectations.txt",    "deqp_khr_gles32_test_expectations.txt",
96     "deqp_gles3_rotate_test_expectations.txt",  "deqp_gles3_rotate_test_expectations.txt",
97     "deqp_gles3_rotate_test_expectations.txt",  "deqp_gles31_rotate_test_expectations.txt",
98     "deqp_gles31_rotate_test_expectations.txt", "deqp_gles31_rotate_test_expectations.txt",
99 };
100 
101 using APIInfo = std::pair<const char *, GPUTestConfig::API>;
102 
103 constexpr APIInfo kEGLDisplayAPIs[] = {
104     {"angle-d3d9", GPUTestConfig::kAPID3D9},
105     {"angle-d3d11", GPUTestConfig::kAPID3D11},
106     {"angle-d3d11-ref", GPUTestConfig::kAPID3D11},
107     {"angle-gl", GPUTestConfig::kAPIGLDesktop},
108     {"angle-gles", GPUTestConfig::kAPIGLES},
109     {"angle-metal", GPUTestConfig::kAPIMetal},
110     {"angle-null", GPUTestConfig::kAPIUnknown},
111     {"angle-swiftshader", GPUTestConfig::kAPISwiftShader},
112     {"angle-vulkan", GPUTestConfig::kAPIVulkan},
113 };
114 
115 constexpr char kdEQPEGLString[]    = "--deqp-egl-display-type=";
116 constexpr char kANGLEEGLString[]   = "--use-angle=";
117 constexpr char kANGLEPreRotation[] = "--emulated-pre-rotation=";
118 constexpr char kdEQPCaseString[]   = "--deqp-case=";
119 constexpr char kVerboseString[]    = "--verbose";
120 constexpr char kRenderDocString[]  = "--renderdoc";
121 
122 std::array<char, 500> gCaseStringBuffer;
123 
124 // For angle_deqp_gles3*_rotateN_tests, default gPreRotation to N.
125 #if defined(ANGLE_DEQP_GLES3_ROTATE90_TESTS) || defined(ANGLE_DEQP_GLES31_ROTATE90_TESTS)
126 constexpr uint32_t kDefaultPreRotation = 90;
127 #elif defined(ANGLE_DEQP_GLES3_ROTATE180_TESTS) || defined(ANGLE_DEQP_GLES31_ROTATE180_TESTS)
128 constexpr uint32_t kDefaultPreRotation = 180;
129 #elif defined(ANGLE_DEQP_GLES3_ROTATE270_TESTS) || defined(ANGLE_DEQP_GLES31_ROTATE270_TESTS)
130 constexpr uint32_t kDefaultPreRotation = 270;
131 #else
132 constexpr uint32_t kDefaultPreRotation = 0;
133 #endif
134 
135 const APIInfo *gInitAPI = nullptr;
136 uint32_t gPreRotation   = kDefaultPreRotation;
137 
138 bool gEnableRenderDocCapture = false;
139 
140 constexpr const char gdEQPEGLConfigNameString[] = "--deqp-gl-config-name=";
141 constexpr const char gdEQPLogImagesString[]     = "--deqp-log-images=";
142 
143 // Default the config to RGBA8
144 const char *gEGLConfigName = "rgba8888d24s8";
145 
146 // Returns the default API for a platform.
GetDefaultAPIName()147 const char *GetDefaultAPIName()
148 {
149 #if defined(ANGLE_PLATFORM_ANDROID) || defined(ANGLE_PLATFORM_LINUX) || \
150     defined(ANGLE_PLATFORM_WINDOWS)
151     return "angle-vulkan";
152 #elif defined(ANGLE_PLATFORM_APPLE)
153     return "angle-gl";
154 #else
155 #    error Unknown platform.
156 #endif
157 }
158 
FindAPIInfo(const std::string & arg)159 const APIInfo *FindAPIInfo(const std::string &arg)
160 {
161     for (auto &displayAPI : kEGLDisplayAPIs)
162     {
163         if (arg == displayAPI.first)
164         {
165             return &displayAPI;
166         }
167     }
168     return nullptr;
169 }
170 
GetDefaultAPIInfo()171 const APIInfo *GetDefaultAPIInfo()
172 {
173     const APIInfo *defaultInfo = FindAPIInfo(GetDefaultAPIName());
174     ASSERT(defaultInfo);
175     return defaultInfo;
176 }
177 
GetTestStatLine(const std::string & key,const std::string & value)178 std::string GetTestStatLine(const std::string &key, const std::string &value)
179 {
180     return std::string(kInfoTag) + ": " + key + ": " + value + "\n";
181 }
182 
183 // During the CaseList initialization we cannot use the GTEST FAIL macro to quit the program because
184 // the initialization is called outside of tests the first time.
Die()185 void Die()
186 {
187     exit(EXIT_FAILURE);
188 }
189 
FindFileFromPath(const char * dirPath,const char * filePath)190 Optional<std::string> FindFileFromPath(const char *dirPath, const char *filePath)
191 {
192     std::stringstream strstr;
193     strstr << dirPath << filePath;
194     std::string path = strstr.str();
195 
196     constexpr size_t kMaxFoundPathLen = 1000;
197     char foundPath[kMaxFoundPathLen];
198     if (angle::FindTestDataPath(path.c_str(), foundPath, kMaxFoundPathLen))
199     {
200         return std::string(foundPath);
201     }
202 
203     return Optional<std::string>::Invalid();
204 }
205 
FindCaseListPath(size_t testModuleIndex)206 Optional<std::string> FindCaseListPath(size_t testModuleIndex)
207 {
208     return FindFileFromPath(kCTSRootPath, gCaseListFiles[testModuleIndex]);
209 }
210 
FindTestExpectationsPath(size_t testModuleIndex)211 Optional<std::string> FindTestExpectationsPath(size_t testModuleIndex)
212 {
213     return FindFileFromPath(kSupportPath, gTestExpectationsFiles[testModuleIndex]);
214 }
215 
216 class dEQPCaseList
217 {
218   public:
219     dEQPCaseList(size_t testModuleIndex);
220 
221     struct CaseInfo
222     {
CaseInfoangle::__anon68ac9fbc0111::dEQPCaseList::CaseInfo223         CaseInfo(const std::string &dEQPName, const std::string &gTestName, int expectation)
224             : mDEQPName(dEQPName), mGTestName(gTestName), mExpectation(expectation)
225         {}
226 
227         std::string mDEQPName;
228         std::string mGTestName;
229         int mExpectation;
230     };
231 
232     void initialize();
233 
getCaseInfo(size_t caseIndex) const234     const CaseInfo &getCaseInfo(size_t caseIndex) const
235     {
236         ASSERT(mInitialized);
237         ASSERT(caseIndex < mCaseInfoList.size());
238         return mCaseInfoList[caseIndex];
239     }
240 
numCases() const241     size_t numCases() const
242     {
243         ASSERT(mInitialized);
244         return mCaseInfoList.size();
245     }
246 
247   private:
248     std::vector<CaseInfo> mCaseInfoList;
249     size_t mTestModuleIndex;
250     bool mInitialized = false;
251 };
252 
dEQPCaseList(size_t testModuleIndex)253 dEQPCaseList::dEQPCaseList(size_t testModuleIndex) : mTestModuleIndex(testModuleIndex) {}
254 
initialize()255 void dEQPCaseList::initialize()
256 {
257     if (mInitialized)
258     {
259         return;
260     }
261 
262     mInitialized = true;
263 
264     Optional<std::string> caseListPath = FindCaseListPath(mTestModuleIndex);
265     if (!caseListPath.valid())
266     {
267         std::cerr << "Failed to find case list file." << std::endl;
268         Die();
269     }
270 
271     Optional<std::string> testExpectationsPath = FindTestExpectationsPath(mTestModuleIndex);
272     if (!testExpectationsPath.valid())
273     {
274         std::cerr << "Failed to find test expectations file." << std::endl;
275         Die();
276     }
277 
278     GPUTestConfig::API api = GetDefaultAPIInfo()->second;
279     // Set the API from the command line, or using the default platform API.
280     if (gInitAPI)
281     {
282         api = gInitAPI->second;
283     }
284 
285     GPUTestConfig testConfig = GPUTestConfig(api, gPreRotation);
286 
287 #if !defined(ANGLE_PLATFORM_ANDROID)
288     // Note: These prints mess up parsing of test list when running on Android.
289     std::cout << "Using test config with:" << std::endl;
290     for (uint32_t condition : testConfig.getConditions())
291     {
292         const char *name = GetConditionName(condition);
293         if (name != nullptr)
294         {
295             std::cout << "  " << name << std::endl;
296         }
297     }
298 #endif
299 
300     TestSuite *testSuite = TestSuite::GetInstance();
301 
302     if (!testSuite->loadTestExpectationsFromFileWithConfig(testConfig,
303                                                            testExpectationsPath.value()))
304     {
305         Die();
306     }
307 
308     std::ifstream caseListStream(caseListPath.value());
309     if (caseListStream.fail())
310     {
311         std::cerr << "Failed to load the case list." << std::endl;
312         Die();
313     }
314 
315     while (!caseListStream.eof())
316     {
317         std::string inString;
318         std::getline(caseListStream, inString);
319 
320         std::string dEQPName = TrimString(inString, kWhitespaceASCII);
321         if (dEQPName.empty())
322             continue;
323         std::string gTestName = DrawElementsToGoogleTestName(dEQPName);
324         if (gTestName.empty())
325             continue;
326 
327         int expectation = testSuite->getTestExpectation(dEQPName);
328         mCaseInfoList.push_back(CaseInfo(dEQPName, gTestName, expectation));
329     }
330 
331     if (testSuite->logAnyUnusedTestExpectations())
332     {
333         Die();
334     }
335 }
336 
337 template <size_t TestModuleIndex>
338 class dEQPTest : public testing::TestWithParam<size_t>
339 {
340   public:
GetTestingRange()341     static testing::internal::ParamGenerator<size_t> GetTestingRange()
342     {
343         return testing::Range<size_t>(0, GetCaseList().numCases());
344     }
345 
GetCaseGTestName(size_t caseIndex)346     static std::string GetCaseGTestName(size_t caseIndex)
347     {
348         const auto &caseInfo = GetCaseList().getCaseInfo(caseIndex);
349         return caseInfo.mGTestName;
350     }
351 
GetCaseList()352     static const dEQPCaseList &GetCaseList()
353     {
354         static dEQPCaseList sCaseList(TestModuleIndex);
355         sCaseList.initialize();
356         return sCaseList;
357     }
358 
359     static void SetUpTestCase();
360     static void TearDownTestCase();
361 
362   protected:
runTest() const363     void runTest() const
364     {
365         if (sTestExceptionCount > 1)
366         {
367             std::cout << "Too many exceptions, skipping all remaining tests." << std::endl;
368             return;
369         }
370 
371         const auto &caseInfo = GetCaseList().getCaseInfo(GetParam());
372         std::cout << caseInfo.mDEQPName << std::endl;
373 
374         // Tests that crash exit the harness before collecting the result. To tally the number of
375         // crashed tests we track how many tests we "tried" to run.
376         sTestCount++;
377 
378         if (caseInfo.mExpectation == GPUTestExpectationsParser::kGpuTestSkip ||
379             caseInfo.mExpectation == GPUTestExpectationsParser::kGpuTestTimeout)
380         {
381             sSkippedTestCount++;
382             std::cout << "Test skipped.\n";
383             return;
384         }
385 
386         gExpectError          = (caseInfo.mExpectation != GPUTestExpectationsParser::kGpuTestPass);
387         dEQPTestResult result = deqp_libtester_run(caseInfo.mDEQPName.c_str());
388 
389         bool testSucceeded = countTestResultAndReturnSuccess(result);
390 
391         // Check the global error flag for unexpected platform errors.
392         if (gGlobalError)
393         {
394             testSucceeded = false;
395             gGlobalError  = false;
396         }
397 
398         if (caseInfo.mExpectation == GPUTestExpectationsParser::kGpuTestPass)
399         {
400             EXPECT_TRUE(testSucceeded);
401 
402             if (!testSucceeded)
403             {
404                 sUnexpectedFailed.push_back(caseInfo.mDEQPName);
405             }
406         }
407         else if (testSucceeded)
408         {
409             std::cout << "Test expected to fail but passed!" << std::endl;
410             sUnexpectedPasses.push_back(caseInfo.mDEQPName);
411         }
412     }
413 
countTestResultAndReturnSuccess(dEQPTestResult result) const414     bool countTestResultAndReturnSuccess(dEQPTestResult result) const
415     {
416         switch (result)
417         {
418             case dEQPTestResult::Pass:
419                 sPassedTestCount++;
420                 return true;
421             case dEQPTestResult::Fail:
422                 sFailedTestCount++;
423                 return false;
424             case dEQPTestResult::NotSupported:
425                 sNotSupportedTestCount++;
426                 return true;
427             case dEQPTestResult::Exception:
428                 sTestExceptionCount++;
429                 return false;
430             default:
431                 std::cerr << "Unexpected test result code: " << static_cast<int>(result) << "\n";
432                 return false;
433         }
434     }
435 
PrintTestStats()436     static void PrintTestStats()
437     {
438         uint32_t crashedCount =
439             sTestCount - (sPassedTestCount + sFailedTestCount + sNotSupportedTestCount +
440                           sTestExceptionCount + sSkippedTestCount);
441 
442         std::cout << GetTestStatLine("Total", std::to_string(sTestCount));
443         std::cout << GetTestStatLine("Passed", std::to_string(sPassedTestCount));
444         std::cout << GetTestStatLine("Failed", std::to_string(sFailedTestCount));
445         std::cout << GetTestStatLine("Skipped", std::to_string(sSkippedTestCount));
446         std::cout << GetTestStatLine("Not Supported", std::to_string(sNotSupportedTestCount));
447         std::cout << GetTestStatLine("Exception", std::to_string(sTestExceptionCount));
448         std::cout << GetTestStatLine("Crashed", std::to_string(crashedCount));
449 
450         if (!sUnexpectedPasses.empty())
451         {
452             std::cout << GetTestStatLine("Unexpected Passed Count",
453                                          std::to_string(sUnexpectedPasses.size()));
454             for (const std::string &testName : sUnexpectedPasses)
455             {
456                 std::cout << GetTestStatLine("Unexpected Passed Tests", testName);
457             }
458         }
459 
460         if (!sUnexpectedFailed.empty())
461         {
462             std::cout << GetTestStatLine("Unexpected Failed Count",
463                                          std::to_string(sUnexpectedFailed.size()));
464             for (const std::string &testName : sUnexpectedFailed)
465             {
466                 std::cout << GetTestStatLine("Unexpected Failed Tests", testName);
467             }
468         }
469     }
470 
471     static uint32_t sTestCount;
472     static uint32_t sPassedTestCount;
473     static uint32_t sFailedTestCount;
474     static uint32_t sTestExceptionCount;
475     static uint32_t sNotSupportedTestCount;
476     static uint32_t sSkippedTestCount;
477 
478     static std::vector<std::string> sUnexpectedFailed;
479     static std::vector<std::string> sUnexpectedPasses;
480 };
481 
482 template <size_t TestModuleIndex>
483 uint32_t dEQPTest<TestModuleIndex>::sTestCount = 0;
484 template <size_t TestModuleIndex>
485 uint32_t dEQPTest<TestModuleIndex>::sPassedTestCount = 0;
486 template <size_t TestModuleIndex>
487 uint32_t dEQPTest<TestModuleIndex>::sFailedTestCount = 0;
488 template <size_t TestModuleIndex>
489 uint32_t dEQPTest<TestModuleIndex>::sTestExceptionCount = 0;
490 template <size_t TestModuleIndex>
491 uint32_t dEQPTest<TestModuleIndex>::sNotSupportedTestCount = 0;
492 template <size_t TestModuleIndex>
493 uint32_t dEQPTest<TestModuleIndex>::sSkippedTestCount = 0;
494 template <size_t TestModuleIndex>
495 std::vector<std::string> dEQPTest<TestModuleIndex>::sUnexpectedFailed;
496 template <size_t TestModuleIndex>
497 std::vector<std::string> dEQPTest<TestModuleIndex>::sUnexpectedPasses;
498 
499 // static
500 template <size_t TestModuleIndex>
SetUpTestCase()501 void dEQPTest<TestModuleIndex>::SetUpTestCase()
502 {
503     sPassedTestCount       = 0;
504     sFailedTestCount       = 0;
505     sNotSupportedTestCount = 0;
506     sTestExceptionCount    = 0;
507     sTestCount             = 0;
508     sSkippedTestCount      = 0;
509     sUnexpectedPasses.clear();
510     sUnexpectedFailed.clear();
511 
512     std::vector<const char *> argv;
513 
514     // Reserve one argument for the binary name.
515     argv.push_back("");
516 
517     // Add init api.
518     const char *targetApi    = gInitAPI ? gInitAPI->first : GetDefaultAPIName();
519     std::string apiArgString = std::string(kdEQPEGLString) + targetApi;
520     argv.push_back(apiArgString.c_str());
521 
522     // Add config name
523     const char *targetConfigName = gEGLConfigName;
524     std::string configArgString  = std::string(gdEQPEGLConfigNameString) + targetConfigName;
525     argv.push_back(configArgString.c_str());
526 
527     // Hide SwiftShader window to prevent a race with Xvfb causing hangs on test bots
528     if (gInitAPI && gInitAPI->second == GPUTestConfig::kAPISwiftShader)
529     {
530         argv.push_back("--deqp-visibility=hidden");
531     }
532 
533     TestSuite *testSuite = TestSuite::GetInstance();
534 
535     std::stringstream logNameStream;
536     logNameStream << "TestResults";
537     if (testSuite->getBatchId() != -1)
538     {
539         logNameStream << "-Batch" << std::setfill('0') << std::setw(3) << testSuite->getBatchId();
540     }
541     logNameStream << ".qpa";
542 
543     std::stringstream logArgStream;
544     logArgStream << "--deqp-log-filename=" << testSuite->addTestArtifact(logNameStream.str());
545 
546     std::string logNameString = logArgStream.str();
547     argv.push_back(logNameString.c_str());
548 
549     if (!gLogImages)
550     {
551         argv.push_back("--deqp-log-images=disable");
552     }
553 
554     // Flushing during multi-process execution punishes HDDs. http://anglebug.com/5157
555     if (testSuite->getBatchId() != -1)
556     {
557         argv.push_back("--deqp-log-flush=disable");
558     }
559 
560     // Init the platform.
561     if (!deqp_libtester_init_platform(static_cast<int>(argv.size()), argv.data(),
562                                       reinterpret_cast<void *>(&HandlePlatformError), gPreRotation,
563                                       gEnableRenderDocCapture))
564     {
565         std::cout << "Aborting test due to dEQP initialization error." << std::endl;
566         exit(1);
567     }
568 }
569 
570 // static
571 template <size_t TestModuleIndex>
TearDownTestCase()572 void dEQPTest<TestModuleIndex>::TearDownTestCase()
573 {
574     PrintTestStats();
575     deqp_libtester_shutdown_platform();
576 }
577 
578 #define ANGLE_INSTANTIATE_DEQP_TEST_CASE(API, N)                              \
579     class dEQP : public dEQPTest<N>                                           \
580     {};                                                                       \
581     TEST_P(dEQP, API) { runTest(); }                                          \
582                                                                               \
583     INSTANTIATE_TEST_SUITE_P(, dEQP, dEQP::GetTestingRange(),                 \
584                              [](const testing::TestParamInfo<size_t> &info) { \
585                                  return dEQP::GetCaseGTestName(info.param);   \
586                              })
587 
588 #ifdef ANGLE_DEQP_GLES2_TESTS
589 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES2, 0);
590 #endif
591 
592 #ifdef ANGLE_DEQP_GLES3_TESTS
593 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES3, 1);
594 #endif
595 
596 #ifdef ANGLE_DEQP_GLES31_TESTS
597 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES31, 2);
598 #endif
599 
600 #ifdef ANGLE_DEQP_EGL_TESTS
601 ANGLE_INSTANTIATE_DEQP_TEST_CASE(EGL, 3);
602 #endif
603 
604 #ifdef ANGLE_DEQP_KHR_GLES2_TESTS
605 ANGLE_INSTANTIATE_DEQP_TEST_CASE(KHR_GLES2, 4);
606 #endif
607 
608 #ifdef ANGLE_DEQP_KHR_GLES3_TESTS
609 ANGLE_INSTANTIATE_DEQP_TEST_CASE(KHR_GLES3, 5);
610 #endif
611 
612 #ifdef ANGLE_DEQP_KHR_GLES31_TESTS
613 ANGLE_INSTANTIATE_DEQP_TEST_CASE(KHR_GLES31, 6);
614 #endif
615 
616 #ifdef ANGLE_DEQP_KHR_GLES32_TESTS
617 ANGLE_INSTANTIATE_DEQP_TEST_CASE(KHR_GLES32, 7);
618 #endif
619 
620 #ifdef ANGLE_DEQP_GLES3_ROTATE90_TESTS
621 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES3_ROTATE90, 8);
622 #endif
623 
624 #ifdef ANGLE_DEQP_GLES3_ROTATE180_TESTS
625 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES3_ROTATE180, 9);
626 #endif
627 
628 #ifdef ANGLE_DEQP_GLES3_ROTATE270_TESTS
629 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES3_ROTATE270, 10);
630 #endif
631 
632 #ifdef ANGLE_DEQP_GLES31_ROTATE90_TESTS
633 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES31_ROTATE90, 11);
634 #endif
635 
636 #ifdef ANGLE_DEQP_GLES31_ROTATE180_TESTS
637 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES31_ROTATE180, 12);
638 #endif
639 
640 #ifdef ANGLE_DEQP_GLES31_ROTATE270_TESTS
641 ANGLE_INSTANTIATE_DEQP_TEST_CASE(GLES31_ROTATE270, 13);
642 #endif
643 
HandleDisplayType(const char * displayTypeString)644 void HandleDisplayType(const char *displayTypeString)
645 {
646     std::stringstream argStream;
647 
648     if (gInitAPI)
649     {
650         std::cout << "Cannot specify two EGL displays!" << std::endl;
651         exit(1);
652     }
653 
654     if (strncmp(displayTypeString, "angle-", strlen("angle-")) != 0)
655     {
656         argStream << "angle-";
657     }
658 
659     argStream << displayTypeString;
660     std::string arg = argStream.str();
661 
662     gInitAPI = FindAPIInfo(arg);
663 
664     if (!gInitAPI)
665     {
666         std::cout << "Unknown ANGLE back-end API: " << displayTypeString << std::endl;
667         exit(1);
668     }
669 }
670 
HandlePreRotation(const char * preRotationString)671 void HandlePreRotation(const char *preRotationString)
672 {
673     std::istringstream argStream(preRotationString);
674 
675     uint32_t preRotation = 0;
676     argStream >> preRotation;
677 
678     if (!argStream ||
679         (preRotation != 0 && preRotation != 90 && preRotation != 180 && preRotation != 270))
680     {
681         std::cout << "Invalid PreRotation '" << preRotationString
682                   << "'; must be either 0, 90, 180 or 270" << std::endl;
683         exit(1);
684     }
685 
686     gPreRotation = preRotation;
687 }
688 
HandleEGLConfigName(const char * configNameString)689 void HandleEGLConfigName(const char *configNameString)
690 {
691     gEGLConfigName = configNameString;
692 }
693 
694 // The --deqp-case flag takes a case expression that is parsed into a --gtest_filter. It converts
695 // the "dEQP" style names (functional.thing.*) into "GoogleTest" style names (functional_thing_*).
696 // Currently it does not handle multiple tests and multiple filters in different arguments.
HandleCaseName(const char * caseString,int * argc,int argIndex,char ** argv)697 void HandleCaseName(const char *caseString, int *argc, int argIndex, char **argv)
698 {
699     std::string googleTestName = DrawElementsToGoogleTestName(caseString);
700     gCaseStringBuffer.fill(0);
701     int bytesWritten = snprintf(gCaseStringBuffer.data(), gCaseStringBuffer.size() - 1,
702                                 "--gtest_filter=*%s", googleTestName.c_str());
703     if (bytesWritten <= 0 || static_cast<size_t>(bytesWritten) >= gCaseStringBuffer.size() - 1)
704     {
705         std::cout << "Error parsing test case string: " << caseString;
706         exit(1);
707     }
708 
709     argv[argIndex] = gCaseStringBuffer.data();
710 }
711 
HandleLogImages(const char * logImagesString)712 void HandleLogImages(const char *logImagesString)
713 {
714     if (strcmp(logImagesString, "enable") == 0)
715     {
716         gLogImages = true;
717     }
718     else if (strcmp(logImagesString, "disable") == 0)
719     {
720         gLogImages = false;
721     }
722     else
723     {
724         std::cout << "Error parsing log images setting. Use enable/disable.";
725         exit(1);
726     }
727 }
728 }  // anonymous namespace
729 
730 // Called from main() to process command-line arguments.
InitTestHarness(int * argc,char ** argv)731 void InitTestHarness(int *argc, char **argv)
732 {
733     int argIndex = 0;
734     while (argIndex < *argc)
735     {
736         if (strncmp(argv[argIndex], kdEQPEGLString, strlen(kdEQPEGLString)) == 0)
737         {
738             HandleDisplayType(argv[argIndex] + strlen(kdEQPEGLString));
739         }
740         else if (strncmp(argv[argIndex], kANGLEEGLString, strlen(kANGLEEGLString)) == 0)
741         {
742             HandleDisplayType(argv[argIndex] + strlen(kANGLEEGLString));
743         }
744         else if (strncmp(argv[argIndex], kANGLEPreRotation, strlen(kANGLEPreRotation)) == 0)
745         {
746             HandlePreRotation(argv[argIndex] + strlen(kANGLEPreRotation));
747         }
748         else if (strncmp(argv[argIndex], gdEQPEGLConfigNameString,
749                          strlen(gdEQPEGLConfigNameString)) == 0)
750         {
751             HandleEGLConfigName(argv[argIndex] + strlen(gdEQPEGLConfigNameString));
752         }
753         else if (strncmp(argv[argIndex], kdEQPCaseString, strlen(kdEQPCaseString)) == 0)
754         {
755             HandleCaseName(argv[argIndex] + strlen(kdEQPCaseString), argc, argIndex, argv);
756         }
757         else if (strncmp(argv[argIndex], kVerboseString, strlen(kVerboseString)) == 0 ||
758                  strcmp(argv[argIndex], "-v") == 0)
759         {
760             gVerbose = true;
761         }
762         else if (strncmp(argv[argIndex], gdEQPLogImagesString, strlen(gdEQPLogImagesString)) == 0)
763         {
764             HandleLogImages(argv[argIndex] + strlen(gdEQPLogImagesString));
765         }
766         else if (strncmp(argv[argIndex], kRenderDocString, strlen(kRenderDocString)) == 0)
767         {
768             gEnableRenderDocCapture = true;
769         }
770         argIndex++;
771     }
772 
773     GPUTestConfig::API api = GetDefaultAPIInfo()->second;
774     if (gInitAPI)
775     {
776         api = gInitAPI->second;
777     }
778     if (gPreRotation != 0 && api != GPUTestConfig::kAPIVulkan &&
779         api != GPUTestConfig::kAPISwiftShader)
780     {
781         std::cout << "PreRotation is only supported on Vulkan" << std::endl;
782         exit(1);
783     }
784 }
785 }  // namespace angle
786