1 // Copyright (c) 2020 Vasyl Teliman 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 #ifndef SOURCE_FUZZ_TRANSFORMATION_PROPAGATE_INSTRUCTION_UP_H_ 16 #define SOURCE_FUZZ_TRANSFORMATION_PROPAGATE_INSTRUCTION_UP_H_ 17 18 #include <map> 19 20 #include "source/fuzz/protobufs/spirvfuzz_protobufs.h" 21 #include "source/fuzz/transformation.h" 22 #include "source/fuzz/transformation_context.h" 23 #include "source/opt/ir_context.h" 24 25 namespace spvtools { 26 namespace fuzz { 27 28 class TransformationPropagateInstructionUp : public Transformation { 29 public: 30 explicit TransformationPropagateInstructionUp( 31 const protobufs::TransformationPropagateInstructionUp& message); 32 33 TransformationPropagateInstructionUp( 34 uint32_t block_id, 35 const std::map<uint32_t, uint32_t>& predecessor_id_to_fresh_id); 36 37 // - |block_id| must be a valid result id of some OpLabel instruction. 38 // - |block_id| must have at least one predecessor 39 // - |block_id| must contain an instruction that can be propagated using this 40 // transformation 41 // - the instruction can be propagated if: 42 // - it's not an OpPhi 43 // - it is supported by this transformation 44 // - it depends only on instructions from different basic blocks or on 45 // OpPhi instructions from the same basic block 46 // - it should be possible to insert the propagated instruction at the end of 47 // each |block_id|'s predecessor 48 // - |predecessor_id_to_fresh_id| must have an entry for at least every 49 // predecessor of |block_id| 50 // - each value in the |predecessor_id_to_fresh_id| map must be a fresh id 51 // - all fresh ids in the |predecessor_id_to_fresh_id| must be unique 52 bool IsApplicable( 53 opt::IRContext* ir_context, 54 const TransformationContext& transformation_context) const override; 55 56 // Inserts a copy of the propagated instruction into each |block_id|'s 57 // predecessor. Replaces the original instruction with an OpPhi referring 58 // inserted copies. 59 void Apply(opt::IRContext* ir_context, 60 TransformationContext* transformation_context) const override; 61 62 std::unordered_set<uint32_t> GetFreshIds() const override; 63 64 protobufs::Transformation ToMessage() const override; 65 66 // Returns true if this transformation can be applied to the block with id 67 // |block_id|. Concretely, returns true iff: 68 // - |block_id| is a valid id of some block in the module 69 // - |block_id| has predecessors 70 // - |block_id| contains an instruction that can be propagated 71 // - it is possible to insert the propagated instruction into every 72 // |block_id|'s predecessor 73 static bool IsApplicableToBlock(opt::IRContext* ir_context, 74 uint32_t block_id); 75 76 private: 77 // Returns the instruction that will be propagated into the predecessors of 78 // the |block_id|. Returns nullptr if no such an instruction exists. 79 static opt::Instruction* GetInstructionToPropagate(opt::IRContext* ir_context, 80 uint32_t block_id); 81 82 // Returns true if |opcode| is supported by this transformation. 83 static bool IsOpcodeSupported(SpvOp opcode); 84 85 protobufs::TransformationPropagateInstructionUp message_; 86 }; 87 88 } // namespace fuzz 89 } // namespace spvtools 90 91 #endif // SOURCE_FUZZ_TRANSFORMATION_PROPAGATE_INSTRUCTION_UP_H_ 92