1 // Copyright (c) 2016 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 #include <algorithm>
16 #include <cstdarg>
17 #include <iostream>
18 #include <sstream>
19 #include <string>
20 #include <unordered_set>
21 #include <vector>
22
23 #include "test/opt/assembly_builder.h"
24 #include "test/opt/pass_fixture.h"
25 #include "test/opt/pass_utils.h"
26
27 namespace spvtools {
28 namespace opt {
29 namespace {
30
31 using EliminateDeadConstantBasicTest = PassTest<::testing::Test>;
32
TEST_F(EliminateDeadConstantBasicTest,BasicAllDeadConstants)33 TEST_F(EliminateDeadConstantBasicTest, BasicAllDeadConstants) {
34 const std::vector<const char*> text = {
35 // clang-format off
36 "OpCapability Shader",
37 "OpCapability Float64",
38 "%1 = OpExtInstImport \"GLSL.std.450\"",
39 "OpMemoryModel Logical GLSL450",
40 "OpEntryPoint Vertex %main \"main\"",
41 "OpName %main \"main\"",
42 "%void = OpTypeVoid",
43 "%4 = OpTypeFunction %void",
44 "%bool = OpTypeBool",
45 "%true = OpConstantTrue %bool",
46 "%false = OpConstantFalse %bool",
47 "%int = OpTypeInt 32 1",
48 "%9 = OpConstant %int 1",
49 "%uint = OpTypeInt 32 0",
50 "%11 = OpConstant %uint 2",
51 "%float = OpTypeFloat 32",
52 "%13 = OpConstant %float 3.1415",
53 "%double = OpTypeFloat 64",
54 "%15 = OpConstant %double 3.14159265358979",
55 "%main = OpFunction %void None %4",
56 "%16 = OpLabel",
57 "OpReturn",
58 "OpFunctionEnd",
59 // clang-format on
60 };
61 // None of the above constants is ever used, so all of them should be
62 // eliminated.
63 const char* const_decl_opcodes[] = {
64 " OpConstantTrue ",
65 " OpConstantFalse ",
66 " OpConstant ",
67 };
68 // Skip lines that have any one of const_decl_opcodes.
69 const std::string expected_disassembly =
70 SelectiveJoin(text, [&const_decl_opcodes](const char* line) {
71 return std::any_of(
72 std::begin(const_decl_opcodes), std::end(const_decl_opcodes),
73 [&line](const char* const_decl_op) {
74 return std::string(line).find(const_decl_op) != std::string::npos;
75 });
76 });
77
78 SinglePassRunAndCheck<EliminateDeadConstantPass>(
79 JoinAllInsts(text), expected_disassembly, /* skip_nop = */ true);
80 }
81
TEST_F(EliminateDeadConstantBasicTest,BasicNoneDeadConstants)82 TEST_F(EliminateDeadConstantBasicTest, BasicNoneDeadConstants) {
83 const std::vector<const char*> text = {
84 // clang-format off
85 "OpCapability Shader",
86 "OpCapability Float64",
87 "%1 = OpExtInstImport \"GLSL.std.450\"",
88 "OpMemoryModel Logical GLSL450",
89 "OpEntryPoint Vertex %main \"main\"",
90 "OpName %main \"main\"",
91 "OpName %btv \"btv\"",
92 "OpName %bfv \"bfv\"",
93 "OpName %iv \"iv\"",
94 "OpName %uv \"uv\"",
95 "OpName %fv \"fv\"",
96 "OpName %dv \"dv\"",
97 "%void = OpTypeVoid",
98 "%10 = OpTypeFunction %void",
99 "%bool = OpTypeBool",
100 "%_ptr_Function_bool = OpTypePointer Function %bool",
101 "%true = OpConstantTrue %bool",
102 "%false = OpConstantFalse %bool",
103 "%int = OpTypeInt 32 1",
104 "%_ptr_Function_int = OpTypePointer Function %int",
105 "%int_1 = OpConstant %int 1",
106 "%uint = OpTypeInt 32 0",
107 "%_ptr_Function_uint = OpTypePointer Function %uint",
108 "%uint_2 = OpConstant %uint 2",
109 "%float = OpTypeFloat 32",
110 "%_ptr_Function_float = OpTypePointer Function %float",
111 "%float_3_1415 = OpConstant %float 3.1415",
112 "%double = OpTypeFloat 64",
113 "%_ptr_Function_double = OpTypePointer Function %double",
114 "%double_3_14159265358979 = OpConstant %double 3.14159265358979",
115 "%main = OpFunction %void None %10",
116 "%27 = OpLabel",
117 "%btv = OpVariable %_ptr_Function_bool Function",
118 "%bfv = OpVariable %_ptr_Function_bool Function",
119 "%iv = OpVariable %_ptr_Function_int Function",
120 "%uv = OpVariable %_ptr_Function_uint Function",
121 "%fv = OpVariable %_ptr_Function_float Function",
122 "%dv = OpVariable %_ptr_Function_double Function",
123 "OpStore %btv %true",
124 "OpStore %bfv %false",
125 "OpStore %iv %int_1",
126 "OpStore %uv %uint_2",
127 "OpStore %fv %float_3_1415",
128 "OpStore %dv %double_3_14159265358979",
129 "OpReturn",
130 "OpFunctionEnd",
131 // clang-format on
132 };
133 // All constants are used, so none of them should be eliminated.
134 SinglePassRunAndCheck<EliminateDeadConstantPass>(
135 JoinAllInsts(text), JoinAllInsts(text), /* skip_nop = */ true);
136 }
137
138 struct EliminateDeadConstantTestCase {
139 // Type declarations and constants that should be kept.
140 std::vector<std::string> used_consts;
141 // Instructions that refer to constants, this is added to create uses for
142 // some constants so they won't be treated as dead constants.
143 std::vector<std::string> main_insts;
144 // Dead constants that should be removed.
145 std::vector<std::string> dead_consts;
146 };
147
148 // All types that are potentially required in EliminateDeadConstantTest.
149 const std::vector<std::string> CommonTypes = {
150 // clang-format off
151 // scalar types
152 "%bool = OpTypeBool",
153 "%uint = OpTypeInt 32 0",
154 "%int = OpTypeInt 32 1",
155 "%float = OpTypeFloat 32",
156 "%double = OpTypeFloat 64",
157 // vector types
158 "%v2bool = OpTypeVector %bool 2",
159 "%v2uint = OpTypeVector %uint 2",
160 "%v2int = OpTypeVector %int 2",
161 "%v3int = OpTypeVector %int 3",
162 "%v4int = OpTypeVector %int 4",
163 "%v2float = OpTypeVector %float 2",
164 "%v3float = OpTypeVector %float 3",
165 "%v2double = OpTypeVector %double 2",
166 // variable pointer types
167 "%_pf_bool = OpTypePointer Function %bool",
168 "%_pf_uint = OpTypePointer Function %uint",
169 "%_pf_int = OpTypePointer Function %int",
170 "%_pf_float = OpTypePointer Function %float",
171 "%_pf_double = OpTypePointer Function %double",
172 "%_pf_v2int = OpTypePointer Function %v2int",
173 "%_pf_v3int = OpTypePointer Function %v3int",
174 "%_pf_v2float = OpTypePointer Function %v2float",
175 "%_pf_v3float = OpTypePointer Function %v3float",
176 "%_pf_v2double = OpTypePointer Function %v2double",
177 // struct types
178 "%inner_struct = OpTypeStruct %bool %int %float %double",
179 "%outer_struct = OpTypeStruct %inner_struct %int %double",
180 "%flat_struct = OpTypeStruct %bool %int %float %double",
181 // clang-format on
182 };
183
184 using EliminateDeadConstantTest =
185 PassTest<::testing::TestWithParam<EliminateDeadConstantTestCase>>;
186
TEST_P(EliminateDeadConstantTest,Custom)187 TEST_P(EliminateDeadConstantTest, Custom) {
188 auto& tc = GetParam();
189 AssemblyBuilder builder;
190 builder.AppendTypesConstantsGlobals(CommonTypes)
191 .AppendTypesConstantsGlobals(tc.used_consts)
192 .AppendInMain(tc.main_insts);
193 const std::string expected = builder.GetCode();
194 builder.AppendTypesConstantsGlobals(tc.dead_consts);
195 const std::string assembly_with_dead_const = builder.GetCode();
196 SinglePassRunAndCheck<EliminateDeadConstantPass>(
197 assembly_with_dead_const, expected, /* skip_nop = */ true);
198 }
199
200 INSTANTIATE_TEST_SUITE_P(
201 ScalarTypeConstants, EliminateDeadConstantTest,
202 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
203 // clang-format off
204 // Scalar type constants, one dead constant and one used constant.
205 {
206 /* .used_consts = */
207 {
208 "%used_const_int = OpConstant %int 1",
209 },
210 /* .main_insts = */
211 {
212 "%int_var = OpVariable %_pf_int Function",
213 "OpStore %int_var %used_const_int",
214 },
215 /* .dead_consts = */
216 {
217 "%dead_const_int = OpConstant %int 1",
218 },
219 },
220 {
221 /* .used_consts = */
222 {
223 "%used_const_uint = OpConstant %uint 1",
224 },
225 /* .main_insts = */
226 {
227 "%uint_var = OpVariable %_pf_uint Function",
228 "OpStore %uint_var %used_const_uint",
229 },
230 /* .dead_consts = */
231 {
232 "%dead_const_uint = OpConstant %uint 1",
233 },
234 },
235 {
236 /* .used_consts = */
237 {
238 "%used_const_float = OpConstant %float 3.1415",
239 },
240 /* .main_insts = */
241 {
242 "%float_var = OpVariable %_pf_float Function",
243 "OpStore %float_var %used_const_float",
244 },
245 /* .dead_consts = */
246 {
247 "%dead_const_float = OpConstant %float 3.1415",
248 },
249 },
250 {
251 /* .used_consts = */
252 {
253 "%used_const_double = OpConstant %double 3.141592653",
254 },
255 /* .main_insts = */
256 {
257 "%double_var = OpVariable %_pf_double Function",
258 "OpStore %double_var %used_const_double",
259 },
260 /* .dead_consts = */
261 {
262 "%dead_const_double = OpConstant %double 3.141592653",
263 },
264 },
265 // clang-format on
266 })));
267
268 INSTANTIATE_TEST_SUITE_P(
269 VectorTypeConstants, EliminateDeadConstantTest,
270 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
271 // clang-format off
272 // Tests eliminating dead constant type ivec2. One dead constant vector
273 // and one used constant vector, each built from its own group of
274 // scalar constants.
275 {
276 /* .used_consts = */
277 {
278 "%used_int_x = OpConstant %int 1",
279 "%used_int_y = OpConstant %int 2",
280 "%used_v2int = OpConstantComposite %v2int %used_int_x %used_int_y",
281 },
282 /* .main_insts = */
283 {
284 "%v2int_var = OpVariable %_pf_v2int Function",
285 "OpStore %v2int_var %used_v2int",
286 },
287 /* .dead_consts = */
288 {
289 "%dead_int_x = OpConstant %int 1",
290 "%dead_int_y = OpConstant %int 2",
291 "%dead_v2int = OpConstantComposite %v2int %dead_int_x %dead_int_y",
292 },
293 },
294 // Tests eliminating dead constant ivec2. One dead constant vector and
295 // one used constant vector. But both built from a same group of
296 // scalar constants.
297 {
298 /* .used_consts = */
299 {
300 "%used_int_x = OpConstant %int 1",
301 "%used_int_y = OpConstant %int 2",
302 "%used_int_z = OpConstant %int 3",
303 "%used_v3int = OpConstantComposite %v3int %used_int_x %used_int_y %used_int_z",
304 },
305 /* .main_insts = */
306 {
307 "%v3int_var = OpVariable %_pf_v3int Function",
308 "OpStore %v3int_var %used_v3int",
309 },
310 /* .dead_consts = */
311 {
312 "%dead_v3int = OpConstantComposite %v3int %used_int_x %used_int_y %used_int_z",
313 },
314 },
315 // Tests eliminating dead cosntant vec2. One dead constant vector and
316 // one used constant vector. Each built from its own group of scalar
317 // constants.
318 {
319 /* .used_consts = */
320 {
321 "%used_float_x = OpConstant %float 3.1415",
322 "%used_float_y = OpConstant %float 4.25",
323 "%used_v2float = OpConstantComposite %v2float %used_float_x %used_float_y",
324 },
325 /* .main_insts = */
326 {
327 "%v2float_var = OpVariable %_pf_v2float Function",
328 "OpStore %v2float_var %used_v2float",
329 },
330 /* .dead_consts = */
331 {
332 "%dead_float_x = OpConstant %float 3.1415",
333 "%dead_float_y = OpConstant %float 4.25",
334 "%dead_v2float = OpConstantComposite %v2float %dead_float_x %dead_float_y",
335 },
336 },
337 // Tests eliminating dead cosntant vec2. One dead constant vector and
338 // one used constant vector. Both built from a same group of scalar
339 // constants.
340 {
341 /* .used_consts = */
342 {
343 "%used_float_x = OpConstant %float 3.1415",
344 "%used_float_y = OpConstant %float 4.25",
345 "%used_float_z = OpConstant %float 4.75",
346 "%used_v3float = OpConstantComposite %v3float %used_float_x %used_float_y %used_float_z",
347 },
348 /* .main_insts = */
349 {
350 "%v3float_var = OpVariable %_pf_v3float Function",
351 "OpStore %v3float_var %used_v3float",
352 },
353 /* .dead_consts = */
354 {
355 "%dead_v3float = OpConstantComposite %v3float %used_float_x %used_float_y %used_float_z",
356 },
357 },
358 // clang-format on
359 })));
360
361 INSTANTIATE_TEST_SUITE_P(
362 StructTypeConstants, EliminateDeadConstantTest,
363 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
364 // clang-format off
365 // A plain struct type dead constants. All of its components are dead
366 // constants too.
367 {
368 /* .used_consts = */ {},
369 /* .main_insts = */ {},
370 /* .dead_consts = */
371 {
372 "%dead_bool = OpConstantTrue %bool",
373 "%dead_int = OpConstant %int 1",
374 "%dead_float = OpConstant %float 2.5",
375 "%dead_double = OpConstant %double 3.14159265358979",
376 "%dead_struct = OpConstantComposite %flat_struct %dead_bool %dead_int %dead_float %dead_double",
377 },
378 },
379 // A plain struct type dead constants. Some of its components are dead
380 // constants while others are not.
381 {
382 /* .used_consts = */
383 {
384 "%used_int = OpConstant %int 1",
385 "%used_double = OpConstant %double 3.14159265358979",
386 },
387 /* .main_insts = */
388 {
389 "%int_var = OpVariable %_pf_int Function",
390 "OpStore %int_var %used_int",
391 "%double_var = OpVariable %_pf_double Function",
392 "OpStore %double_var %used_double",
393 },
394 /* .dead_consts = */
395 {
396 "%dead_bool = OpConstantTrue %bool",
397 "%dead_float = OpConstant %float 2.5",
398 "%dead_struct = OpConstantComposite %flat_struct %dead_bool %used_int %dead_float %used_double",
399 },
400 },
401 // A nesting struct type dead constants. All components of both outer
402 // and inner structs are dead and should be removed after dead constant
403 // elimination.
404 {
405 /* .used_consts = */ {},
406 /* .main_insts = */ {},
407 /* .dead_consts = */
408 {
409 "%dead_bool = OpConstantTrue %bool",
410 "%dead_int = OpConstant %int 1",
411 "%dead_float = OpConstant %float 2.5",
412 "%dead_double = OpConstant %double 3.1415926535",
413 "%dead_inner_struct = OpConstantComposite %inner_struct %dead_bool %dead_int %dead_float %dead_double",
414 "%dead_int2 = OpConstant %int 2",
415 "%dead_double2 = OpConstant %double 1.428571428514",
416 "%dead_outer_struct = OpConstantComposite %outer_struct %dead_inner_struct %dead_int2 %dead_double2",
417 },
418 },
419 // A nesting struct type dead constants. Some of its components are
420 // dead constants while others are not.
421 {
422 /* .used_consts = */
423 {
424 "%used_int = OpConstant %int 1",
425 "%used_double = OpConstant %double 3.14159265358979",
426 },
427 /* .main_insts = */
428 {
429 "%int_var = OpVariable %_pf_int Function",
430 "OpStore %int_var %used_int",
431 "%double_var = OpVariable %_pf_double Function",
432 "OpStore %double_var %used_double",
433 },
434 /* .dead_consts = */
435 {
436 "%dead_bool = OpConstantTrue %bool",
437 "%dead_float = OpConstant %float 2.5",
438 "%dead_inner_struct = OpConstantComposite %inner_struct %dead_bool %used_int %dead_float %used_double",
439 "%dead_int = OpConstant %int 2",
440 "%dead_outer_struct = OpConstantComposite %outer_struct %dead_inner_struct %dead_int %used_double",
441 },
442 },
443 // A nesting struct case. The inner struct is used while the outer struct is not
444 {
445 /* .used_const = */
446 {
447 "%used_bool = OpConstantTrue %bool",
448 "%used_int = OpConstant %int 1",
449 "%used_float = OpConstant %float 1.25",
450 "%used_double = OpConstant %double 1.23456789012345",
451 "%used_inner_struct = OpConstantComposite %inner_struct %used_bool %used_int %used_float %used_double",
452 },
453 /* .main_insts = */
454 {
455 "%bool_var = OpVariable %_pf_bool Function",
456 "%bool_from_inner_struct = OpCompositeExtract %bool %used_inner_struct 0",
457 "OpStore %bool_var %bool_from_inner_struct",
458 },
459 /* .dead_consts = */
460 {
461 "%dead_int = OpConstant %int 2",
462 "%dead_outer_struct = OpConstantComposite %outer_struct %used_inner_struct %dead_int %used_double"
463 },
464 },
465 // A nesting struct case. The outer struct is used, so the inner struct should not
466 // be removed even though it is not used anywhere.
467 {
468 /* .used_const = */
469 {
470 "%used_bool = OpConstantTrue %bool",
471 "%used_int = OpConstant %int 1",
472 "%used_float = OpConstant %float 1.25",
473 "%used_double = OpConstant %double 1.23456789012345",
474 "%used_inner_struct = OpConstantComposite %inner_struct %used_bool %used_int %used_float %used_double",
475 "%used_outer_struct = OpConstantComposite %outer_struct %used_inner_struct %used_int %used_double"
476 },
477 /* .main_insts = */
478 {
479 "%int_var = OpVariable %_pf_int Function",
480 "%int_from_outer_struct = OpCompositeExtract %int %used_outer_struct 1",
481 "OpStore %int_var %int_from_outer_struct",
482 },
483 /* .dead_consts = */ {},
484 },
485 // clang-format on
486 })));
487
488 INSTANTIATE_TEST_SUITE_P(
489 ScalarTypeSpecConstants, EliminateDeadConstantTest,
490 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
491 // clang-format off
492 // All scalar type spec constants.
493 {
494 /* .used_consts = */
495 {
496 "%used_bool = OpSpecConstantTrue %bool",
497 "%used_uint = OpSpecConstant %uint 2",
498 "%used_int = OpSpecConstant %int 2",
499 "%used_float = OpSpecConstant %float 2.5",
500 "%used_double = OpSpecConstant %double 1.42857142851",
501 },
502 /* .main_insts = */
503 {
504 "%bool_var = OpVariable %_pf_bool Function",
505 "%uint_var = OpVariable %_pf_uint Function",
506 "%int_var = OpVariable %_pf_int Function",
507 "%float_var = OpVariable %_pf_float Function",
508 "%double_var = OpVariable %_pf_double Function",
509 "OpStore %bool_var %used_bool", "OpStore %uint_var %used_uint",
510 "OpStore %int_var %used_int", "OpStore %float_var %used_float",
511 "OpStore %double_var %used_double",
512 },
513 /* .dead_consts = */
514 {
515 "%dead_bool = OpSpecConstantTrue %bool",
516 "%dead_uint = OpSpecConstant %uint 2",
517 "%dead_int = OpSpecConstant %int 2",
518 "%dead_float = OpSpecConstant %float 2.5",
519 "%dead_double = OpSpecConstant %double 1.42857142851",
520 },
521 },
522 // clang-format on
523 })));
524
525 INSTANTIATE_TEST_SUITE_P(
526 VectorTypeSpecConstants, EliminateDeadConstantTest,
527 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
528 // clang-format off
529 // Bool vector type spec constants. One vector has all component dead,
530 // another vector has one dead boolean and one used boolean.
531 {
532 /* .used_consts = */
533 {
534 "%used_bool = OpSpecConstantTrue %bool",
535 },
536 /* .main_insts = */
537 {
538 "%bool_var = OpVariable %_pf_bool Function",
539 "OpStore %bool_var %used_bool",
540 },
541 /* .dead_consts = */
542 {
543 "%dead_bool = OpSpecConstantFalse %bool",
544 "%dead_bool_vec1 = OpSpecConstantComposite %v2bool %dead_bool %dead_bool",
545 "%dead_bool_vec2 = OpSpecConstantComposite %v2bool %dead_bool %used_bool",
546 },
547 },
548
549 // Uint vector type spec constants. One vector has all component dead,
550 // another vector has one dead unsigend integer and one used unsigned
551 // integer.
552 {
553 /* .used_consts = */
554 {
555 "%used_uint = OpSpecConstant %uint 3",
556 },
557 /* .main_insts = */
558 {
559 "%uint_var = OpVariable %_pf_uint Function",
560 "OpStore %uint_var %used_uint",
561 },
562 /* .dead_consts = */
563 {
564 "%dead_uint = OpSpecConstant %uint 1",
565 "%dead_uint_vec1 = OpSpecConstantComposite %v2uint %dead_uint %dead_uint",
566 "%dead_uint_vec2 = OpSpecConstantComposite %v2uint %dead_uint %used_uint",
567 },
568 },
569
570 // Int vector type spec constants. One vector has all component dead,
571 // another vector has one dead integer and one used integer.
572 {
573 /* .used_consts = */
574 {
575 "%used_int = OpSpecConstant %int 3",
576 },
577 /* .main_insts = */
578 {
579 "%int_var = OpVariable %_pf_int Function",
580 "OpStore %int_var %used_int",
581 },
582 /* .dead_consts = */
583 {
584 "%dead_int = OpSpecConstant %int 1",
585 "%dead_int_vec1 = OpSpecConstantComposite %v2int %dead_int %dead_int",
586 "%dead_int_vec2 = OpSpecConstantComposite %v2int %dead_int %used_int",
587 },
588 },
589
590 // Int vector type spec constants built with both spec constants and
591 // front-end constants.
592 {
593 /* .used_consts = */
594 {
595 "%used_spec_int = OpSpecConstant %int 3",
596 "%used_front_end_int = OpConstant %int 3",
597 },
598 /* .main_insts = */
599 {
600 "%int_var1 = OpVariable %_pf_int Function",
601 "OpStore %int_var1 %used_spec_int",
602 "%int_var2 = OpVariable %_pf_int Function",
603 "OpStore %int_var2 %used_front_end_int",
604 },
605 /* .dead_consts = */
606 {
607 "%dead_spec_int = OpSpecConstant %int 1",
608 "%dead_front_end_int = OpConstant %int 1",
609 // Dead front-end and dead spec constants
610 "%dead_int_vec1 = OpSpecConstantComposite %v2int %dead_spec_int %dead_front_end_int",
611 // Used front-end and dead spec constants
612 "%dead_int_vec2 = OpSpecConstantComposite %v2int %dead_spec_int %used_front_end_int",
613 // Dead front-end and used spec constants
614 "%dead_int_vec3 = OpSpecConstantComposite %v2int %dead_front_end_int %used_spec_int",
615 },
616 },
617 // clang-format on
618 })));
619
620 INSTANTIATE_TEST_SUITE_P(
621 SpecConstantOp, EliminateDeadConstantTest,
622 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
623 // clang-format off
624 // Cast operations: uint <-> int <-> bool
625 {
626 /* .used_consts = */ {},
627 /* .main_insts = */ {},
628 /* .dead_consts = */
629 {
630 // Assistant constants, only used in dead spec constant
631 // operations.
632 "%signed_zero = OpConstant %int 0",
633 "%signed_zero_vec = OpConstantComposite %v2int %signed_zero %signed_zero",
634 "%unsigned_zero = OpConstant %uint 0",
635 "%unsigned_zero_vec = OpConstantComposite %v2uint %unsigned_zero %unsigned_zero",
636 "%signed_one = OpConstant %int 1",
637 "%signed_one_vec = OpConstantComposite %v2int %signed_one %signed_one",
638 "%unsigned_one = OpConstant %uint 1",
639 "%unsigned_one_vec = OpConstantComposite %v2uint %unsigned_one %unsigned_one",
640
641 // Spec constants that support casting to each other.
642 "%dead_bool = OpSpecConstantTrue %bool",
643 "%dead_uint = OpSpecConstant %uint 1",
644 "%dead_int = OpSpecConstant %int 2",
645 "%dead_bool_vec = OpSpecConstantComposite %v2bool %dead_bool %dead_bool",
646 "%dead_uint_vec = OpSpecConstantComposite %v2uint %dead_uint %dead_uint",
647 "%dead_int_vec = OpSpecConstantComposite %v2int %dead_int %dead_int",
648
649 // Scalar cast to boolean spec constant.
650 "%int_to_bool = OpSpecConstantOp %bool INotEqual %dead_int %signed_zero",
651 "%uint_to_bool = OpSpecConstantOp %bool INotEqual %dead_uint %unsigned_zero",
652
653 // Vector cast to boolean spec constant.
654 "%int_to_bool_vec = OpSpecConstantOp %v2bool INotEqual %dead_int_vec %signed_zero_vec",
655 "%uint_to_bool_vec = OpSpecConstantOp %v2bool INotEqual %dead_uint_vec %unsigned_zero_vec",
656
657 // Scalar cast to int spec constant.
658 "%bool_to_int = OpSpecConstantOp %int Select %dead_bool %signed_one %signed_zero",
659 "%uint_to_int = OpSpecConstantOp %uint IAdd %dead_uint %unsigned_zero",
660
661 // Vector cast to int spec constant.
662 "%bool_to_int_vec = OpSpecConstantOp %v2int Select %dead_bool_vec %signed_one_vec %signed_zero_vec",
663 "%uint_to_int_vec = OpSpecConstantOp %v2uint IAdd %dead_uint_vec %unsigned_zero_vec",
664
665 // Scalar cast to uint spec constant.
666 "%bool_to_uint = OpSpecConstantOp %uint Select %dead_bool %unsigned_one %unsigned_zero",
667 "%int_to_uint_vec = OpSpecConstantOp %uint IAdd %dead_int %signed_zero",
668
669 // Vector cast to uint spec constant.
670 "%bool_to_uint_vec = OpSpecConstantOp %v2uint Select %dead_bool_vec %unsigned_one_vec %unsigned_zero_vec",
671 "%int_to_uint = OpSpecConstantOp %v2uint IAdd %dead_int_vec %signed_zero_vec",
672 },
673 },
674
675 // Add, sub, mul, div, rem.
676 {
677 /* .used_consts = */ {},
678 /* .main_insts = */ {},
679 /* .dead_consts = */
680 {
681 "%dead_spec_int_a = OpSpecConstant %int 1",
682 "%dead_spec_int_a_vec = OpSpecConstantComposite %v2int %dead_spec_int_a %dead_spec_int_a",
683
684 "%dead_spec_int_b = OpSpecConstant %int 2",
685 "%dead_spec_int_b_vec = OpSpecConstantComposite %v2int %dead_spec_int_b %dead_spec_int_b",
686
687 "%dead_const_int_c = OpConstant %int 3",
688 "%dead_const_int_c_vec = OpConstantComposite %v2int %dead_const_int_c %dead_const_int_c",
689
690 // Add
691 "%add_a_b = OpSpecConstantOp %int IAdd %dead_spec_int_a %dead_spec_int_b",
692 "%add_a_b_vec = OpSpecConstantOp %v2int IAdd %dead_spec_int_a_vec %dead_spec_int_b_vec",
693
694 // Sub
695 "%sub_a_b = OpSpecConstantOp %int ISub %dead_spec_int_a %dead_spec_int_b",
696 "%sub_a_b_vec = OpSpecConstantOp %v2int ISub %dead_spec_int_a_vec %dead_spec_int_b_vec",
697
698 // Mul
699 "%mul_a_b = OpSpecConstantOp %int IMul %dead_spec_int_a %dead_spec_int_b",
700 "%mul_a_b_vec = OpSpecConstantOp %v2int IMul %dead_spec_int_a_vec %dead_spec_int_b_vec",
701
702 // Div
703 "%div_a_b = OpSpecConstantOp %int SDiv %dead_spec_int_a %dead_spec_int_b",
704 "%div_a_b_vec = OpSpecConstantOp %v2int SDiv %dead_spec_int_a_vec %dead_spec_int_b_vec",
705
706 // Bitwise Xor
707 "%xor_a_b = OpSpecConstantOp %int BitwiseXor %dead_spec_int_a %dead_spec_int_b",
708 "%xor_a_b_vec = OpSpecConstantOp %v2int BitwiseXor %dead_spec_int_a_vec %dead_spec_int_b_vec",
709
710 // Scalar Comparison
711 "%less_a_b = OpSpecConstantOp %bool SLessThan %dead_spec_int_a %dead_spec_int_b",
712 },
713 },
714
715 // Vectors without used swizzles should be removed.
716 {
717 /* .used_consts = */
718 {
719 "%used_int = OpConstant %int 3",
720 },
721 /* .main_insts = */
722 {
723 "%int_var = OpVariable %_pf_int Function",
724 "OpStore %int_var %used_int",
725 },
726 /* .dead_consts = */
727 {
728 "%dead_int = OpConstant %int 3",
729
730 "%dead_spec_int_a = OpSpecConstant %int 1",
731 "%vec_a = OpSpecConstantComposite %v4int %dead_spec_int_a %dead_spec_int_a %dead_int %dead_int",
732
733 "%dead_spec_int_b = OpSpecConstant %int 2",
734 "%vec_b = OpSpecConstantComposite %v4int %dead_spec_int_b %dead_spec_int_b %used_int %used_int",
735
736 // Extract scalar
737 "%a_x = OpSpecConstantOp %int CompositeExtract %vec_a 0",
738 "%b_x = OpSpecConstantOp %int CompositeExtract %vec_b 0",
739
740 // Extract vector
741 "%a_xy = OpSpecConstantOp %v2int VectorShuffle %vec_a %vec_a 0 1",
742 "%b_xy = OpSpecConstantOp %v2int VectorShuffle %vec_b %vec_b 0 1",
743 },
744 },
745 // Vectors with used swizzles should not be removed.
746 {
747 /* .used_consts = */
748 {
749 "%used_int = OpConstant %int 3",
750 "%used_spec_int_a = OpSpecConstant %int 1",
751 "%used_spec_int_b = OpSpecConstant %int 2",
752 // Create vectors
753 "%vec_a = OpSpecConstantComposite %v4int %used_spec_int_a %used_spec_int_a %used_int %used_int",
754 "%vec_b = OpSpecConstantComposite %v4int %used_spec_int_b %used_spec_int_b %used_int %used_int",
755 // Extract vector
756 "%a_xy = OpSpecConstantOp %v2int VectorShuffle %vec_a %vec_a 0 1",
757 "%b_xy = OpSpecConstantOp %v2int VectorShuffle %vec_b %vec_b 0 1",
758 },
759 /* .main_insts = */
760 {
761 "%v2int_var_a = OpVariable %_pf_v2int Function",
762 "%v2int_var_b = OpVariable %_pf_v2int Function",
763 "OpStore %v2int_var_a %a_xy",
764 "OpStore %v2int_var_b %b_xy",
765 },
766 /* .dead_consts = */ {},
767 },
768 // clang-format on
769 })));
770
771 INSTANTIATE_TEST_SUITE_P(
772 LongDefUseChain, EliminateDeadConstantTest,
773 ::testing::ValuesIn(std::vector<EliminateDeadConstantTestCase>({
774 // clang-format off
775 // Long Def-Use chain with binary operations.
776 {
777 /* .used_consts = */
778 {
779 "%array_size = OpConstant %int 4",
780 "%type_arr_int_4 = OpTypeArray %int %array_size",
781 "%used_int_0 = OpConstant %int 100",
782 "%used_int_1 = OpConstant %int 1",
783 "%used_int_2 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_1",
784 "%used_int_3 = OpSpecConstantOp %int ISub %used_int_0 %used_int_2",
785 "%used_int_4 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_3",
786 "%used_int_5 = OpSpecConstantOp %int ISub %used_int_0 %used_int_4",
787 "%used_int_6 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_5",
788 "%used_int_7 = OpSpecConstantOp %int ISub %used_int_0 %used_int_6",
789 "%used_int_8 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_7",
790 "%used_int_9 = OpSpecConstantOp %int ISub %used_int_0 %used_int_8",
791 "%used_int_10 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_9",
792 "%used_int_11 = OpSpecConstantOp %int ISub %used_int_0 %used_int_10",
793 "%used_int_12 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_11",
794 "%used_int_13 = OpSpecConstantOp %int ISub %used_int_0 %used_int_12",
795 "%used_int_14 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_13",
796 "%used_int_15 = OpSpecConstantOp %int ISub %used_int_0 %used_int_14",
797 "%used_int_16 = OpSpecConstantOp %int ISub %used_int_0 %used_int_15",
798 "%used_int_17 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_16",
799 "%used_int_18 = OpSpecConstantOp %int ISub %used_int_0 %used_int_17",
800 "%used_int_19 = OpSpecConstantOp %int IAdd %used_int_0 %used_int_18",
801 "%used_int_20 = OpSpecConstantOp %int ISub %used_int_0 %used_int_19",
802 "%used_vec_a = OpSpecConstantComposite %v2int %used_int_18 %used_int_19",
803 "%used_vec_b = OpSpecConstantOp %v2int IMul %used_vec_a %used_vec_a",
804 "%used_int_21 = OpSpecConstantOp %int CompositeExtract %used_vec_b 0",
805 "%used_array = OpConstantComposite %type_arr_int_4 %used_int_20 %used_int_20 %used_int_21 %used_int_21",
806 },
807 /* .main_insts = */
808 {
809 "%int_var = OpVariable %_pf_int Function",
810 "%used_array_2 = OpCompositeExtract %int %used_array 2",
811 "OpStore %int_var %used_array_2",
812 },
813 /* .dead_consts = */
814 {
815 "%dead_int_1 = OpConstant %int 2",
816 "%dead_int_2 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_1",
817 "%dead_int_3 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_2",
818 "%dead_int_4 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_3",
819 "%dead_int_5 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_4",
820 "%dead_int_6 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_5",
821 "%dead_int_7 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_6",
822 "%dead_int_8 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_7",
823 "%dead_int_9 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_8",
824 "%dead_int_10 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_9",
825 "%dead_int_11 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_10",
826 "%dead_int_12 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_11",
827 "%dead_int_13 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_12",
828 "%dead_int_14 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_13",
829 "%dead_int_15 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_14",
830 "%dead_int_16 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_15",
831 "%dead_int_17 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_16",
832 "%dead_int_18 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_17",
833 "%dead_int_19 = OpSpecConstantOp %int IAdd %used_int_0 %dead_int_18",
834 "%dead_int_20 = OpSpecConstantOp %int ISub %used_int_0 %dead_int_19",
835 "%dead_vec_a = OpSpecConstantComposite %v2int %dead_int_18 %dead_int_19",
836 "%dead_vec_b = OpSpecConstantOp %v2int IMul %dead_vec_a %dead_vec_a",
837 "%dead_int_21 = OpSpecConstantOp %int CompositeExtract %dead_vec_b 0",
838 "%dead_array = OpConstantComposite %type_arr_int_4 %dead_int_20 %used_int_20 %dead_int_19 %used_int_19",
839 },
840 },
841 // Long Def-Use chain with swizzle
842 // clang-format on
843 })));
844
845 } // namespace
846 } // namespace opt
847 } // namespace spvtools
848