1 // Copyright (c) 2018 LunarG Inc.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #include <sstream>
16 #include <string>
17 
18 #include "gmock/gmock.h"
19 #include "test/unit_spirv.h"
20 #include "test/val/val_fixtures.h"
21 
22 namespace spvtools {
23 namespace val {
24 namespace {
25 
26 using ::testing::HasSubstr;
27 using ::testing::Not;
28 
29 using ValidateAdjacency = spvtest::ValidateBase<bool>;
30 
TEST_F(ValidateAdjacency,OpPhiBeginsModuleFail)31 TEST_F(ValidateAdjacency, OpPhiBeginsModuleFail) {
32   const std::string module = R"(
33 %result = OpPhi %bool %true %true_label %false %false_label
34 OpCapability Shader
35 OpMemoryModel Logical GLSL450
36 OpEntryPoint Fragment %main "main"
37 OpExecutionMode %main OriginUpperLeft
38 %void = OpTypeVoid
39 %bool = OpTypeBool
40 %true = OpConstantTrue %bool
41 %false = OpConstantFalse %bool
42 %func = OpTypeFunction %void
43 %main = OpFunction %void None %func
44 %main_entry = OpLabel
45 OpBranch %true_label
46 %true_label = OpLabel
47 OpBranch %false_label
48 %false_label = OpLabel
49 OpBranch %end_label
50 OpReturn
51 OpFunctionEnd
52 )";
53 
54   CompileSuccessfully(module);
55   EXPECT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
56   EXPECT_THAT(getDiagnosticString(),
57               HasSubstr("ID 1[%bool] has not been defined"));
58 }
59 
TEST_F(ValidateAdjacency,OpLoopMergeEndsModuleFail)60 TEST_F(ValidateAdjacency, OpLoopMergeEndsModuleFail) {
61   const std::string module = R"(
62 OpCapability Shader
63 OpMemoryModel Logical GLSL450
64 OpEntryPoint Fragment %main "main"
65 OpExecutionMode %main OriginUpperLeft
66 %void = OpTypeVoid
67 %func = OpTypeFunction %void
68 %main = OpFunction %void None %func
69 %main_entry = OpLabel
70 OpBranch %loop
71 %loop = OpLabel
72 OpLoopMerge %end %loop None
73 )";
74 
75   CompileSuccessfully(module);
76   EXPECT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
77   EXPECT_THAT(getDiagnosticString(),
78               HasSubstr("Missing OpFunctionEnd at end of module"));
79 }
80 
TEST_F(ValidateAdjacency,OpSelectionMergeEndsModuleFail)81 TEST_F(ValidateAdjacency, OpSelectionMergeEndsModuleFail) {
82   const std::string module = R"(
83 OpCapability Shader
84 OpMemoryModel Logical GLSL450
85 OpEntryPoint Fragment %main "main"
86 OpExecutionMode %main OriginUpperLeft
87 %void = OpTypeVoid
88 %func = OpTypeFunction %void
89 %main = OpFunction %void None %func
90 %main_entry = OpLabel
91 OpBranch %merge
92 %merge = OpLabel
93 OpSelectionMerge %merge None
94 )";
95 
96   CompileSuccessfully(module);
97   EXPECT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
98   EXPECT_THAT(getDiagnosticString(),
99               HasSubstr("Missing OpFunctionEnd at end of module"));
100 }
101 
GenerateShaderCode(const std::string & body,const std::string & capabilities_and_extensions="OpCapability Shader",const std::string & execution_model="Fragment")102 std::string GenerateShaderCode(
103     const std::string& body,
104     const std::string& capabilities_and_extensions = "OpCapability Shader",
105     const std::string& execution_model = "Fragment") {
106   std::ostringstream ss;
107   ss << capabilities_and_extensions << "\n";
108   ss << "OpMemoryModel Logical GLSL450\n";
109   ss << "OpEntryPoint " << execution_model << " %main \"main\"\n";
110   if (execution_model == "Fragment") {
111     ss << "OpExecutionMode %main OriginUpperLeft\n";
112   }
113 
114   ss << R"(
115 %string = OpString ""
116 %void = OpTypeVoid
117 %bool = OpTypeBool
118 %int = OpTypeInt 32 0
119 %true = OpConstantTrue %bool
120 %false = OpConstantFalse %bool
121 %zero = OpConstant %int 0
122 %int_1 = OpConstant %int 1
123 %func = OpTypeFunction %void
124 %func_int = OpTypePointer Function %int
125 %main = OpFunction %void None %func
126 %main_entry = OpLabel
127 )";
128 
129   ss << body;
130 
131   ss << R"(
132 OpReturn
133 OpFunctionEnd)";
134 
135   return ss.str();
136 }
137 
TEST_F(ValidateAdjacency,OpPhiPreceededByOpLabelSuccess)138 TEST_F(ValidateAdjacency, OpPhiPreceededByOpLabelSuccess) {
139   const std::string body = R"(
140 OpSelectionMerge %end_label None
141 OpBranchConditional %true %true_label %false_label
142 %true_label = OpLabel
143 OpBranch %end_label
144 %false_label = OpLabel
145 OpBranch %end_label
146 %end_label = OpLabel
147 OpLine %string 0 0
148 %result = OpPhi %bool %true %true_label %false %false_label
149 )";
150 
151   CompileSuccessfully(GenerateShaderCode(body));
152   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
153 }
154 
TEST_F(ValidateAdjacency,OpPhiPreceededByOpPhiSuccess)155 TEST_F(ValidateAdjacency, OpPhiPreceededByOpPhiSuccess) {
156   const std::string body = R"(
157 OpSelectionMerge %end_label None
158 OpBranchConditional %true %true_label %false_label
159 %true_label = OpLabel
160 OpBranch %end_label
161 %false_label = OpLabel
162 OpBranch %end_label
163 %end_label = OpLabel
164 %1 = OpPhi %bool %true %true_label %false %false_label
165 %2 = OpPhi %bool %true %true_label %false %false_label
166 )";
167 
168   CompileSuccessfully(GenerateShaderCode(body));
169   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
170 }
171 
TEST_F(ValidateAdjacency,OpPhiPreceededByOpLineSuccess)172 TEST_F(ValidateAdjacency, OpPhiPreceededByOpLineSuccess) {
173   const std::string body = R"(
174 OpSelectionMerge %end_label None
175 OpBranchConditional %true %true_label %false_label
176 %true_label = OpLabel
177 OpBranch %end_label
178 %false_label = OpLabel
179 OpBranch %end_label
180 %end_label = OpLabel
181 OpLine %string 0 0
182 %result = OpPhi %bool %true %true_label %false %false_label
183 )";
184 
185   CompileSuccessfully(GenerateShaderCode(body));
186   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
187 }
188 
TEST_F(ValidateAdjacency,OpPhiPreceededByBadOpFail)189 TEST_F(ValidateAdjacency, OpPhiPreceededByBadOpFail) {
190   const std::string body = R"(
191 OpSelectionMerge %end_label None
192 OpBranchConditional %true %true_label %false_label
193 %true_label = OpLabel
194 OpBranch %end_label
195 %false_label = OpLabel
196 OpBranch %end_label
197 %end_label = OpLabel
198 OpNop
199 %result = OpPhi %bool %true %true_label %false %false_label
200 )";
201 
202   CompileSuccessfully(GenerateShaderCode(body));
203   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
204   EXPECT_THAT(getDiagnosticString(),
205               HasSubstr("OpPhi must appear within a non-entry block before all "
206                         "non-OpPhi instructions"));
207 }
208 
TEST_F(ValidateAdjacency,OpPhiPreceededByOpLineAndBadOpFail)209 TEST_F(ValidateAdjacency, OpPhiPreceededByOpLineAndBadOpFail) {
210   const std::string body = R"(
211 OpSelectionMerge %end_label None
212 OpBranchConditional %true %true_label %false_label
213 %true_label = OpLabel
214 OpBranch %end_label
215 %false_label = OpLabel
216 OpBranch %end_label
217 %end_label = OpLabel
218 OpNop
219 OpLine %string 1 1
220 %result = OpPhi %bool %true %true_label %false %false_label
221 )";
222 
223   CompileSuccessfully(GenerateShaderCode(body));
224   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
225   EXPECT_THAT(getDiagnosticString(),
226               HasSubstr("OpPhi must appear within a non-entry block before all "
227                         "non-OpPhi instructions"));
228 }
229 
TEST_F(ValidateAdjacency,OpPhiFollowedByOpLineGood)230 TEST_F(ValidateAdjacency, OpPhiFollowedByOpLineGood) {
231   const std::string body = R"(
232 OpSelectionMerge %end_label None
233 OpBranchConditional %true %true_label %false_label
234 %true_label = OpLabel
235 OpBranch %end_label
236 %false_label = OpLabel
237 OpBranch %end_label
238 %end_label = OpLabel
239 %result = OpPhi %bool %true %true_label %false %false_label
240 OpLine %string 1 1
241 OpNop
242 OpNop
243 OpLine %string 2 1
244 OpNop
245 OpLine %string 3 1
246 )";
247 
248   CompileSuccessfully(GenerateShaderCode(body));
249   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
250 }
251 
TEST_F(ValidateAdjacency,OpPhiMultipleOpLineAndOpPhiFail)252 TEST_F(ValidateAdjacency, OpPhiMultipleOpLineAndOpPhiFail) {
253   const std::string body = R"(
254 OpSelectionMerge %end_label None
255 OpBranchConditional %true %true_label %false_label
256 %true_label = OpLabel
257 OpBranch %end_label
258 %false_label = OpLabel
259 OpBranch %end_label
260 %end_label = OpLabel
261 OpLine %string 1 1
262 %value = OpPhi %int %zero %true_label %int_1 %false_label
263 OpNop
264 OpLine %string 2 1
265 OpNop
266 OpLine %string 3 1
267 %result = OpPhi %bool %true %true_label %false %false_label
268 )";
269 
270   CompileSuccessfully(GenerateShaderCode(body));
271   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
272   EXPECT_THAT(getDiagnosticString(),
273               HasSubstr("OpPhi must appear within a non-entry block before all "
274                         "non-OpPhi instructions"));
275 }
276 
TEST_F(ValidateAdjacency,OpPhiMultipleOpLineAndOpPhiGood)277 TEST_F(ValidateAdjacency, OpPhiMultipleOpLineAndOpPhiGood) {
278   const std::string body = R"(
279 OpSelectionMerge %end_label None
280 OpBranchConditional %true %true_label %false_label
281 %true_label = OpLabel
282 OpBranch %end_label
283 %false_label = OpLabel
284 OpBranch %end_label
285 %end_label = OpLabel
286 OpLine %string 1 1
287 %value = OpPhi %int %zero %true_label %int_1 %false_label
288 OpLine %string 2 1
289 %result = OpPhi %bool %true %true_label %false %false_label
290 OpLine %string 3 1
291 OpNop
292 OpNop
293 OpLine %string 4 1
294 OpNop
295 )";
296 
297   CompileSuccessfully(GenerateShaderCode(body));
298   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
299 }
300 
TEST_F(ValidateAdjacency,OpPhiInEntryBlockBad)301 TEST_F(ValidateAdjacency, OpPhiInEntryBlockBad) {
302   const std::string body = R"(
303 OpLine %string 1 1
304 %value = OpPhi %int
305 OpLine %string 2 1
306 OpNop
307 OpLine %string 3 1
308 OpNop
309 )";
310 
311   CompileSuccessfully(GenerateShaderCode(body));
312   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
313   EXPECT_THAT(getDiagnosticString(),
314               HasSubstr("OpPhi must appear within a non-entry block before all "
315                         "non-OpPhi instructions"));
316 }
317 
TEST_F(ValidateAdjacency,NonSemanticBeforeOpPhiBad)318 TEST_F(ValidateAdjacency, NonSemanticBeforeOpPhiBad) {
319   const std::string body = R"(
320 OpSelectionMerge %end_label None
321 OpBranchConditional %true %true_label %false_label
322 %true_label = OpLabel
323 OpBranch %end_label
324 %false_label = OpLabel
325 OpBranch %end_label
326 %end_label = OpLabel
327 %placeholder = OpExtInst %void %extinst 123 %int_1
328 %result = OpPhi %bool %true %true_label %false %false_label
329 )";
330 
331   const std::string extra = R"(OpCapability Shader
332 OpExtension "SPV_KHR_non_semantic_info"
333 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
334 )";
335 
336   CompileSuccessfully(GenerateShaderCode(body, extra));
337   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
338   EXPECT_THAT(getDiagnosticString(),
339               HasSubstr("OpPhi must appear within a non-entry block before all "
340                         "non-OpPhi instructions"));
341 }
342 
TEST_F(ValidateAdjacency,NonSemanticBetweenOpPhiBad)343 TEST_F(ValidateAdjacency, NonSemanticBetweenOpPhiBad) {
344   const std::string body = R"(
345 OpSelectionMerge %end_label None
346 OpBranchConditional %true %true_label %false_label
347 %true_label = OpLabel
348 OpBranch %end_label
349 %false_label = OpLabel
350 OpBranch %end_label
351 %end_label = OpLabel
352 %result1 = OpPhi %bool %true %true_label %false %false_label
353 %placeholder = OpExtInst %void %extinst 123 %int_1
354 %result2 = OpPhi %bool %true %true_label %false %false_label
355 )";
356 
357   const std::string extra = R"(OpCapability Shader
358 OpExtension "SPV_KHR_non_semantic_info"
359 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
360 )";
361 
362   CompileSuccessfully(GenerateShaderCode(body, extra));
363   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
364   EXPECT_THAT(getDiagnosticString(),
365               HasSubstr("OpPhi must appear within a non-entry block before all "
366                         "non-OpPhi instructions"));
367 }
368 
TEST_F(ValidateAdjacency,NonSemanticAfterOpPhiGood)369 TEST_F(ValidateAdjacency, NonSemanticAfterOpPhiGood) {
370   const std::string body = R"(
371 OpSelectionMerge %end_label None
372 OpBranchConditional %true %true_label %false_label
373 %true_label = OpLabel
374 OpBranch %end_label
375 %false_label = OpLabel
376 OpBranch %end_label
377 %end_label = OpLabel
378 OpLine %string 0 0
379 %result = OpPhi %bool %true %true_label %false %false_label
380 %placeholder = OpExtInst %void %extinst 123 %int_1
381 )";
382 
383   const std::string extra = R"(OpCapability Shader
384 OpExtension "SPV_KHR_non_semantic_info"
385 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
386 )";
387 
388   CompileSuccessfully(GenerateShaderCode(body, extra));
389   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
390 }
391 
TEST_F(ValidateAdjacency,NonSemanticBeforeOpFunctionParameterBad)392 TEST_F(ValidateAdjacency, NonSemanticBeforeOpFunctionParameterBad) {
393   const std::string body = R"(
394 OpCapability Shader
395 OpExtension "SPV_KHR_non_semantic_info"
396 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
397 OpMemoryModel Logical GLSL450
398 OpEntryPoint Fragment %main "main"
399 OpExecutionMode %main OriginUpperLeft
400 
401 %string = OpString ""
402 %void = OpTypeVoid
403 %bool = OpTypeBool
404 %int = OpTypeInt 32 0
405 %true = OpConstantTrue %bool
406 %false = OpConstantFalse %bool
407 %zero = OpConstant %int 0
408 %int_1 = OpConstant %int 1
409 %func = OpTypeFunction %void
410 %func_int = OpTypePointer Function %int
411 %paramfunc_type = OpTypeFunction %void %int %int
412 
413 %paramfunc = OpFunction %void None %paramfunc_type
414 %placeholder = OpExtInst %void %extinst 123 %int_1
415 %a = OpFunctionParameter %int
416 %b = OpFunctionParameter %int
417 %paramfunc_entry = OpLabel
418 OpReturn
419 OpFunctionEnd
420 
421 %main = OpFunction %void None %func
422 %main_entry = OpLabel
423 OpReturn
424 OpFunctionEnd
425 )";
426 
427   CompileSuccessfully(body);
428   EXPECT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
429   EXPECT_THAT(getDiagnosticString(),
430               HasSubstr("Non-semantic OpExtInst within function definition "
431                         "must appear in a block"));
432 }
433 
TEST_F(ValidateAdjacency,NonSemanticBetweenOpFunctionParameterBad)434 TEST_F(ValidateAdjacency, NonSemanticBetweenOpFunctionParameterBad) {
435   const std::string body = R"(
436 OpCapability Shader
437 OpExtension "SPV_KHR_non_semantic_info"
438 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
439 OpMemoryModel Logical GLSL450
440 OpEntryPoint Fragment %main "main"
441 OpExecutionMode %main OriginUpperLeft
442 
443 %string = OpString ""
444 %void = OpTypeVoid
445 %bool = OpTypeBool
446 %int = OpTypeInt 32 0
447 %true = OpConstantTrue %bool
448 %false = OpConstantFalse %bool
449 %zero = OpConstant %int 0
450 %int_1 = OpConstant %int 1
451 %func = OpTypeFunction %void
452 %func_int = OpTypePointer Function %int
453 %paramfunc_type = OpTypeFunction %void %int %int
454 
455 %paramfunc = OpFunction %void None %paramfunc_type
456 %a = OpFunctionParameter %int
457 %placeholder = OpExtInst %void %extinst 123 %int_1
458 %b = OpFunctionParameter %int
459 %paramfunc_entry = OpLabel
460 OpReturn
461 OpFunctionEnd
462 
463 %main = OpFunction %void None %func
464 %main_entry = OpLabel
465 OpReturn
466 OpFunctionEnd
467 )";
468 
469   CompileSuccessfully(body);
470   EXPECT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
471   EXPECT_THAT(getDiagnosticString(),
472               HasSubstr("Non-semantic OpExtInst within function definition "
473                         "must appear in a block"));
474 }
475 
TEST_F(ValidateAdjacency,NonSemanticAfterOpFunctionParameterGood)476 TEST_F(ValidateAdjacency, NonSemanticAfterOpFunctionParameterGood) {
477   const std::string body = R"(
478 OpCapability Shader
479 OpExtension "SPV_KHR_non_semantic_info"
480 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
481 OpMemoryModel Logical GLSL450
482 OpEntryPoint Fragment %main "main"
483 OpExecutionMode %main OriginUpperLeft
484 
485 %string = OpString ""
486 %void = OpTypeVoid
487 %bool = OpTypeBool
488 %int = OpTypeInt 32 0
489 %true = OpConstantTrue %bool
490 %false = OpConstantFalse %bool
491 %zero = OpConstant %int 0
492 %int_1 = OpConstant %int 1
493 %func = OpTypeFunction %void
494 %func_int = OpTypePointer Function %int
495 %paramfunc_type = OpTypeFunction %void %int %int
496 
497 %paramfunc = OpFunction %void None %paramfunc_type
498 %a = OpFunctionParameter %int
499 %b = OpFunctionParameter %int
500 %paramfunc_entry = OpLabel
501 %placeholder = OpExtInst %void %extinst 123 %int_1
502 OpReturn
503 OpFunctionEnd
504 
505 %main = OpFunction %void None %func
506 %main_entry = OpLabel
507 OpReturn
508 OpFunctionEnd
509 )";
510 
511   CompileSuccessfully(body);
512   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
513 }
514 
TEST_F(ValidateAdjacency,NonSemanticBetweenFunctionsGood)515 TEST_F(ValidateAdjacency, NonSemanticBetweenFunctionsGood) {
516   const std::string body = R"(
517 OpCapability Shader
518 OpExtension "SPV_KHR_non_semantic_info"
519 %extinst = OpExtInstImport "NonSemantic.Testing.Set"
520 OpMemoryModel Logical GLSL450
521 OpEntryPoint Fragment %main "main"
522 OpExecutionMode %main OriginUpperLeft
523 
524 %string = OpString ""
525 %void = OpTypeVoid
526 %bool = OpTypeBool
527 %int = OpTypeInt 32 0
528 %true = OpConstantTrue %bool
529 %false = OpConstantFalse %bool
530 %zero = OpConstant %int 0
531 %int_1 = OpConstant %int 1
532 %func = OpTypeFunction %void
533 %func_int = OpTypePointer Function %int
534 %paramfunc_type = OpTypeFunction %void %int %int
535 
536 %paramfunc = OpFunction %void None %paramfunc_type
537 %a = OpFunctionParameter %int
538 %b = OpFunctionParameter %int
539 %paramfunc_entry = OpLabel
540 OpReturn
541 OpFunctionEnd
542 
543 %placeholder = OpExtInst %void %extinst 123 %int_1
544 
545 %main = OpFunction %void None %func
546 %main_entry = OpLabel
547 OpReturn
548 OpFunctionEnd
549 )";
550 
551   CompileSuccessfully(body);
552   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
553 }
554 
TEST_F(ValidateAdjacency,OpVariableInFunctionGood)555 TEST_F(ValidateAdjacency, OpVariableInFunctionGood) {
556   const std::string body = R"(
557 OpLine %string 1 1
558 %var = OpVariable %func_int Function
559 OpLine %string 2 1
560 OpNop
561 OpLine %string 3 1
562 OpNop
563 )";
564 
565   CompileSuccessfully(GenerateShaderCode(body));
566   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
567 }
568 
TEST_F(ValidateAdjacency,OpVariableInFunctionMultipleGood)569 TEST_F(ValidateAdjacency, OpVariableInFunctionMultipleGood) {
570   const std::string body = R"(
571 OpLine %string 1 1
572 %1 = OpVariable %func_int Function
573 OpLine %string 2 1
574 %2 = OpVariable %func_int Function
575 %3 = OpVariable %func_int Function
576 OpNop
577 OpLine %string 3 1
578 OpNop
579 )";
580 
581   CompileSuccessfully(GenerateShaderCode(body));
582   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
583 }
584 
TEST_F(ValidateAdjacency,OpVariableInFunctionBad)585 TEST_F(ValidateAdjacency, OpVariableInFunctionBad) {
586   const std::string body = R"(
587 %1 = OpUndef %int
588 %2 = OpVariable %func_int Function
589 )";
590 
591   CompileSuccessfully(GenerateShaderCode(body));
592   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
593   EXPECT_THAT(getDiagnosticString(),
594               HasSubstr("All OpVariable instructions in a function must be the "
595                         "first instructions"));
596 }
597 
TEST_F(ValidateAdjacency,OpVariableInFunctionMultipleBad)598 TEST_F(ValidateAdjacency, OpVariableInFunctionMultipleBad) {
599   const std::string body = R"(
600 OpNop
601 %1 = OpVariable %func_int Function
602 OpLine %string 1 1
603 %2 = OpVariable %func_int Function
604 OpNop
605 OpNop
606 OpLine %string 2 1
607 %3 = OpVariable %func_int Function
608 )";
609 
610   CompileSuccessfully(GenerateShaderCode(body));
611   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
612   EXPECT_THAT(getDiagnosticString(),
613               HasSubstr("All OpVariable instructions in a function must be the "
614                         "first instructions"));
615 }
616 
TEST_F(ValidateAdjacency,OpLoopMergePreceedsOpBranchSuccess)617 TEST_F(ValidateAdjacency, OpLoopMergePreceedsOpBranchSuccess) {
618   const std::string body = R"(
619 OpBranch %loop
620 %loop = OpLabel
621 OpLoopMerge %end %loop None
622 OpBranch %loop
623 %end = OpLabel
624 )";
625 
626   CompileSuccessfully(GenerateShaderCode(body));
627   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
628 }
629 
TEST_F(ValidateAdjacency,OpLoopMergePreceedsOpBranchConditionalSuccess)630 TEST_F(ValidateAdjacency, OpLoopMergePreceedsOpBranchConditionalSuccess) {
631   const std::string body = R"(
632 OpBranch %loop
633 %loop = OpLabel
634 OpLoopMerge %end %loop None
635 OpBranchConditional %true %loop %end
636 %end = OpLabel
637 )";
638 
639   CompileSuccessfully(GenerateShaderCode(body));
640   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
641 }
642 
TEST_F(ValidateAdjacency,OpLoopMergePreceedsBadOpFail)643 TEST_F(ValidateAdjacency, OpLoopMergePreceedsBadOpFail) {
644   const std::string body = R"(
645 OpBranch %loop
646 %loop = OpLabel
647 OpLoopMerge %end %loop None
648 OpNop
649 OpBranchConditional %true %loop %end
650 %end = OpLabel
651 )";
652 
653   CompileSuccessfully(GenerateShaderCode(body));
654   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
655   EXPECT_THAT(getDiagnosticString(),
656               HasSubstr("OpLoopMerge must immediately precede either an "
657                         "OpBranch or OpBranchConditional instruction."));
658 }
659 
TEST_F(ValidateAdjacency,OpSelectionMergePreceedsOpBranchConditionalSuccess)660 TEST_F(ValidateAdjacency, OpSelectionMergePreceedsOpBranchConditionalSuccess) {
661   const std::string body = R"(
662 OpSelectionMerge %end_label None
663 OpBranchConditional %true %true_label %false_label
664 %true_label = OpLabel
665 OpBranch %end_label
666 %false_label = OpLabel
667 OpBranch %end_label
668 %end_label = OpLabel
669 )";
670 
671   CompileSuccessfully(GenerateShaderCode(body));
672   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
673 }
674 
TEST_F(ValidateAdjacency,OpSelectionMergePreceedsOpSwitchSuccess)675 TEST_F(ValidateAdjacency, OpSelectionMergePreceedsOpSwitchSuccess) {
676   const std::string body = R"(
677 OpSelectionMerge %merge None
678 OpSwitch %zero %merge 0 %label
679 %label = OpLabel
680 OpBranch %merge
681 %merge = OpLabel
682 )";
683 
684   CompileSuccessfully(GenerateShaderCode(body));
685   EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
686 }
687 
TEST_F(ValidateAdjacency,OpSelectionMergePreceedsBadOpFail)688 TEST_F(ValidateAdjacency, OpSelectionMergePreceedsBadOpFail) {
689   const std::string body = R"(
690 OpSelectionMerge %merge None
691 OpNop
692 OpSwitch %zero %merge 0 %label
693 %label = OpLabel
694 OpBranch %merge
695 %merge = OpLabel
696 )";
697 
698   CompileSuccessfully(GenerateShaderCode(body));
699   EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
700   EXPECT_THAT(getDiagnosticString(),
701               HasSubstr("OpSelectionMerge must immediately precede either an "
702                         "OpBranchConditional or OpSwitch instruction"));
703 }
704 
705 }  // namespace
706 }  // namespace val
707 }  // namespace spvtools
708