|  | // Copyright (c) 2019 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/fuzz/transformation_move_block_down.h" | 
|  |  | 
|  | #include "source/opt/basic_block.h" | 
|  |  | 
|  | namespace spvtools { | 
|  | namespace fuzz { | 
|  |  | 
|  | TransformationMoveBlockDown::TransformationMoveBlockDown( | 
|  | const spvtools::fuzz::protobufs::TransformationMoveBlockDown& message) | 
|  | : message_(message) {} | 
|  |  | 
|  | TransformationMoveBlockDown::TransformationMoveBlockDown(uint32_t id) { | 
|  | message_.set_block_id(id); | 
|  | } | 
|  |  | 
|  | bool TransformationMoveBlockDown::IsApplicable( | 
|  | opt::IRContext* ir_context, const TransformationContext& /*unused*/) const { | 
|  | // Go through every block in every function, looking for a block whose id | 
|  | // matches that of the block we want to consider moving down. | 
|  | for (auto& function : *ir_context->module()) { | 
|  | for (auto block_it = function.begin(); block_it != function.end(); | 
|  | ++block_it) { | 
|  | if (block_it->id() == message_.block_id()) { | 
|  | // We have found a match. | 
|  | if (block_it == function.begin()) { | 
|  | // The block is the first one appearing in the function.  We are not | 
|  | // allowed to move this block down. | 
|  | return false; | 
|  | } | 
|  | // Record the block we would like to consider moving down. | 
|  | opt::BasicBlock* block_matching_id = &*block_it; | 
|  | if (!ir_context->GetDominatorAnalysis(&function)->IsReachable( | 
|  | block_matching_id)) { | 
|  | // The block is not reachable.  We are not allowed to move it down. | 
|  | return false; | 
|  | } | 
|  | // Now see whether there is some block following that block in program | 
|  | // order. | 
|  | ++block_it; | 
|  | if (block_it == function.end()) { | 
|  | // There is no such block; i.e., the block we are considering moving | 
|  | // is the last one in the function.  The transformation thus does not | 
|  | // apply. | 
|  | return false; | 
|  | } | 
|  | opt::BasicBlock* next_block_in_program_order = &*block_it; | 
|  | // We can move the block of interest down if and only if it does not | 
|  | // dominate the block that comes next. | 
|  | return !ir_context->GetDominatorAnalysis(&function)->Dominates( | 
|  | block_matching_id, next_block_in_program_order); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // We did not find a matching block, so the transformation is not applicable: | 
|  | // there is no relevant block to move. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void TransformationMoveBlockDown::Apply( | 
|  | opt::IRContext* ir_context, TransformationContext* /*unused*/) const { | 
|  | // Go through every block in every function, looking for a block whose id | 
|  | // matches that of the block we want to move down. | 
|  | for (auto& function : *ir_context->module()) { | 
|  | for (auto block_it = function.begin(); block_it != function.end(); | 
|  | ++block_it) { | 
|  | if (block_it->id() == message_.block_id()) { | 
|  | ++block_it; | 
|  | assert(block_it != function.end() && | 
|  | "To be able to move a block down, it needs to have a " | 
|  | "program-order successor."); | 
|  | function.MoveBasicBlockToAfter(message_.block_id(), &*block_it); | 
|  | // For performance, it is vital to keep the dominator analysis valid | 
|  | // (which due to https://github.com/KhronosGroup/SPIRV-Tools/issues/2889 | 
|  | // requires keeping the CFG analysis valid). | 
|  | ir_context->InvalidateAnalysesExceptFor( | 
|  | opt::IRContext::Analysis::kAnalysisDefUse | | 
|  | opt::IRContext::Analysis::kAnalysisCFG | | 
|  | opt::IRContext::Analysis::kAnalysisDominatorAnalysis); | 
|  |  | 
|  | return; | 
|  | } | 
|  | } | 
|  | } | 
|  | assert(false && "No block was found to move down."); | 
|  | } | 
|  |  | 
|  | protobufs::Transformation TransformationMoveBlockDown::ToMessage() const { | 
|  | protobufs::Transformation result; | 
|  | *result.mutable_move_block_down() = message_; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | }  // namespace fuzz | 
|  | }  // namespace spvtools |