1 // Copyright (c) 2017 Google 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 illegal literals
16
17 #include <string>
18 #include <utility>
19
20 #include "gmock/gmock.h"
21 #include "test/val/val_fixtures.h"
22
23 namespace spvtools {
24 namespace val {
25 namespace {
26
27 using ::testing::HasSubstr;
28
29 using ValidateLiterals = spvtest::ValidateBase<std::string>;
30 using ValidateLiteralsShader = spvtest::ValidateBase<std::string>;
31 using ValidateLiteralsKernel = spvtest::ValidateBase<std::string>;
32
GenerateShaderCode()33 std::string GenerateShaderCode() {
34 std::string str = R"(
35 OpCapability Shader
36 OpCapability Linkage
37 OpCapability Int16
38 OpCapability Int64
39 OpCapability Float16
40 OpCapability Float64
41 OpMemoryModel Logical GLSL450
42 %int16 = OpTypeInt 16 1
43 %uint16 = OpTypeInt 16 0
44 %int32 = OpTypeInt 32 1
45 %uint32 = OpTypeInt 32 0
46 %int64 = OpTypeInt 64 1
47 %uint64 = OpTypeInt 64 0
48 %half = OpTypeFloat 16
49 %float = OpTypeFloat 32
50 %double = OpTypeFloat 64
51 %10 = OpTypeVoid
52 )";
53 return str;
54 }
55
GenerateKernelCode()56 std::string GenerateKernelCode() {
57 std::string str = R"(
58 OpCapability Kernel
59 OpCapability Addresses
60 OpCapability Linkage
61 OpCapability Int8
62 OpMemoryModel Physical64 OpenCL
63 %uint8 = OpTypeInt 8 0
64 )";
65 return str;
66 }
67
TEST_F(ValidateLiterals,LiteralsShaderGood)68 TEST_F(ValidateLiterals, LiteralsShaderGood) {
69 std::string str = GenerateShaderCode() + R"(
70 %11 = OpConstant %int16 !0x00007FFF
71 %12 = OpConstant %int16 !0xFFFF8000
72 %13 = OpConstant %int16 !0xFFFFABCD
73 %14 = OpConstant %uint16 !0x0000ABCD
74 %15 = OpConstant %int16 -32768
75 %16 = OpConstant %uint16 65535
76 %17 = OpConstant %int32 -2147483648
77 %18 = OpConstant %uint32 4294967295
78 %19 = OpConstant %int64 -9223372036854775808
79 %20 = OpConstant %uint64 18446744073709551615
80 %21 = OpConstant %half !0x0000FFFF
81 %22 = OpConstant %float !0xFFFFFFFF
82 %23 = OpConstant %double !0xFFFFFFFF !0xFFFFFFFF
83 )";
84 CompileSuccessfully(str);
85 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
86 }
87
TEST_F(ValidateLiterals,InvalidInt)88 TEST_F(ValidateLiterals, InvalidInt) {
89 std::string str = GenerateShaderCode() + R"(
90 %11 = OpTypeInt 32 90
91 )";
92 CompileSuccessfully(str);
93 EXPECT_EQ(SPV_ERROR_INVALID_VALUE, ValidateInstructions());
94 EXPECT_THAT(getDiagnosticString(),
95 HasSubstr("OpTypeInt has invalid signedness:"));
96 }
97
TEST_P(ValidateLiteralsShader,LiteralsShaderBad)98 TEST_P(ValidateLiteralsShader, LiteralsShaderBad) {
99 std::string str = GenerateShaderCode() + GetParam();
100 std::string inst_id = "11";
101 CompileSuccessfully(str);
102 EXPECT_EQ(SPV_ERROR_INVALID_VALUE, ValidateInstructions());
103 EXPECT_THAT(
104 getDiagnosticString(),
105 HasSubstr("The high-order bits of a literal number in instruction <id> " +
106 inst_id +
107 " must be 0 for a floating-point type, "
108 "or 0 for an integer type with Signedness of 0, "
109 "or sign extended when Signedness is 1"));
110 }
111
112 INSTANTIATE_TEST_SUITE_P(
113 LiteralsShaderCases, ValidateLiteralsShader,
114 ::testing::Values("%11 = OpConstant %int16 !0xFFFF0000", // Sign bit is 0
115 "%11 = OpConstant %int16 !0x00008000", // Sign bit is 1
116 "%11 = OpConstant %int16 !0xABCD8000", // Sign bit is 1
117 "%11 = OpConstant %int16 !0xABCD0000",
118 "%11 = OpConstant %uint16 !0xABCD0000",
119 "%11 = OpConstant %half !0xABCD0000",
120 "%11 = OpConstant %half !0x00010000"));
121
TEST_F(ValidateLiterals,LiteralsKernelGood)122 TEST_F(ValidateLiterals, LiteralsKernelGood) {
123 std::string str = GenerateKernelCode() + R"(
124 %4 = OpConstant %uint8 !0x000000AB
125 %6 = OpConstant %uint8 255
126 )";
127 CompileSuccessfully(str);
128 ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
129 }
130
TEST_P(ValidateLiteralsKernel,LiteralsKernelBad)131 TEST_P(ValidateLiteralsKernel, LiteralsKernelBad) {
132 std::string str = GenerateKernelCode() + GetParam();
133 std::string inst_id = "2";
134 CompileSuccessfully(str);
135 EXPECT_EQ(SPV_ERROR_INVALID_VALUE, ValidateInstructions());
136 EXPECT_THAT(
137 getDiagnosticString(),
138 HasSubstr("The high-order bits of a literal number in instruction <id> " +
139 inst_id +
140 " must be 0 for a floating-point type, "
141 "or 0 for an integer type with Signedness of 0, "
142 "or sign extended when Signedness is 1"));
143 }
144
145 INSTANTIATE_TEST_SUITE_P(
146 LiteralsKernelCases, ValidateLiteralsKernel,
147 ::testing::Values("%2 = OpConstant %uint8 !0xABCDEF00",
148 "%2 = OpConstant %uint8 !0xABCDEFFF"));
149
150 } // namespace
151 } // namespace val
152 } // namespace spvtools
153