| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531 |
- // Copyright (c) 2018 Google LLC
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- #include "source/opt/struct_cfg_analysis.h"
- #include <string>
- #include "gmock/gmock.h"
- #include "test/opt/assembly_builder.h"
- #include "test/opt/pass_fixture.h"
- #include "test/opt/pass_utils.h"
- namespace spvtools {
- namespace opt {
- namespace {
- using StructCFGAnalysisTest = PassTest<::testing::Test>;
- using ::testing::UnorderedElementsAre;
- TEST_F(StructCFGAnalysisTest, BBInSelection) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The header is not in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // BB2 is in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The merge node is not in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- }
- TEST_F(StructCFGAnalysisTest, BBInLoop) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpLoopMerge %3 %4 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpBranch %3
- %4 = OpLabel
- OpBranch %1
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The header is not in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // BB2 is in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 1);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 1);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The merge node is not in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The continue block is in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 1);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 1);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(4));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_TRUE(analysis.IsInContinueConstruct(4));
- EXPECT_FALSE(analysis.IsMergeBlock(4));
- }
- TEST_F(StructCFGAnalysisTest, SelectionInLoop) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpLoopMerge %3 %4 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpSelectionMerge %6 None
- OpBranchConditional %undef_bool %5 %6
- %5 = OpLabel
- OpBranch %6
- %6 = OpLabel
- OpBranch %3
- %4 = OpLabel
- OpBranch %1
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The loop header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // Selection header is in the loop only.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 1);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 1);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The loop merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The continue block is in the loop only.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 1);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 1);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(4));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_TRUE(analysis.IsInContinueConstruct(4));
- EXPECT_FALSE(analysis.IsMergeBlock(4));
- // BB5 is in the selection and the loop.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 1);
- EXPECT_EQ(analysis.MergeBlock(5), 6);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 1);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(5));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_FALSE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- // The selection merge is in the loop only.
- EXPECT_EQ(analysis.ContainingConstruct(6), 1);
- EXPECT_EQ(analysis.ContainingLoop(6), 1);
- EXPECT_EQ(analysis.MergeBlock(6), 3);
- EXPECT_EQ(analysis.NestingDepth(6), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(6), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(6), 1);
- EXPECT_EQ(analysis.ContainingSwitch(6), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(6), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(6));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(6));
- EXPECT_FALSE(analysis.IsInContinueConstruct(6));
- EXPECT_TRUE(analysis.IsMergeBlock(6));
- }
- TEST_F(StructCFGAnalysisTest, LoopInSelection) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpLoopMerge %4 %5 None
- OpBranchConditional %undef_bool %4 %6
- %5 = OpLabel
- OpBranch %2
- %6 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The selection header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // Loop header is in the selection only.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The selection merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The loop merge is in the selection only.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 0);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 0);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_FALSE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // The loop continue target is in the loop.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 2);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 1);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(5));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_TRUE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- // BB6 is in the loop.
- EXPECT_EQ(analysis.ContainingConstruct(6), 2);
- EXPECT_EQ(analysis.ContainingLoop(6), 2);
- EXPECT_EQ(analysis.MergeBlock(6), 4);
- EXPECT_EQ(analysis.NestingDepth(6), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(6), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(6), 1);
- EXPECT_EQ(analysis.ContainingSwitch(6), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(6), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(6));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(6));
- EXPECT_FALSE(analysis.IsInContinueConstruct(6));
- EXPECT_FALSE(analysis.IsMergeBlock(6));
- }
- TEST_F(StructCFGAnalysisTest, SelectionInSelection) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpSelectionMerge %4 None
- OpBranchConditional %undef_bool %4 %5
- %5 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The outer selection header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // The inner header is in the outer selection.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The outer merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The inner merge is in the outer selection.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 0);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 0);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_FALSE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // BB5 is in the inner selection.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 0);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 0);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(5));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_FALSE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- }
- TEST_F(StructCFGAnalysisTest, LoopInLoop) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpLoopMerge %3 %7 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpLoopMerge %4 %5 None
- OpBranchConditional %undef_bool %4 %6
- %5 = OpLabel
- OpBranch %2
- %6 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %3
- %7 = OpLabel
- OpBranch %1
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The outer loop header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // The inner loop header is in the outer loop.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 1);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 1);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The outer merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The inner merge is in the outer loop.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 1);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 1);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_FALSE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // The inner continue target is in the inner loop.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 2);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 2);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(5));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_TRUE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- // BB6 is in the loop.
- EXPECT_EQ(analysis.ContainingConstruct(6), 2);
- EXPECT_EQ(analysis.ContainingLoop(6), 2);
- EXPECT_EQ(analysis.MergeBlock(6), 4);
- EXPECT_EQ(analysis.NestingDepth(6), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(6), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(6), 2);
- EXPECT_EQ(analysis.ContainingSwitch(6), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(6), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(6));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(6));
- EXPECT_FALSE(analysis.IsInContinueConstruct(6));
- EXPECT_FALSE(analysis.IsMergeBlock(6));
- // The outer continue target is in the outer loop.
- EXPECT_EQ(analysis.ContainingConstruct(7), 1);
- EXPECT_EQ(analysis.ContainingLoop(7), 1);
- EXPECT_EQ(analysis.MergeBlock(7), 3);
- EXPECT_EQ(analysis.NestingDepth(7), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(7), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(7), 1);
- EXPECT_EQ(analysis.ContainingSwitch(7), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(7), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(7));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(7));
- EXPECT_TRUE(analysis.IsInContinueConstruct(7));
- EXPECT_FALSE(analysis.IsMergeBlock(7));
- }
- TEST_F(StructCFGAnalysisTest, KernelTest) {
- const std::string text = R"(
- OpCapability Kernel
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %1 = OpLabel
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // No structured control flow, so none of the basic block are in any
- // construct.
- for (uint32_t i = 1; i <= 3; i++) {
- EXPECT_EQ(analysis.ContainingConstruct(i), 0);
- EXPECT_EQ(analysis.ContainingLoop(i), 0);
- EXPECT_EQ(analysis.MergeBlock(i), 0);
- EXPECT_EQ(analysis.NestingDepth(i), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(i), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(i), 0);
- EXPECT_EQ(analysis.ContainingSwitch(i), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(i), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(i));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(i));
- EXPECT_FALSE(analysis.IsInContinueConstruct(i));
- EXPECT_FALSE(analysis.IsMergeBlock(i));
- }
- }
- TEST_F(StructCFGAnalysisTest, EmptyFunctionTest) {
- const std::string text = R"(
- OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- OpDecorate %func LinkageAttributes "x" Import
- %void = OpTypeVoid
- %void_fn = OpTypeFunction %void
- %func = OpFunction %void None %void_fn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- // #2451: This segfaulted on empty functions.
- StructuredCFGAnalysis analysis(context.get());
- }
- TEST_F(StructCFGAnalysisTest, BBInSwitch) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpSwitch %uint_undef %2 0 %3
- %2 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The header is not in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // BB2 is in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 1);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 3);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The merge node is not in the construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- }
- TEST_F(StructCFGAnalysisTest, LoopInSwitch) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpSwitch %uint_undef %2 1 %3
- %2 = OpLabel
- OpLoopMerge %4 %5 None
- OpBranchConditional %undef_bool %4 %6
- %5 = OpLabel
- OpBranch %2
- %6 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The selection header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // Loop header is in the selection only.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 1);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 3);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The selection merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The loop merge is in the selection only.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 0);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 0);
- EXPECT_EQ(analysis.ContainingSwitch(4), 1);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 3);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_FALSE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // The loop continue target is in the loop.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 2);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 1);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(5));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_TRUE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- // BB6 is in the loop.
- EXPECT_EQ(analysis.ContainingConstruct(6), 2);
- EXPECT_EQ(analysis.ContainingLoop(6), 2);
- EXPECT_EQ(analysis.MergeBlock(6), 4);
- EXPECT_EQ(analysis.NestingDepth(6), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(6), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(6), 1);
- EXPECT_EQ(analysis.ContainingSwitch(6), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(6), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(6));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(6));
- EXPECT_FALSE(analysis.IsInContinueConstruct(6));
- EXPECT_FALSE(analysis.IsMergeBlock(6));
- }
- TEST_F(StructCFGAnalysisTest, SelectionInSwitch) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpSwitch %uint_undef %2 10 %3
- %2 = OpLabel
- OpSelectionMerge %4 None
- OpBranchConditional %undef_bool %4 %5
- %5 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The outer selection header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // The inner header is in the outer selection.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 1);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 3);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The outer merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The inner merge is in the outer selection.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 0);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 0);
- EXPECT_EQ(analysis.ContainingSwitch(4), 1);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 3);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_FALSE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // BB5 is in the inner selection.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 0);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 0);
- EXPECT_EQ(analysis.ContainingSwitch(5), 1);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 3);
- EXPECT_FALSE(analysis.IsContinueBlock(5));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_FALSE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- }
- TEST_F(StructCFGAnalysisTest, SwitchInSelection) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpSelectionMerge %3 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpSelectionMerge %4 None
- OpSwitch %uint_undef %4 7 %5
- %5 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The outer selection header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // The inner header is in the outer selection.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 0);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 0);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The outer merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The inner merge is in the outer selection.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 0);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 0);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_FALSE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // BB5 is in the inner selection.
- EXPECT_EQ(analysis.ContainingConstruct(5), 2);
- EXPECT_EQ(analysis.ContainingLoop(5), 0);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 0);
- EXPECT_EQ(analysis.ContainingSwitch(5), 2);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 4);
- EXPECT_FALSE(analysis.IsContinueBlock(5));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_FALSE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- }
- TEST_F(StructCFGAnalysisTest, SelectionInContinue) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpLoopMerge %3 %4 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpBranch %3
- %4 = OpLabel
- OpSelectionMerge %6 None
- OpBranchConditional %undef_bool %5 %6
- %5 = OpLabel
- OpBranch %6
- %6 = OpLabel
- OpBranch %1
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The loop header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // Selection header is in the loop only.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 1);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 1);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The loop merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The continue block is in the loop only.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 1);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 1);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(4));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_TRUE(analysis.IsInContinueConstruct(4));
- EXPECT_FALSE(analysis.IsMergeBlock(4));
- // BB5 is in the selection and the continue for the loop.
- EXPECT_EQ(analysis.ContainingConstruct(5), 4);
- EXPECT_EQ(analysis.ContainingLoop(5), 1);
- EXPECT_EQ(analysis.MergeBlock(5), 6);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 1);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(5));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_TRUE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- // BB5 is in the continue for the loop.
- EXPECT_EQ(analysis.ContainingConstruct(6), 1);
- EXPECT_EQ(analysis.ContainingLoop(6), 1);
- EXPECT_EQ(analysis.MergeBlock(6), 3);
- EXPECT_EQ(analysis.NestingDepth(6), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(6), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(6), 1);
- EXPECT_EQ(analysis.ContainingSwitch(6), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(6), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(6));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(6));
- EXPECT_TRUE(analysis.IsInContinueConstruct(6));
- EXPECT_TRUE(analysis.IsMergeBlock(6));
- }
- TEST_F(StructCFGAnalysisTest, LoopInContinue) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %main "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %bool_undef = OpUndef %bool
- %uint = OpTypeInt 32 0
- %uint_undef = OpUndef %uint
- %void_func = OpTypeFunction %void
- %main = OpFunction %void None %void_func
- %entry_lab = OpLabel
- OpBranch %1
- %1 = OpLabel
- OpLoopMerge %3 %7 None
- OpBranchConditional %undef_bool %2 %3
- %2 = OpLabel
- OpBranchConditional %undef_bool %3 %7
- %7 = OpLabel
- OpLoopMerge %4 %5 None
- OpBranchConditional %undef_bool %4 %6
- %5 = OpLabel
- OpBranch %7
- %6 = OpLabel
- OpBranch %4
- %4 = OpLabel
- OpBranch %1
- %3 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- // The outer loop header is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(1), 0);
- EXPECT_EQ(analysis.ContainingLoop(1), 0);
- EXPECT_EQ(analysis.MergeBlock(1), 0);
- EXPECT_EQ(analysis.NestingDepth(1), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(1), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(1), 0);
- EXPECT_EQ(analysis.ContainingSwitch(1), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(1), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(1));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(1));
- EXPECT_FALSE(analysis.IsInContinueConstruct(1));
- EXPECT_FALSE(analysis.IsMergeBlock(1));
- // BB2 is a regular block in the inner loop.
- EXPECT_EQ(analysis.ContainingConstruct(2), 1);
- EXPECT_EQ(analysis.ContainingLoop(2), 1);
- EXPECT_EQ(analysis.MergeBlock(2), 3);
- EXPECT_EQ(analysis.NestingDepth(2), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(2), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(2), 1);
- EXPECT_EQ(analysis.ContainingSwitch(2), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(2), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(2));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(2));
- EXPECT_FALSE(analysis.IsInContinueConstruct(2));
- EXPECT_FALSE(analysis.IsMergeBlock(2));
- // The outer merge node is not in either construct.
- EXPECT_EQ(analysis.ContainingConstruct(3), 0);
- EXPECT_EQ(analysis.ContainingLoop(3), 0);
- EXPECT_EQ(analysis.MergeBlock(3), 0);
- EXPECT_EQ(analysis.NestingDepth(3), 0);
- EXPECT_EQ(analysis.LoopMergeBlock(3), 0);
- EXPECT_EQ(analysis.LoopNestingDepth(3), 0);
- EXPECT_EQ(analysis.ContainingSwitch(3), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(3), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(3));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(3));
- EXPECT_FALSE(analysis.IsInContinueConstruct(3));
- EXPECT_TRUE(analysis.IsMergeBlock(3));
- // The inner merge is in the continue of the outer loop.
- EXPECT_EQ(analysis.ContainingConstruct(4), 1);
- EXPECT_EQ(analysis.ContainingLoop(4), 1);
- EXPECT_EQ(analysis.MergeBlock(4), 3);
- EXPECT_EQ(analysis.NestingDepth(4), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(4), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(4), 1);
- EXPECT_EQ(analysis.ContainingSwitch(4), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(4), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(4));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(4));
- EXPECT_TRUE(analysis.IsInContinueConstruct(4));
- EXPECT_TRUE(analysis.IsMergeBlock(4));
- // The inner continue target is in the inner loop.
- EXPECT_EQ(analysis.ContainingConstruct(5), 7);
- EXPECT_EQ(analysis.ContainingLoop(5), 7);
- EXPECT_EQ(analysis.MergeBlock(5), 4);
- EXPECT_EQ(analysis.NestingDepth(5), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(5), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(5), 2);
- EXPECT_EQ(analysis.ContainingSwitch(5), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(5), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(5));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(5));
- EXPECT_TRUE(analysis.IsInContinueConstruct(5));
- EXPECT_FALSE(analysis.IsMergeBlock(5));
- // BB6 is a regular block in the inner loop.
- EXPECT_EQ(analysis.ContainingConstruct(6), 7);
- EXPECT_EQ(analysis.ContainingLoop(6), 7);
- EXPECT_EQ(analysis.MergeBlock(6), 4);
- EXPECT_EQ(analysis.NestingDepth(6), 2);
- EXPECT_EQ(analysis.LoopMergeBlock(6), 4);
- EXPECT_EQ(analysis.LoopNestingDepth(6), 2);
- EXPECT_EQ(analysis.ContainingSwitch(6), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(6), 0);
- EXPECT_FALSE(analysis.IsContinueBlock(6));
- EXPECT_FALSE(analysis.IsInContainingLoopsContinueConstruct(6));
- EXPECT_TRUE(analysis.IsInContinueConstruct(6));
- EXPECT_FALSE(analysis.IsMergeBlock(6));
- // The outer continue target is in the outer loop.
- EXPECT_EQ(analysis.ContainingConstruct(7), 1);
- EXPECT_EQ(analysis.ContainingLoop(7), 1);
- EXPECT_EQ(analysis.MergeBlock(7), 3);
- EXPECT_EQ(analysis.NestingDepth(7), 1);
- EXPECT_EQ(analysis.LoopMergeBlock(7), 3);
- EXPECT_EQ(analysis.LoopNestingDepth(7), 1);
- EXPECT_EQ(analysis.ContainingSwitch(7), 0);
- EXPECT_EQ(analysis.SwitchMergeBlock(7), 0);
- EXPECT_TRUE(analysis.IsContinueBlock(7));
- EXPECT_TRUE(analysis.IsInContainingLoopsContinueConstruct(7));
- EXPECT_TRUE(analysis.IsInContinueConstruct(7));
- EXPECT_FALSE(analysis.IsMergeBlock(7));
- }
- TEST_F(StructCFGAnalysisTest, FuncCallInContinueDirect) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %1 "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %4 = OpUndef %bool
- %uint = OpTypeInt 32 0
- %6 = OpUndef %uint
- %7 = OpTypeFunction %void
- %1 = OpFunction %void None %7
- %8 = OpLabel
- OpBranch %9
- %9 = OpLabel
- OpLoopMerge %10 %11 None
- OpBranchConditional %12 %10 %11
- %11 = OpLabel
- %13 = OpFunctionCall %void %14
- OpBranch %9
- %10 = OpLabel
- %15 = OpFunctionCall %void %16
- OpReturn
- OpFunctionEnd
- %14 = OpFunction %void None %7
- %17 = OpLabel
- OpReturn
- OpFunctionEnd
- %16 = OpFunction %void None %7
- %18 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- auto c = analysis.FindFuncsCalledFromContinue();
- EXPECT_THAT(c, UnorderedElementsAre(14u));
- }
- TEST_F(StructCFGAnalysisTest, FuncCallInContinueIndirect) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %1 "main"
- %void = OpTypeVoid
- %bool = OpTypeBool
- %4 = OpUndef %bool
- %uint = OpTypeInt 32 0
- %6 = OpUndef %uint
- %7 = OpTypeFunction %void
- %1 = OpFunction %void None %7
- %8 = OpLabel
- OpBranch %9
- %9 = OpLabel
- OpLoopMerge %10 %11 None
- OpBranchConditional %12 %10 %11
- %11 = OpLabel
- %13 = OpFunctionCall %void %14
- OpBranch %9
- %10 = OpLabel
- %15 = OpFunctionCall %void %16
- OpReturn
- OpFunctionEnd
- %14 = OpFunction %void None %7
- %17 = OpLabel
- %19 = OpFunctionCall %void %16
- OpReturn
- OpFunctionEnd
- %16 = OpFunction %void None %7
- %18 = OpLabel
- %20 = OpFunctionCall %void %21
- OpReturn
- OpFunctionEnd
- %21 = OpFunction %void None %7
- %22 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- auto c = analysis.FindFuncsCalledFromContinue();
- EXPECT_THAT(c, UnorderedElementsAre(14u, 16u, 21u));
- }
- TEST_F(StructCFGAnalysisTest, SingleBlockLoop) {
- const std::string text = R"(
- OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- %void = OpTypeVoid
- %bool = OpTypeBool
- %undef = OpUndef %bool
- %void_fn = OpTypeFunction %void
- %main = OpFunction %void None %void_fn
- %2 = OpLabel
- OpBranch %3
- %3 = OpLabel
- OpLoopMerge %4 %3 None
- OpBranchConditional %undef %3 %4
- %4 = OpLabel
- OpReturn
- OpFunctionEnd
- )";
- std::unique_ptr<IRContext> context =
- BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
- SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
- StructuredCFGAnalysis analysis(context.get());
- EXPECT_TRUE(analysis.IsInContinueConstruct(3));
- }
- } // namespace
- } // namespace opt
- } // namespace spvtools
|