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