1 // Copyright (c) 2015-2016 The Khronos Group 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 // Validation tests for Logical Layout
16
17 #include <algorithm>
18 #include <functional>
19 #include <sstream>
20 #include <string>
21 #include <tuple>
22 #include <utility>
23 #include <vector>
24
25 #include "gmock/gmock.h"
26 #include "source/diagnostic.h"
27 #include "test/unit_spirv.h"
28 #include "test/val/val_fixtures.h"
29
30 namespace spvtools {
31 namespace val {
32 namespace {
33
34 using ::testing::Eq;
35 using ::testing::HasSubstr;
36 using ::testing::StrEq;
37
38 using pred_type = std::function<spv_result_t(int)>;
39 using ValidateLayout = spvtest::ValidateBase<
40 std::tuple<int, std::tuple<std::string, pred_type, pred_type>>>;
41
42 // returns true if order is equal to VAL
43 template <int VAL, spv_result_t RET = SPV_ERROR_INVALID_LAYOUT>
Equals(int order)44 spv_result_t Equals(int order) {
45 return order == VAL ? SPV_SUCCESS : RET;
46 }
47
48 // returns true if order is between MIN and MAX(inclusive)
49 template <int MIN, int MAX, spv_result_t RET = SPV_ERROR_INVALID_LAYOUT>
50 struct Range {
Rangespvtools::val::__anon9db2ff4f0111::Range51 explicit Range(bool inverse = false) : inverse_(inverse) {}
operator ()spvtools::val::__anon9db2ff4f0111::Range52 spv_result_t operator()(int order) {
53 return (inverse_ ^ (order >= MIN && order <= MAX)) ? SPV_SUCCESS : RET;
54 }
55
56 private:
57 bool inverse_;
58 };
59
60 template <typename... T>
InvalidSet(int order)61 spv_result_t InvalidSet(int order) {
62 for (spv_result_t val : {T(true)(order)...})
63 if (val != SPV_SUCCESS) return val;
64 return SPV_SUCCESS;
65 }
66
67 // SPIRV source used to test the logical layout
getInstructions()68 const std::vector<std::string>& getInstructions() {
69 // clang-format off
70 static const std::vector<std::string> instructions = {
71 "OpCapability Shader",
72 "OpExtension \"TestExtension\"",
73 "%inst = OpExtInstImport \"GLSL.std.450\"",
74 "OpMemoryModel Logical GLSL450",
75 "OpEntryPoint GLCompute %func \"\"",
76 "OpExecutionMode %func LocalSize 1 1 1",
77 "OpExecutionModeId %func LocalSizeId %one %one %one",
78 "%str = OpString \"Test String\"",
79 "%str2 = OpString \"blabla\"",
80 "OpSource GLSL 450 %str \"uniform vec3 var = vec3(4.0);\"",
81 "OpSourceContinued \"void main(){return;}\"",
82 "OpSourceExtension \"Test extension\"",
83 "OpName %func \"MyFunction\"",
84 "OpMemberName %struct 1 \"my_member\"",
85 "OpDecorate %dgrp RowMajor",
86 "OpMemberDecorate %struct 1 RowMajor",
87 "%dgrp = OpDecorationGroup",
88 "OpGroupDecorate %dgrp %mat33 %mat44",
89 "%intt = OpTypeInt 32 1",
90 "%floatt = OpTypeFloat 32",
91 "%voidt = OpTypeVoid",
92 "%boolt = OpTypeBool",
93 "%vec4 = OpTypeVector %floatt 4",
94 "%vec3 = OpTypeVector %floatt 3",
95 "%mat33 = OpTypeMatrix %vec3 3",
96 "%mat44 = OpTypeMatrix %vec4 4",
97 "%struct = OpTypeStruct %intt %mat33",
98 "%vfunct = OpTypeFunction %voidt",
99 "%viifunct = OpTypeFunction %voidt %intt %intt",
100 "%one = OpConstant %intt 1",
101 // TODO(umar): OpConstant fails because the type is not defined
102 // TODO(umar): OpGroupMemberDecorate
103 "OpLine %str 3 4",
104 "OpNoLine",
105 "%func = OpFunction %voidt None %vfunct",
106 "%l = OpLabel",
107 "OpReturn ; %func return",
108 "OpFunctionEnd ; %func end",
109 "%func2 = OpFunction %voidt None %viifunct",
110 "%funcp1 = OpFunctionParameter %intt",
111 "%funcp2 = OpFunctionParameter %intt",
112 "%fLabel = OpLabel",
113 "OpNop",
114 "OpReturn ; %func2 return",
115 "OpFunctionEnd"
116 };
117 return instructions;
118 }
119
120 static const int kRangeEnd = 1000;
121 pred_type All = Range<0, kRangeEnd>();
122
123 INSTANTIATE_TEST_CASE_P(InstructionsOrder,
124 ValidateLayout,
125 ::testing::Combine(::testing::Range((int)0, (int)getInstructions().size()),
126 // Note: Because of ID dependencies between instructions, some instructions
127 // are not free to be placed anywhere without triggering an non-layout
128 // validation error. Therefore, "Lines to compile" for some instructions
129 // are not "All" in the below.
130 //
131 // | Instruction | Line(s) valid | Lines to compile
132 ::testing::Values(std::make_tuple(std::string("OpCapability") , Equals<0> , Range<0, 2>())
133 , std::make_tuple(std::string("OpExtension") , Equals<1> , All)
134 , std::make_tuple(std::string("OpExtInstImport") , Equals<2> , All)
135 , std::make_tuple(std::string("OpMemoryModel") , Equals<3> , Range<1, kRangeEnd>())
136 , std::make_tuple(std::string("OpEntryPoint") , Equals<4> , All)
137 , std::make_tuple(std::string("OpExecutionMode ") , Range<5, 6>() , All)
138 , std::make_tuple(std::string("OpExecutionModeId") , Range<5, 6>() , All)
139 , std::make_tuple(std::string("OpSource ") , Range<7, 11>() , Range<8, kRangeEnd>())
140 , std::make_tuple(std::string("OpSourceContinued ") , Range<7, 11>() , All)
141 , std::make_tuple(std::string("OpSourceExtension ") , Range<7, 11>() , All)
142 , std::make_tuple(std::string("%str2 = OpString ") , Range<7, 11>() , All)
143 , std::make_tuple(std::string("OpName ") , Range<12, 13>() , All)
144 , std::make_tuple(std::string("OpMemberName ") , Range<12, 13>() , All)
145 , std::make_tuple(std::string("OpDecorate ") , Range<14, 17>() , All)
146 , std::make_tuple(std::string("OpMemberDecorate ") , Range<14, 17>() , All)
147 , std::make_tuple(std::string("OpGroupDecorate ") , Range<14, 17>() , Range<17, kRangeEnd>())
148 , std::make_tuple(std::string("OpDecorationGroup") , Range<14, 17>() , Range<0, 16>())
149 , std::make_tuple(std::string("OpTypeBool") , Range<18, 31>() , All)
150 , std::make_tuple(std::string("OpTypeVoid") , Range<18, 31>() , Range<0, 26>())
151 , std::make_tuple(std::string("OpTypeFloat") , Range<18, 31>() , Range<0,21>())
152 , std::make_tuple(std::string("OpTypeInt") , Range<18, 31>() , Range<0, 21>())
153 , std::make_tuple(std::string("OpTypeVector %floatt 4") , Range<18, 31>() , Range<20, 24>())
154 , std::make_tuple(std::string("OpTypeMatrix %vec4 4") , Range<18, 31>() , Range<23, kRangeEnd>())
155 , std::make_tuple(std::string("OpTypeStruct") , Range<18, 31>() , Range<25, kRangeEnd>())
156 , std::make_tuple(std::string("%vfunct = OpTypeFunction"), Range<18, 31>() , Range<21, 31>())
157 , std::make_tuple(std::string("OpConstant") , Range<18, 31>() , Range<21, kRangeEnd>())
158 , std::make_tuple(std::string("OpLine ") , Range<18, kRangeEnd>() , Range<8, kRangeEnd>())
159 , std::make_tuple(std::string("OpNoLine") , Range<18, kRangeEnd>() , All)
160 , std::make_tuple(std::string("%fLabel = OpLabel") , Equals<39> , All)
161 , std::make_tuple(std::string("OpNop") , Equals<40> , Range<40,kRangeEnd>())
162 , std::make_tuple(std::string("OpReturn ; %func2 return") , Equals<41> , All)
163 )),);
164 // clang-format on
165
166 // Creates a new vector which removes the string if the substr is found in the
167 // instructions vector and reinserts it in the location specified by order.
168 // NOTE: This will not work correctly if there are two instances of substr in
169 // instructions
GenerateCode(std::string substr,int order)170 std::vector<std::string> GenerateCode(std::string substr, int order) {
171 std::vector<std::string> code(getInstructions().size());
172 std::vector<std::string> inst(1);
173 partition_copy(std::begin(getInstructions()), std::end(getInstructions()),
174 std::begin(code), std::begin(inst),
175 [=](const std::string& str) {
176 return std::string::npos == str.find(substr);
177 });
178
179 code.insert(std::begin(code) + order, inst.front());
180 return code;
181 }
182
183 // This test will check the logical layout of a binary by removing each
184 // instruction in the pair of the INSTANTIATE_TEST_CASE_P call and moving it in
185 // the SPIRV source formed by combining the vector "instructions".
TEST_P(ValidateLayout,Layout)186 TEST_P(ValidateLayout, Layout) {
187 int order;
188 std::string instruction;
189 pred_type pred;
190 pred_type test_pred; // Predicate to determine if the test should be build
191 std::tuple<std::string, pred_type, pred_type> testCase;
192
193 std::tie(order, testCase) = GetParam();
194 std::tie(instruction, pred, test_pred) = testCase;
195
196 // Skip test which break the code generation
197 if (test_pred(order)) return;
198
199 std::vector<std::string> code = GenerateCode(instruction, order);
200
201 std::stringstream ss;
202 std::copy(std::begin(code), std::end(code),
203 std::ostream_iterator<std::string>(ss, "\n"));
204
205 const auto env = SPV_ENV_UNIVERSAL_1_3;
206 // printf("code: \n%s\n", ss.str().c_str());
207 CompileSuccessfully(ss.str(), env);
208 spv_result_t result;
209 // clang-format off
210 ASSERT_EQ(pred(order), result = ValidateInstructions(env))
211 << "Actual: " << spvResultToString(result)
212 << "\nExpected: " << spvResultToString(pred(order))
213 << "\nOrder: " << order
214 << "\nInstruction: " << instruction
215 << "\nCode: \n" << ss.str();
216 // clang-format on
217 }
218
TEST_F(ValidateLayout,MemoryModelMissingBeforeEntryPoint)219 TEST_F(ValidateLayout, MemoryModelMissingBeforeEntryPoint) {
220 std::string str = R"(
221 OpCapability Matrix
222 OpExtension "TestExtension"
223 %inst = OpExtInstImport "GLSL.std.450"
224 OpEntryPoint GLCompute %func ""
225 OpExecutionMode %func LocalSize 1 1 1
226 )";
227
228 CompileSuccessfully(str);
229 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
230 EXPECT_THAT(
231 getDiagnosticString(),
232 HasSubstr(
233 "EntryPoint cannot appear before the memory model instruction"));
234 }
235
TEST_F(ValidateLayout,MemoryModelMissing)236 TEST_F(ValidateLayout, MemoryModelMissing) {
237 char str[] = R"(OpCapability Linkage)";
238 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
239 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT,
240 ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
241 EXPECT_THAT(getDiagnosticString(),
242 HasSubstr("Missing required OpMemoryModel instruction"));
243 }
244
TEST_F(ValidateLayout,MemoryModelSpecifiedTwice)245 TEST_F(ValidateLayout, MemoryModelSpecifiedTwice) {
246 char str[] = R"(
247 OpCapability Linkage
248 OpCapability Shader
249 OpMemoryModel Logical Simple
250 OpMemoryModel Logical Simple
251 )";
252
253 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
254 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT,
255 ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
256 EXPECT_THAT(getDiagnosticString(),
257 HasSubstr("OpMemoryModel should only be provided once"));
258 }
259
TEST_F(ValidateLayout,FunctionDefinitionBeforeDeclarationBad)260 TEST_F(ValidateLayout, FunctionDefinitionBeforeDeclarationBad) {
261 char str[] = R"(
262 OpCapability Shader
263 OpMemoryModel Logical GLSL450
264 OpDecorate %var Restrict
265 %intt = OpTypeInt 32 1
266 %voidt = OpTypeVoid
267 %vfunct = OpTypeFunction %voidt
268 %vifunct = OpTypeFunction %voidt %intt
269 %ptrt = OpTypePointer Function %intt
270 %func = OpFunction %voidt None %vfunct
271 %funcl = OpLabel
272 OpNop
273 OpReturn
274 OpFunctionEnd
275 %func2 = OpFunction %voidt None %vifunct ; must appear before definition
276 %func2p = OpFunctionParameter %intt
277 OpFunctionEnd
278 )";
279
280 CompileSuccessfully(str);
281 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
282 EXPECT_THAT(
283 getDiagnosticString(),
284 HasSubstr(
285 "Function declarations must appear before function definitions."));
286 }
287
288 // TODO(umar): Passes but gives incorrect error message. Should be fixed after
289 // type checking
TEST_F(ValidateLayout,LabelBeforeFunctionParameterBad)290 TEST_F(ValidateLayout, LabelBeforeFunctionParameterBad) {
291 char str[] = R"(
292 OpCapability Shader
293 OpMemoryModel Logical GLSL450
294 OpDecorate %var Restrict
295 %intt = OpTypeInt 32 1
296 %voidt = OpTypeVoid
297 %vfunct = OpTypeFunction %voidt
298 %vifunct = OpTypeFunction %voidt %intt
299 %ptrt = OpTypePointer Function %intt
300 %func = OpFunction %voidt None %vifunct
301 %funcl = OpLabel ; Label appears before function parameter
302 %func2p = OpFunctionParameter %intt
303 OpNop
304 OpReturn
305 OpFunctionEnd
306 )";
307
308 CompileSuccessfully(str);
309 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
310 EXPECT_THAT(getDiagnosticString(),
311 HasSubstr("Function parameters must only appear immediately "
312 "after the function definition"));
313 }
314
TEST_F(ValidateLayout,FuncParameterNotImmediatlyAfterFuncBad)315 TEST_F(ValidateLayout, FuncParameterNotImmediatlyAfterFuncBad) {
316 char str[] = R"(
317 OpCapability Shader
318 OpMemoryModel Logical GLSL450
319 OpDecorate %var Restrict
320 %intt = OpTypeInt 32 1
321 %voidt = OpTypeVoid
322 %vfunct = OpTypeFunction %voidt
323 %vifunct = OpTypeFunction %voidt %intt
324 %ptrt = OpTypePointer Function %intt
325 %func = OpFunction %voidt None %vifunct
326 %funcl = OpLabel
327 OpNop
328 OpBranch %next
329 %func2p = OpFunctionParameter %intt ;FunctionParameter appears in a function but not immediately afterwards
330 %next = OpLabel
331 OpNop
332 OpReturn
333 OpFunctionEnd
334 )";
335
336 CompileSuccessfully(str);
337 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
338 EXPECT_THAT(getDiagnosticString(),
339 HasSubstr("Function parameters must only appear immediately "
340 "after the function definition"));
341 }
342
TEST_F(ValidateLayout,OpUndefCanAppearInTypeDeclarationSection)343 TEST_F(ValidateLayout, OpUndefCanAppearInTypeDeclarationSection) {
344 std::string str = R"(
345 OpCapability Kernel
346 OpCapability Linkage
347 OpMemoryModel Logical OpenCL
348 %voidt = OpTypeVoid
349 %uintt = OpTypeInt 32 0
350 %funct = OpTypeFunction %voidt
351 %udef = OpUndef %uintt
352 %func = OpFunction %voidt None %funct
353 %entry = OpLabel
354 OpReturn
355 OpFunctionEnd
356 )";
357
358 CompileSuccessfully(str);
359 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
360 }
361
TEST_F(ValidateLayout,OpUndefCanAppearInBlock)362 TEST_F(ValidateLayout, OpUndefCanAppearInBlock) {
363 std::string str = R"(
364 OpCapability Kernel
365 OpCapability Linkage
366 OpMemoryModel Logical OpenCL
367 %voidt = OpTypeVoid
368 %uintt = OpTypeInt 32 0
369 %funct = OpTypeFunction %voidt
370 %func = OpFunction %voidt None %funct
371 %entry = OpLabel
372 %udef = OpUndef %uintt
373 OpReturn
374 OpFunctionEnd
375 )";
376
377 CompileSuccessfully(str);
378 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
379 }
380
TEST_F(ValidateLayout,MissingFunctionEndForFunctionWithBody)381 TEST_F(ValidateLayout, MissingFunctionEndForFunctionWithBody) {
382 const auto s = R"(
383 OpCapability Shader
384 OpCapability Linkage
385 OpMemoryModel Logical GLSL450
386 %void = OpTypeVoid
387 %tf = OpTypeFunction %void
388 %f = OpFunction %void None %tf
389 %l = OpLabel
390 OpReturn
391 )";
392
393 CompileSuccessfully(s);
394 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
395 EXPECT_THAT(getDiagnosticString(),
396 StrEq("Missing OpFunctionEnd at end of module."));
397 }
398
TEST_F(ValidateLayout,MissingFunctionEndForFunctionPrototype)399 TEST_F(ValidateLayout, MissingFunctionEndForFunctionPrototype) {
400 const auto s = R"(
401 OpCapability Shader
402 OpCapability Linkage
403 OpMemoryModel Logical GLSL450
404 %void = OpTypeVoid
405 %tf = OpTypeFunction %void
406 %f = OpFunction %void None %tf
407 )";
408
409 CompileSuccessfully(s);
410 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
411 EXPECT_THAT(getDiagnosticString(),
412 StrEq("Missing OpFunctionEnd at end of module."));
413 }
414
415 using ValidateOpFunctionParameter = spvtest::ValidateBase<int>;
416
TEST_F(ValidateOpFunctionParameter,OpLineBetweenParameters)417 TEST_F(ValidateOpFunctionParameter, OpLineBetweenParameters) {
418 const auto s = R"(
419 OpCapability Shader
420 OpCapability Linkage
421 OpMemoryModel Logical GLSL450
422 %foo_frag = OpString "foo.frag"
423 %i32 = OpTypeInt 32 1
424 %tf = OpTypeFunction %i32 %i32 %i32
425 %c = OpConstant %i32 123
426 %f = OpFunction %i32 None %tf
427 OpLine %foo_frag 1 1
428 %p1 = OpFunctionParameter %i32
429 OpNoLine
430 %p2 = OpFunctionParameter %i32
431 %l = OpLabel
432 OpReturnValue %c
433 OpFunctionEnd
434 )";
435 CompileSuccessfully(s);
436 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
437 }
438
TEST_F(ValidateOpFunctionParameter,TooManyParameters)439 TEST_F(ValidateOpFunctionParameter, TooManyParameters) {
440 const auto s = R"(
441 OpCapability Shader
442 OpCapability Linkage
443 OpMemoryModel Logical GLSL450
444 %i32 = OpTypeInt 32 1
445 %tf = OpTypeFunction %i32 %i32 %i32
446 %c = OpConstant %i32 123
447 %f = OpFunction %i32 None %tf
448 %p1 = OpFunctionParameter %i32
449 %p2 = OpFunctionParameter %i32
450 %xp3 = OpFunctionParameter %i32
451 %xp4 = OpFunctionParameter %i32
452 %xp5 = OpFunctionParameter %i32
453 %xp6 = OpFunctionParameter %i32
454 %xp7 = OpFunctionParameter %i32
455 %l = OpLabel
456 OpReturnValue %c
457 OpFunctionEnd
458 )";
459 CompileSuccessfully(s);
460 ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
461 }
462
463 using ValidateEntryPoint = spvtest::ValidateBase<bool>;
464
465 // Tests that not having OpEntryPoint causes an error.
TEST_F(ValidateEntryPoint,NoEntryPointBad)466 TEST_F(ValidateEntryPoint, NoEntryPointBad) {
467 std::string spirv = R"(
468 OpCapability Shader
469 OpMemoryModel Logical GLSL450)";
470 CompileSuccessfully(spirv);
471 EXPECT_EQ(SPV_ERROR_INVALID_BINARY, ValidateInstructions());
472 EXPECT_THAT(getDiagnosticString(),
473 HasSubstr("No OpEntryPoint instruction was found. This is only "
474 "allowed if the Linkage capability is being used."));
475 }
476
477 // Invalid. A function may not be a target of both OpEntryPoint and
478 // OpFunctionCall.
TEST_F(ValidateEntryPoint,FunctionIsTargetOfEntryPointAndFunctionCallBad)479 TEST_F(ValidateEntryPoint, FunctionIsTargetOfEntryPointAndFunctionCallBad) {
480 std::string spirv = R"(
481 OpCapability Shader
482 OpMemoryModel Logical GLSL450
483 OpEntryPoint Fragment %foo "foo"
484 OpExecutionMode %foo OriginUpperLeft
485 %voidt = OpTypeVoid
486 %funct = OpTypeFunction %voidt
487 %foo = OpFunction %voidt None %funct
488 %entry = OpLabel
489 %recurse = OpFunctionCall %voidt %foo
490 OpReturn
491 OpFunctionEnd
492 )";
493 CompileSuccessfully(spirv);
494 EXPECT_EQ(SPV_ERROR_INVALID_BINARY, ValidateInstructions());
495 EXPECT_THAT(
496 getDiagnosticString(),
497 HasSubstr("A function (1) may not be targeted by both an OpEntryPoint "
498 "instruction and an OpFunctionCall instruction."));
499 }
500
501 // Invalid. Must be within a function to make a function call.
TEST_F(ValidateEntryPoint,FunctionCallOutsideFunctionBody)502 TEST_F(ValidateEntryPoint, FunctionCallOutsideFunctionBody) {
503 std::string spirv = R"(
504 OpCapability Shader
505 %1 = OpExtInstImport "GLSL.std.450"
506 OpMemoryModel Logical GLSL450
507 OpName %variableName "variableName"
508 %34 = OpFunctionCall %variableName %1
509 )";
510 CompileSuccessfully(spirv);
511 EXPECT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions());
512 EXPECT_THAT(getDiagnosticString(),
513 HasSubstr("FunctionCall must happen within a function body."));
514 }
515
516 // Valid. Module with a function but no entry point is valid when Linkage
517 // Capability is used.
TEST_F(ValidateEntryPoint,NoEntryPointWithLinkageCapGood)518 TEST_F(ValidateEntryPoint, NoEntryPointWithLinkageCapGood) {
519 std::string spirv = R"(
520 OpCapability Shader
521 OpCapability Linkage
522 OpMemoryModel Logical GLSL450
523 %voidt = OpTypeVoid
524 %funct = OpTypeFunction %voidt
525 %foo = OpFunction %voidt None %funct
526 %entry = OpLabel
527 OpReturn
528 OpFunctionEnd
529 )";
530 CompileSuccessfully(spirv);
531 EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
532 }
533
TEST_F(ValidateLayout,ModuleProcessedInvalidIn10)534 TEST_F(ValidateLayout, ModuleProcessedInvalidIn10) {
535 char str[] = R"(
536 OpCapability Shader
537 OpCapability Linkage
538 OpMemoryModel Logical GLSL450
539 OpName %void "void"
540 OpModuleProcessed "this is ok in 1.1 and later"
541 OpDecorate %void Volatile ; bogus, but makes the example short
542 %void = OpTypeVoid
543 )";
544
545 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
546 ASSERT_EQ(SPV_ERROR_WRONG_VERSION,
547 ValidateInstructions(SPV_ENV_UNIVERSAL_1_0));
548 // In a 1.0 environment the version check fails.
549 EXPECT_THAT(getDiagnosticString(),
550 HasSubstr("Invalid SPIR-V binary version 1.1 for target "
551 "environment SPIR-V 1.0."));
552 }
553
TEST_F(ValidateLayout,ModuleProcessedValidIn11)554 TEST_F(ValidateLayout, ModuleProcessedValidIn11) {
555 char str[] = R"(
556 OpCapability Shader
557 OpCapability Linkage
558 OpMemoryModel Logical GLSL450
559 OpName %void "void"
560 OpModuleProcessed "this is ok in 1.1 and later"
561 OpDecorate %void Volatile ; bogus, but makes the example short
562 %void = OpTypeVoid
563 )";
564
565 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
566 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
567 EXPECT_THAT(getDiagnosticString(), Eq(""));
568 }
569
TEST_F(ValidateLayout,ModuleProcessedBeforeLastNameIsTooEarly)570 TEST_F(ValidateLayout, ModuleProcessedBeforeLastNameIsTooEarly) {
571 char str[] = R"(
572 OpCapability Shader
573 OpCapability Linkage
574 OpMemoryModel Logical GLSL450
575 OpModuleProcessed "this is too early"
576 OpName %void "void"
577 %void = OpTypeVoid
578 )";
579
580 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
581 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT,
582 ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
583 // By the mechanics of the validator, we assume ModuleProcessed is in the
584 // right spot, but then that OpName is in the wrong spot.
585 EXPECT_THAT(getDiagnosticString(),
586 HasSubstr("Name cannot appear in a function declaration"));
587 }
588
TEST_F(ValidateLayout,ModuleProcessedInvalidAfterFirstAnnotation)589 TEST_F(ValidateLayout, ModuleProcessedInvalidAfterFirstAnnotation) {
590 char str[] = R"(
591 OpCapability Shader
592 OpCapability Linkage
593 OpMemoryModel Logical GLSL450
594 OpDecorate %void Volatile ; this is bogus, but keeps the example short
595 OpModuleProcessed "this is too late"
596 %void = OpTypeVoid
597 )";
598
599 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
600 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT,
601 ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
602 EXPECT_THAT(
603 getDiagnosticString(),
604 HasSubstr("ModuleProcessed cannot appear in a function declaration"));
605 }
606
TEST_F(ValidateLayout,ModuleProcessedInvalidInFunctionBeforeLabel)607 TEST_F(ValidateLayout, ModuleProcessedInvalidInFunctionBeforeLabel) {
608 char str[] = R"(
609 OpCapability Shader
610 OpMemoryModel Logical GLSL450
611 OpEntryPoint GLCompute %main "main"
612 %void = OpTypeVoid
613 %voidfn = OpTypeFunction %void
614 %main = OpFunction %void None %voidfn
615 OpModuleProcessed "this is too late, in function before label"
616 %entry = OpLabel
617 OpReturn
618 OpFunctionEnd
619 )";
620
621 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
622 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT,
623 ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
624 EXPECT_THAT(
625 getDiagnosticString(),
626 HasSubstr("ModuleProcessed cannot appear in a function declaration"));
627 }
628
TEST_F(ValidateLayout,ModuleProcessedInvalidInBasicBlock)629 TEST_F(ValidateLayout, ModuleProcessedInvalidInBasicBlock) {
630 char str[] = R"(
631 OpCapability Shader
632 OpMemoryModel Logical GLSL450
633 OpEntryPoint GLCompute %main "main"
634 %void = OpTypeVoid
635 %voidfn = OpTypeFunction %void
636 %main = OpFunction %void None %voidfn
637 %entry = OpLabel
638 OpModuleProcessed "this is too late, in basic block"
639 OpReturn
640 OpFunctionEnd
641 )";
642
643 CompileSuccessfully(str, SPV_ENV_UNIVERSAL_1_1);
644 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT,
645 ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
646 EXPECT_THAT(
647 getDiagnosticString(),
648 HasSubstr("ModuleProcessed cannot appear in a function declaration"));
649 }
650
TEST_F(ValidateLayout,WebGPUCallerBeforeCalleeBad)651 TEST_F(ValidateLayout, WebGPUCallerBeforeCalleeBad) {
652 char str[] = R"(
653 OpCapability Shader
654 OpCapability VulkanMemoryModelKHR
655 OpExtension "SPV_KHR_vulkan_memory_model"
656 OpMemoryModel Logical VulkanKHR
657 OpEntryPoint GLCompute %main "main"
658 %void = OpTypeVoid
659 %voidfn = OpTypeFunction %void
660 %main = OpFunction %void None %voidfn
661 %1 = OpLabel
662 %2 = OpFunctionCall %void %callee
663 OpReturn
664 OpFunctionEnd
665 %callee = OpFunction %void None %voidfn
666 %3 = OpLabel
667 OpReturn
668 OpFunctionEnd
669 )";
670
671 CompileSuccessfully(str, SPV_ENV_WEBGPU_0);
672 ASSERT_EQ(SPV_ERROR_INVALID_LAYOUT, ValidateInstructions(SPV_ENV_WEBGPU_0));
673 EXPECT_THAT(getDiagnosticString(),
674 HasSubstr("For WebGPU, functions need to be defined before being "
675 "called.\n %5 = OpFunctionCall %void %6\n"));
676 }
677
TEST_F(ValidateLayout,WebGPUCalleeBeforeCallerGood)678 TEST_F(ValidateLayout, WebGPUCalleeBeforeCallerGood) {
679 char str[] = R"(
680 OpCapability Shader
681 OpCapability VulkanMemoryModelKHR
682 OpExtension "SPV_KHR_vulkan_memory_model"
683 OpMemoryModel Logical VulkanKHR
684 OpEntryPoint GLCompute %main "main"
685 %void = OpTypeVoid
686 %voidfn = OpTypeFunction %void
687 %callee = OpFunction %void None %voidfn
688 %3 = OpLabel
689 OpReturn
690 OpFunctionEnd
691 %main = OpFunction %void None %voidfn
692 %1 = OpLabel
693 %2 = OpFunctionCall %void %callee
694 OpReturn
695 OpFunctionEnd
696 )";
697
698 CompileSuccessfully(str, SPV_ENV_WEBGPU_0);
699 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_WEBGPU_0));
700 }
701
702 // TODO(umar): Test optional instructions
703
704 } // namespace
705 } // namespace val
706 } // namespace spvtools
707