#include // 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. #ifndef SOURCE_FUZZ_TRANSFORMATION_REPLACE_CONSTANT_WITH_UNIFORM_H_ #define SOURCE_FUZZ_TRANSFORMATION_REPLACE_CONSTANT_WITH_UNIFORM_H_ #include "source/fuzz/id_use_descriptor.h" #include "source/fuzz/protobufs/spirvfuzz_protobufs.h" #include "source/fuzz/transformation.h" #include "source/opt/ir_context.h" namespace spvtools { namespace fuzz { class TransformationReplaceConstantWithUniform : public Transformation { public: explicit TransformationReplaceConstantWithUniform( protobufs::TransformationReplaceConstantWithUniform message); TransformationReplaceConstantWithUniform( protobufs::IdUseDescriptor id_use, protobufs::UniformBufferElementDescriptor uniform_descriptor, uint32_t fresh_id_for_access_chain, uint32_t fresh_id_for_load); // - |message_.fresh_id_for_access_chain| and |message_.fresh_id_for_load| // must be distinct fresh ids. // - |message_.uniform_descriptor| specifies a result id and a list of integer // literal indices. // As an example, suppose |message_.uniform_descriptor| is (18, [0, 1, 0]) // It is required that: // - the result id (18 in our example) is the id of some uniform variable // - the module contains an integer constant instruction corresponding to // each of the literal indices; in our example there must thus be // OpConstant instructions %A and %B say for each of 0 and 1 // - it is legitimate to index into the uniform variable using the // sequence of indices; in our example this means indexing into %18 // using the sequence %A %B %A // - the module contains a uniform pointer type corresponding to the type // of the uniform data element obtained by following these indices // - |message_.id_use_descriptor| identifies the use of some id %C. It is // required that: // - this use does indeed exist in the module // - %C is an OpConstant // - According to the fact manager, the uniform data element specified by // |message_.uniform_descriptor| holds a value with the same type and // value as %C bool IsApplicable( opt::IRContext* ir_context, const TransformationContext& transformation_context) const override; // - Introduces two new instructions: // - An access chain targeting the uniform data element specified by // |message_.uniform_descriptor|, with result id // |message_.fresh_id_for_access_chain| // - A load from this access chain, with id |message_.fresh_id_for_load| // - Replaces the id use specified by |message_.id_use_descriptor| with // |message_.fresh_id_for_load| void Apply(opt::IRContext* ir_context, TransformationContext* transformation_context) const override; std::unordered_set GetFreshIds() const override; protobufs::Transformation ToMessage() const override; private: // Helper method to create an access chain for the uniform element associated // with the transformation. std::unique_ptr MakeAccessChainInstruction( spvtools::opt::IRContext* ir_context, uint32_t constant_type_id) const; // Helper to create a load instruction. std::unique_ptr MakeLoadInstruction( spvtools::opt::IRContext* ir_context, uint32_t constant_type_id) const; // OpAccessChain and OpLoad will be inserted above the instruction returned // by this function. Returns nullptr if no such instruction is present. opt::Instruction* GetInsertBeforeInstruction( opt::IRContext* ir_context) const; protobufs::TransformationReplaceConstantWithUniform message_; }; } // namespace fuzz } // namespace spvtools #endif // SOURCE_FUZZ_TRANSFORMATION_REPLACE_CONSTANT_WITH_UNIFORM_H_