• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 // Copyright 2019 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 
7 #include "GPUTestExpectationsParser.h"
8 
9 #include <stddef.h>
10 #include <stdint.h>
11 #include <string.h>
12 
13 #include "common/angleutils.h"
14 #include "common/debug.h"
15 #include "common/string_utils.h"
16 
17 namespace angle
18 {
19 
20 namespace
21 {
22 
23 enum LineParserStage
24 {
25     kLineParserBegin = 0,
26     kLineParserBugID,
27     kLineParserConfigs,
28     kLineParserColon,
29     kLineParserTestName,
30     kLineParserEqual,
31     kLineParserExpectations,
32 };
33 
34 enum Token
35 {
36     // os
37     kConfigWinXP = 0,
38     kConfigWinVista,
39     kConfigWin7,
40     kConfigWin8,
41     kConfigWin10,
42     kConfigWin,
43     kConfigMacLeopard,
44     kConfigMacSnowLeopard,
45     kConfigMacLion,
46     kConfigMacMountainLion,
47     kConfigMacMavericks,
48     kConfigMacYosemite,
49     kConfigMacElCapitan,
50     kConfigMacSierra,
51     kConfigMacHighSierra,
52     kConfigMacMojave,
53     kConfigMac,
54     kConfigIOS,
55     kConfigLinux,
56     kConfigChromeOS,
57     kConfigAndroid,
58     // gpu vendor
59     kConfigNVIDIA,
60     kConfigAMD,
61     kConfigIntel,
62     kConfigVMWare,
63     kConfigApple,
64     // build type
65     kConfigRelease,
66     kConfigDebug,
67     // ANGLE renderer
68     kConfigD3D9,
69     kConfigD3D11,
70     kConfigGLDesktop,
71     kConfigGLES,
72     kConfigVulkan,
73     kConfigSwiftShader,
74     kConfigMetal,
75     // Android devices
76     kConfigNexus5X,
77     kConfigPixel2,
78     kConfigPixel4,
79     // GPU devices
80     kConfigNVIDIAQuadroP400,
81     kConfigNVIDIAGTX1660,
82     // PreRotation
83     kConfigPreRotation,
84     kConfigPreRotation90,
85     kConfigPreRotation180,
86     kConfigPreRotation270,
87     // Sanitizers
88     kConfigNoSan,
89     kConfigASan,
90     kConfigTSan,
91     kConfigUBSan,
92     // expectation
93     kExpectationPass,
94     kExpectationFail,
95     kExpectationFlaky,
96     kExpectationTimeout,
97     kExpectationSkip,
98     // separator
99     kSeparatorColon,
100     kSeparatorEqual,
101 
102     kNumberOfExactMatchTokens,
103 
104     // others
105     kTokenComment,
106     kTokenWord,
107 
108     kNumberOfTokens,
109 };
110 
111 enum ErrorType
112 {
113     kErrorFileIO = 0,
114     kErrorIllegalEntry,
115     kErrorInvalidEntry,
116     kErrorEntryWithExpectationConflicts,
117     kErrorEntryWithDisallowedExpectation,
118     kErrorEntriesOverlap,
119 
120     kNumberOfErrors,
121 };
122 
123 struct TokenInfo
124 {
TokenInfoangle::__anonb7af6e1f0111::TokenInfo125     constexpr TokenInfo()
126         : name(nullptr),
127           condition(GPUTestConfig::kConditionNone),
128           expectation(GPUTestExpectationsParser::kGpuTestPass)
129     {}
130 
TokenInfoangle::__anonb7af6e1f0111::TokenInfo131     constexpr TokenInfo(const char *nameIn,
132                         GPUTestConfig::Condition conditionIn,
133                         GPUTestExpectationsParser::GPUTestExpectation expectationIn)
134         : name(nameIn), condition(conditionIn), expectation(expectationIn)
135     {}
136 
TokenInfoangle::__anonb7af6e1f0111::TokenInfo137     constexpr TokenInfo(const char *nameIn, GPUTestConfig::Condition conditionIn)
138         : TokenInfo(nameIn, conditionIn, GPUTestExpectationsParser::kGpuTestPass)
139     {}
140 
141     const char *name;
142     GPUTestConfig::Condition condition;
143     GPUTestExpectationsParser::GPUTestExpectation expectation;
144 };
145 
146 constexpr TokenInfo kTokenData[kNumberOfTokens] = {
147     {"xp", GPUTestConfig::kConditionWinXP},
148     {"vista", GPUTestConfig::kConditionWinVista},
149     {"win7", GPUTestConfig::kConditionWin7},
150     {"win8", GPUTestConfig::kConditionWin8},
151     {"win10", GPUTestConfig::kConditionWin10},
152     {"win", GPUTestConfig::kConditionWin},
153     {"leopard", GPUTestConfig::kConditionMacLeopard},
154     {"snowleopard", GPUTestConfig::kConditionMacSnowLeopard},
155     {"lion", GPUTestConfig::kConditionMacLion},
156     {"mountainlion", GPUTestConfig::kConditionMacMountainLion},
157     {"mavericks", GPUTestConfig::kConditionMacMavericks},
158     {"yosemite", GPUTestConfig::kConditionMacYosemite},
159     {"elcapitan", GPUTestConfig::kConditionMacElCapitan},
160     {"sierra", GPUTestConfig::kConditionMacSierra},
161     {"highsierra", GPUTestConfig::kConditionMacHighSierra},
162     {"mojave", GPUTestConfig::kConditionMacMojave},
163     {"mac", GPUTestConfig::kConditionMac},
164     {"ios", GPUTestConfig::kConditionIOS},
165     {"linux", GPUTestConfig::kConditionLinux},
166     {"chromeos", GPUTestConfig::kConditionNone},  // https://anglebug.com/3363 CrOS not supported
167     {"android", GPUTestConfig::kConditionAndroid},
168     {"nvidia", GPUTestConfig::kConditionNVIDIA},
169     {"amd", GPUTestConfig::kConditionAMD},
170     {"intel", GPUTestConfig::kConditionIntel},
171     {"vmware", GPUTestConfig::kConditionVMWare},
172     {"apple", GPUTestConfig::kConditionApple},
173     {"release", GPUTestConfig::kConditionRelease},
174     {"debug", GPUTestConfig::kConditionDebug},
175     {"d3d9", GPUTestConfig::kConditionD3D9},
176     {"d3d11", GPUTestConfig::kConditionD3D11},
177     {"opengl", GPUTestConfig::kConditionGLDesktop},
178     {"gles", GPUTestConfig::kConditionGLES},
179     {"vulkan", GPUTestConfig::kConditionVulkan},
180     {"swiftshader", GPUTestConfig::kConditionSwiftShader},
181     {"metal", GPUTestConfig::kConditionMetal},
182     {"nexus5x", GPUTestConfig::kConditionNexus5X},
183     {"pixel2orxl", GPUTestConfig::kConditionPixel2OrXL},
184     {"pixel4orxl", GPUTestConfig::kConditionPixel4OrXL},
185     {"quadrop400", GPUTestConfig::kConditionNVIDIAQuadroP400},
186     {"gtx1660", GPUTestConfig::kConditionNVIDIAGTX1660},
187     {"prerotation", GPUTestConfig::kConditionPreRotation},
188     {"prerotation90", GPUTestConfig::kConditionPreRotation90},
189     {"prerotation180", GPUTestConfig::kConditionPreRotation180},
190     {"prerotation270", GPUTestConfig::kConditionPreRotation270},
191     {"nosan", GPUTestConfig::kConditionNoSan},
192     {"asan", GPUTestConfig::kConditionASan},
193     {"tsan", GPUTestConfig::kConditionTSan},
194     {"ubsan", GPUTestConfig::kConditionUBSan},
195     {"pass", GPUTestConfig::kConditionNone, GPUTestExpectationsParser::kGpuTestPass},
196     {"fail", GPUTestConfig::kConditionNone, GPUTestExpectationsParser::kGpuTestFail},
197     {"flaky", GPUTestConfig::kConditionNone, GPUTestExpectationsParser::kGpuTestFlaky},
198     {"timeout", GPUTestConfig::kConditionNone, GPUTestExpectationsParser::kGpuTestTimeout},
199     {"skip", GPUTestConfig::kConditionNone, GPUTestExpectationsParser::kGpuTestSkip},
200     {":", GPUTestConfig::kConditionNone},  // kSeparatorColon
201     {"=", GPUTestConfig::kConditionNone},  // kSeparatorEqual
202     {},                                    // kNumberOfExactMatchTokens
203     {},                                    // kTokenComment
204     {},                                    // kTokenWord
205 };
206 
207 const char *kErrorMessage[kNumberOfErrors] = {
208     "file IO failed",
209     "entry with wrong format",
210     "entry invalid, likely unimplemented modifiers",
211     "entry with expectation modifier conflicts",
212     "entry with unsupported expectation",
213     "two entries' configs overlap",
214 };
215 
StartsWithASCII(const std::string & str,const std::string & search,bool caseSensitive)216 inline bool StartsWithASCII(const std::string &str, const std::string &search, bool caseSensitive)
217 {
218     ASSERT(!caseSensitive);
219     return str.compare(0, search.length(), search) == 0;
220 }
221 
222 template <class Char>
ToLowerASCII(Char c)223 inline Char ToLowerASCII(Char c)
224 {
225     return (c >= 'A' && c <= 'Z') ? (c + ('a' - 'A')) : c;
226 }
227 
228 template <typename Iter>
DoLowerCaseEqualsASCII(Iter a_begin,Iter a_end,const char * b)229 inline bool DoLowerCaseEqualsASCII(Iter a_begin, Iter a_end, const char *b)
230 {
231     for (Iter it = a_begin; it != a_end; ++it, ++b)
232     {
233         if (!*b || ToLowerASCII(*it) != *b)
234             return false;
235     }
236     return *b == 0;
237 }
238 
LowerCaseEqualsASCII(const std::string & a,const char * b)239 inline bool LowerCaseEqualsASCII(const std::string &a, const char *b)
240 {
241     return DoLowerCaseEqualsASCII(a.begin(), a.end(), b);
242 }
243 
ParseToken(const std::string & word)244 inline Token ParseToken(const std::string &word)
245 {
246     if (StartsWithASCII(word, "//", false))
247         return kTokenComment;
248 
249     for (int32_t i = 0; i < kNumberOfExactMatchTokens; ++i)
250     {
251         if (LowerCaseEqualsASCII(word, kTokenData[i].name))
252             return static_cast<Token>(i);
253     }
254     return kTokenWord;
255 }
256 
ConditionArrayIsSubset(const GPUTestConfig::ConditionArray & subset,const GPUTestConfig::ConditionArray & superset)257 bool ConditionArrayIsSubset(const GPUTestConfig::ConditionArray &subset,
258                             const GPUTestConfig::ConditionArray &superset)
259 {
260     for (size_t subsetCondition : subset)
261     {
262         bool foundCondition = false;
263         for (size_t supersetCondition : superset)
264         {
265             if (subsetCondition == supersetCondition)
266             {
267                 foundCondition = true;
268                 break;
269             }
270         }
271 
272         if (!foundCondition)
273         {
274             return false;
275         }
276     }
277 
278     return true;
279 }
280 
281 // If one array is completely contained within the other, then we say the conditions overlap.
ConditionsOverlap(const GPUTestConfig::ConditionArray & conditionsI,const GPUTestConfig::ConditionArray & conditionsJ)282 bool ConditionsOverlap(const GPUTestConfig::ConditionArray &conditionsI,
283                        const GPUTestConfig::ConditionArray &conditionsJ)
284 {
285     return ConditionArrayIsSubset(conditionsI, conditionsJ) ||
286            ConditionArrayIsSubset(conditionsJ, conditionsI);
287 }
288 }  // anonymous namespace
289 
GetConditionName(uint32_t condition)290 const char *GetConditionName(uint32_t condition)
291 {
292     if (condition == GPUTestConfig::kConditionNone)
293     {
294         return nullptr;
295     }
296 
297     for (const TokenInfo &info : kTokenData)
298     {
299         if (info.condition == condition)
300         {
301             // kConditionNone is used to tag tokens that aren't conditions, but this case has been
302             // handled above.
303             ASSERT(info.condition != GPUTestConfig::kConditionNone);
304             return info.name;
305         }
306     }
307 
308     return nullptr;
309 }
310 
GPUTestExpectationsParser()311 GPUTestExpectationsParser::GPUTestExpectationsParser()
312     : mExpectationsAllowMask(
313           GPUTestExpectationsParser::kGpuTestPass | GPUTestExpectationsParser::kGpuTestFail |
314           GPUTestExpectationsParser::kGpuTestFlaky | GPUTestExpectationsParser::kGpuTestTimeout |
315           GPUTestExpectationsParser::kGpuTestSkip)
316 {
317     // Some initial checks.
318     ASSERT((static_cast<unsigned int>(kNumberOfTokens)) ==
319            (sizeof(kTokenData) / sizeof(kTokenData[0])));
320     ASSERT((static_cast<unsigned int>(kNumberOfErrors)) ==
321            (sizeof(kErrorMessage) / sizeof(kErrorMessage[0])));
322 }
323 
324 GPUTestExpectationsParser::~GPUTestExpectationsParser() = default;
325 
loadTestExpectationsImpl(const GPUTestConfig * config,const std::string & data)326 bool GPUTestExpectationsParser::loadTestExpectationsImpl(const GPUTestConfig *config,
327                                                          const std::string &data)
328 {
329     mEntries.clear();
330     mErrorMessages.clear();
331 
332     std::vector<std::string> lines = SplitString(data, "\n", TRIM_WHITESPACE, SPLIT_WANT_ALL);
333     bool rt                        = true;
334     for (size_t i = 0; i < lines.size(); ++i)
335     {
336         if (!parseLine(config, lines[i], i + 1))
337             rt = false;
338     }
339     if (detectConflictsBetweenEntries())
340     {
341         mEntries.clear();
342         rt = false;
343     }
344 
345     return rt;
346 }
347 
loadTestExpectations(const GPUTestConfig & config,const std::string & data)348 bool GPUTestExpectationsParser::loadTestExpectations(const GPUTestConfig &config,
349                                                      const std::string &data)
350 {
351     return loadTestExpectationsImpl(&config, data);
352 }
353 
loadAllTestExpectations(const std::string & data)354 bool GPUTestExpectationsParser::loadAllTestExpectations(const std::string &data)
355 {
356     return loadTestExpectationsImpl(nullptr, data);
357 }
358 
loadTestExpectationsFromFileImpl(const GPUTestConfig * config,const std::string & path)359 bool GPUTestExpectationsParser::loadTestExpectationsFromFileImpl(const GPUTestConfig *config,
360                                                                  const std::string &path)
361 {
362     mEntries.clear();
363     mErrorMessages.clear();
364 
365     std::string data;
366     if (!ReadFileToString(path, &data))
367     {
368         mErrorMessages.push_back(kErrorMessage[kErrorFileIO]);
369         return false;
370     }
371     return loadTestExpectationsImpl(config, data);
372 }
373 
loadTestExpectationsFromFile(const GPUTestConfig & config,const std::string & path)374 bool GPUTestExpectationsParser::loadTestExpectationsFromFile(const GPUTestConfig &config,
375                                                              const std::string &path)
376 {
377     return loadTestExpectationsFromFileImpl(&config, path);
378 }
379 
loadAllTestExpectationsFromFile(const std::string & path)380 bool GPUTestExpectationsParser::loadAllTestExpectationsFromFile(const std::string &path)
381 {
382     return loadTestExpectationsFromFileImpl(nullptr, path);
383 }
384 
getTestExpectationImpl(const GPUTestConfig * config,const std::string & testName)385 int32_t GPUTestExpectationsParser::getTestExpectationImpl(const GPUTestConfig *config,
386                                                           const std::string &testName)
387 {
388     size_t maxExpectationLen            = 0;
389     GPUTestExpectationEntry *foundEntry = nullptr;
390     for (GPUTestExpectationEntry &entry : mEntries)
391     {
392         if (NamesMatchWithWildcard(entry.testName.c_str(), testName.c_str()))
393         {
394             size_t expectationLen = entry.testName.length();
395 
396             // Filter by condition first.
397             bool satisfiesConditions = true;
398             if (config)
399             {
400                 for (size_t condition : entry.conditions)
401                 {
402                     if (!config->getConditions()[condition])
403                     {
404                         satisfiesConditions = false;
405                         break;
406                     }
407                 }
408             }
409 
410             // The longest/most specific matching expectation overrides any others.
411             if (satisfiesConditions && expectationLen > maxExpectationLen)
412             {
413                 maxExpectationLen = expectationLen;
414                 foundEntry        = &entry;
415             }
416         }
417     }
418     if (foundEntry != nullptr)
419     {
420         foundEntry->used = true;
421         return foundEntry->testExpectation;
422     }
423     return kGpuTestPass;
424 }
425 
getTestExpectation(const std::string & testName)426 int32_t GPUTestExpectationsParser::getTestExpectation(const std::string &testName)
427 {
428     return getTestExpectationImpl(nullptr, testName);
429 }
430 
getTestExpectationWithConfig(const GPUTestConfig & config,const std::string & testName)431 int32_t GPUTestExpectationsParser::getTestExpectationWithConfig(const GPUTestConfig &config,
432                                                                 const std::string &testName)
433 {
434     return getTestExpectationImpl(&config, testName);
435 }
436 
getErrorMessages() const437 const std::vector<std::string> &GPUTestExpectationsParser::getErrorMessages() const
438 {
439     return mErrorMessages;
440 }
441 
getUnusedExpectationsMessages() const442 std::vector<std::string> GPUTestExpectationsParser::getUnusedExpectationsMessages() const
443 {
444     std::vector<std::string> messages;
445     std::vector<GPUTestExpectationsParser::GPUTestExpectationEntry> unusedExpectations =
446         getUnusedExpectations();
447     for (size_t i = 0; i < unusedExpectations.size(); ++i)
448     {
449         std::string message =
450             "Line " + ToString(unusedExpectations[i].lineNumber) + ": expectation was unused.";
451         messages.push_back(message);
452     }
453     return messages;
454 }
455 
parseLine(const GPUTestConfig * config,const std::string & lineData,size_t lineNumber)456 bool GPUTestExpectationsParser::parseLine(const GPUTestConfig *config,
457                                           const std::string &lineData,
458                                           size_t lineNumber)
459 {
460     std::vector<std::string> tokens =
461         SplitString(lineData, kWhitespaceASCII, KEEP_WHITESPACE, SPLIT_WANT_NONEMPTY);
462     int32_t stage = kLineParserBegin;
463     GPUTestExpectationEntry entry;
464     entry.lineNumber = lineNumber;
465     entry.used       = false;
466     bool skipLine    = false;
467     for (size_t i = 0; i < tokens.size() && !skipLine; ++i)
468     {
469         Token token = ParseToken(tokens[i]);
470         switch (token)
471         {
472             case kTokenComment:
473                 skipLine = true;
474                 break;
475             case kConfigWinXP:
476             case kConfigWinVista:
477             case kConfigWin7:
478             case kConfigWin8:
479             case kConfigWin10:
480             case kConfigWin:
481             case kConfigMacLeopard:
482             case kConfigMacSnowLeopard:
483             case kConfigMacLion:
484             case kConfigMacMountainLion:
485             case kConfigMacMavericks:
486             case kConfigMacYosemite:
487             case kConfigMacElCapitan:
488             case kConfigMacSierra:
489             case kConfigMacHighSierra:
490             case kConfigMacMojave:
491             case kConfigMac:
492             case kConfigIOS:
493             case kConfigLinux:
494             case kConfigChromeOS:
495             case kConfigAndroid:
496             case kConfigNVIDIA:
497             case kConfigAMD:
498             case kConfigIntel:
499             case kConfigVMWare:
500             case kConfigApple:
501             case kConfigRelease:
502             case kConfigDebug:
503             case kConfigD3D9:
504             case kConfigD3D11:
505             case kConfigGLDesktop:
506             case kConfigGLES:
507             case kConfigVulkan:
508             case kConfigSwiftShader:
509             case kConfigMetal:
510             case kConfigNexus5X:
511             case kConfigPixel2:
512             case kConfigPixel4:
513             case kConfigNVIDIAQuadroP400:
514             case kConfigNVIDIAGTX1660:
515             case kConfigPreRotation:
516             case kConfigPreRotation90:
517             case kConfigPreRotation180:
518             case kConfigPreRotation270:
519             case kConfigNoSan:
520             case kConfigASan:
521             case kConfigTSan:
522             case kConfigUBSan:
523                 // MODIFIERS, check each condition and add accordingly.
524                 if (stage != kLineParserConfigs && stage != kLineParserBugID)
525                 {
526                     pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
527                     return false;
528                 }
529                 {
530                     bool err = false;
531                     if (config)
532                     {
533                         if (!checkTokenCondition(*config, err, token, lineNumber))
534                         {
535                             skipLine = true;  // Move to the next line without adding this one.
536                         }
537                     }
538                     else
539                     {
540                         // Store the conditions for later comparison if we don't have a config.
541                         entry.conditions[kTokenData[token].condition] = true;
542                     }
543                     if (err)
544                     {
545                         return false;
546                     }
547                 }
548                 if (stage == kLineParserBugID)
549                 {
550                     stage++;
551                 }
552                 break;
553             case kSeparatorColon:
554                 // :
555                 // If there are no modifiers, move straight to separator colon
556                 if (stage == kLineParserBugID)
557                 {
558                     stage++;
559                 }
560                 if (stage != kLineParserConfigs)
561                 {
562                     pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
563                     return false;
564                 }
565                 stage++;
566                 break;
567             case kSeparatorEqual:
568                 // =
569                 if (stage != kLineParserTestName)
570                 {
571                     pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
572                     return false;
573                 }
574                 stage++;
575                 break;
576             case kTokenWord:
577                 // BUG_ID or TEST_NAME
578                 if (stage == kLineParserBegin)
579                 {
580                     // Bug ID is not used for anything; ignore it.
581                 }
582                 else if (stage == kLineParserColon)
583                 {
584                     entry.testName = tokens[i];
585                 }
586                 else
587                 {
588                     pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
589                     return false;
590                 }
591                 stage++;
592                 break;
593             case kExpectationPass:
594             case kExpectationFail:
595             case kExpectationFlaky:
596             case kExpectationTimeout:
597             case kExpectationSkip:
598                 // TEST_EXPECTATIONS
599                 if (stage != kLineParserEqual && stage != kLineParserExpectations)
600                 {
601                     pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
602                     return false;
603                 }
604                 if (entry.testExpectation != 0)
605                 {
606                     pushErrorMessage(kErrorMessage[kErrorEntryWithExpectationConflicts],
607                                      lineNumber);
608                     return false;
609                 }
610                 if ((mExpectationsAllowMask & kTokenData[token].expectation) == 0)
611                 {
612                     pushErrorMessage(kErrorMessage[kErrorEntryWithDisallowedExpectation],
613                                      lineNumber);
614                     return false;
615                 }
616                 entry.testExpectation = kTokenData[token].expectation;
617                 if (stage == kLineParserEqual)
618                     stage++;
619                 break;
620             default:
621                 ASSERT(false);
622                 break;
623         }
624     }
625     if (stage == kLineParserBegin || skipLine)
626     {
627         // The whole line is empty or all comments, or has been skipped to to a condition token.
628         return true;
629     }
630     if (stage == kLineParserExpectations)
631     {
632         mEntries.push_back(entry);
633         return true;
634     }
635     pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
636     return false;
637 }
638 
checkTokenCondition(const GPUTestConfig & config,bool & err,int32_t token,size_t lineNumber)639 bool GPUTestExpectationsParser::checkTokenCondition(const GPUTestConfig &config,
640                                                     bool &err,
641                                                     int32_t token,
642                                                     size_t lineNumber)
643 {
644     if (token >= kNumberOfTokens)
645     {
646         pushErrorMessage(kErrorMessage[kErrorIllegalEntry], lineNumber);
647         err = true;
648         return false;
649     }
650 
651     if (kTokenData[token].condition == GPUTestConfig::kConditionNone ||
652         kTokenData[token].condition >= GPUTestConfig::kNumberOfConditions)
653     {
654         pushErrorMessage(kErrorMessage[kErrorInvalidEntry], lineNumber);
655         // error on any unsupported conditions
656         err = true;
657         return false;
658     }
659     err = false;
660     return config.getConditions()[kTokenData[token].condition];
661 }
662 
detectConflictsBetweenEntries()663 bool GPUTestExpectationsParser::detectConflictsBetweenEntries()
664 {
665     bool rt = false;
666     for (size_t i = 0; i < mEntries.size(); ++i)
667     {
668         for (size_t j = i + 1; j < mEntries.size(); ++j)
669         {
670             const GPUTestExpectationEntry &entryI = mEntries[i];
671             const GPUTestExpectationEntry &entryJ = mEntries[j];
672             if (entryI.testName == entryJ.testName &&
673                 ConditionsOverlap(entryI.conditions, entryJ.conditions))
674             {
675                 pushErrorMessage(kErrorMessage[kErrorEntriesOverlap], entryI.lineNumber,
676                                  entryJ.lineNumber);
677                 rt = true;
678             }
679         }
680     }
681     return rt;
682 }
683 
684 std::vector<GPUTestExpectationsParser::GPUTestExpectationEntry>
getUnusedExpectations() const685 GPUTestExpectationsParser::getUnusedExpectations() const
686 {
687     std::vector<GPUTestExpectationsParser::GPUTestExpectationEntry> unusedExpectations;
688     for (size_t i = 0; i < mEntries.size(); ++i)
689     {
690         if (!mEntries[i].used)
691         {
692             unusedExpectations.push_back(mEntries[i]);
693         }
694     }
695     return unusedExpectations;
696 }
697 
pushErrorMessage(const std::string & message,size_t lineNumber)698 void GPUTestExpectationsParser::pushErrorMessage(const std::string &message, size_t lineNumber)
699 {
700     mErrorMessages.push_back("Line " + ToString(lineNumber) + " : " + message.c_str());
701 }
702 
pushErrorMessage(const std::string & message,size_t entry1LineNumber,size_t entry2LineNumber)703 void GPUTestExpectationsParser::pushErrorMessage(const std::string &message,
704                                                  size_t entry1LineNumber,
705                                                  size_t entry2LineNumber)
706 {
707     mErrorMessages.push_back("Line " + ToString(entry1LineNumber) + " and " +
708                              ToString(entry2LineNumber) + " : " + message.c_str());
709 }
710 
GPUTestExpectationEntry()711 GPUTestExpectationsParser::GPUTestExpectationEntry::GPUTestExpectationEntry()
712     : testExpectation(0), lineNumber(0)
713 {}
714 
715 }  // namespace angle
716