| // 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 |