| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558 |
- // Copyright (c) 2016 Google Inc.
- //
- // 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_OPT_MODULE_H_
- #define SOURCE_OPT_MODULE_H_
- #include <functional>
- #include <memory>
- #include <string_view>
- #include <unordered_map>
- #include <utility>
- #include <vector>
- #include "source/opt/function.h"
- #include "source/opt/instruction.h"
- #include "source/opt/iterator.h"
- namespace spvtools {
- namespace opt {
- class IRContext;
- // A struct for containing the module header information.
- struct ModuleHeader {
- uint32_t magic_number;
- uint32_t version;
- uint32_t generator;
- uint32_t bound;
- uint32_t schema;
- };
- // A SPIR-V module. It contains all the information for a SPIR-V module and
- // serves as the backbone of optimization transformations.
- class Module {
- public:
- using iterator = UptrVectorIterator<Function>;
- using const_iterator = UptrVectorIterator<Function, true>;
- using inst_iterator = InstructionList::iterator;
- using const_inst_iterator = InstructionList::const_iterator;
- // Creates an empty module with zero'd header.
- Module() : header_({}), contains_debug_info_(false) {}
- // Sets the header to the given |header|.
- void SetHeader(const ModuleHeader& header) { header_ = header; }
- // Sets the Id bound. The Id bound cannot be set to 0.
- void SetIdBound(uint32_t bound) {
- assert(bound != 0);
- header_.bound = bound;
- }
- // Returns the Id bound.
- uint32_t IdBound() const { return header_.bound; }
- // Returns the current Id bound and increases it to the next available value.
- // If the id bound has already reached its maximum value, then 0 is returned.
- // The maximum value for the id bound is obtained from the context. If there
- // is none, then the minimum that limit can be according to the spir-v
- // specification.
- // TODO(1841): Update the uses to check for a 0 return value.
- uint32_t TakeNextIdBound();
- // Appends a capability instruction to this module.
- inline void AddCapability(std::unique_ptr<Instruction> c);
- // Appends an extension instruction to this module.
- inline void AddExtension(std::unique_ptr<Instruction> e);
- // Appends an extended instruction set instruction to this module.
- inline void AddExtInstImport(std::unique_ptr<Instruction> e);
- // Set the memory model for this module.
- inline void SetMemoryModel(std::unique_ptr<Instruction> m);
- // Set the sampled image addressing mode for this module.
- inline void SetSampledImageAddressMode(std::unique_ptr<Instruction> m);
- // Appends an entry point instruction to this module.
- inline void AddEntryPoint(std::unique_ptr<Instruction> e);
- // Appends an execution mode instruction to this module.
- inline void AddExecutionMode(std::unique_ptr<Instruction> e);
- // Appends a debug 1 instruction (excluding OpLine & OpNoLine) to this module.
- // "debug 1" instructions are the ones in layout section 7.a), see section
- // 2.4 Logical Layout of a Module from the SPIR-V specification.
- inline void AddDebug1Inst(std::unique_ptr<Instruction> d);
- // Appends a debug 2 instruction (excluding OpLine & OpNoLine) to this module.
- // "debug 2" instructions are the ones in layout section 7.b), see section
- // 2.4 Logical Layout of a Module from the SPIR-V specification.
- inline void AddDebug2Inst(std::unique_ptr<Instruction> d);
- // Appends a debug 3 instruction (OpModuleProcessed) to this module.
- // This is due to decision by the SPIR Working Group, pending publication.
- inline void AddDebug3Inst(std::unique_ptr<Instruction> d);
- // Appends a debug info extension (OpenCL.DebugInfo.100,
- // NonSemantic.Shader.DebugInfo.100, or DebugInfo) instruction to this module.
- inline void AddExtInstDebugInfo(std::unique_ptr<Instruction> d);
- // Appends an annotation instruction to this module.
- inline void AddAnnotationInst(std::unique_ptr<Instruction> a);
- // Appends a type-declaration instruction to this module.
- inline void AddType(std::unique_ptr<Instruction> t);
- // Appends a constant, global variable, or OpUndef instruction to this module.
- inline void AddGlobalValue(std::unique_ptr<Instruction> v);
- // Prepends a function declaration to this module.
- inline void AddFunctionDeclaration(std::unique_ptr<Function> f);
- // Appends a function to this module.
- inline void AddFunction(std::unique_ptr<Function> f);
- // Sets |contains_debug_info_| as true.
- inline void SetContainsDebugInfo();
- inline bool ContainsDebugInfo() { return contains_debug_info_; }
- // Returns a vector of pointers to type-declaration instructions in this
- // module.
- std::vector<Instruction*> GetTypes();
- std::vector<const Instruction*> GetTypes() const;
- // Returns a vector of pointers to constant-creation instructions in this
- // module.
- std::vector<Instruction*> GetConstants();
- std::vector<const Instruction*> GetConstants() const;
- // Return result id of global value with |opcode|, 0 if not present.
- uint32_t GetGlobalValue(spv::Op opcode) const;
- // Add global value with |opcode|, |result_id| and |type_id|
- void AddGlobalValue(spv::Op opcode, uint32_t result_id, uint32_t type_id);
- inline uint32_t id_bound() const { return header_.bound; }
- inline uint32_t version() const { return header_.version; }
- inline uint32_t generator() const { return header_.generator; }
- inline uint32_t schema() const { return header_.schema; }
- inline void set_version(uint32_t v) { header_.version = v; }
- // Iterators for capabilities instructions contained in this module.
- inline inst_iterator capability_begin();
- inline inst_iterator capability_end();
- inline IteratorRange<inst_iterator> capabilities();
- inline IteratorRange<const_inst_iterator> capabilities() const;
- // Iterators for ext_inst_imports instructions contained in this module.
- inline inst_iterator ext_inst_import_begin();
- inline inst_iterator ext_inst_import_end();
- inline IteratorRange<inst_iterator> ext_inst_imports();
- inline IteratorRange<const_inst_iterator> ext_inst_imports() const;
- // Return the memory model instruction contained in this module.
- inline Instruction* GetMemoryModel() { return memory_model_.get(); }
- inline const Instruction* GetMemoryModel() const {
- return memory_model_.get();
- }
- // Return the sampled image address mode instruction contained in this module.
- inline Instruction* GetSampledImageAddressMode() {
- return sampled_image_address_mode_.get();
- }
- inline const Instruction* GetSampledImageAddressMode() const {
- return sampled_image_address_mode_.get();
- }
- // There are several kinds of debug instructions, according to where they can
- // appear in the logical layout of a module:
- // - Section 7a: OpString, OpSourceExtension, OpSource, OpSourceContinued
- // - Section 7b: OpName, OpMemberName
- // - Section 7c: OpModuleProcessed
- // - Mostly anywhere: OpLine and OpNoLine
- //
- // Iterators for debug 1 instructions (excluding OpLine & OpNoLine) contained
- // in this module. These are for layout section 7a.
- inline inst_iterator debug1_begin();
- inline inst_iterator debug1_end();
- inline IteratorRange<inst_iterator> debugs1();
- inline IteratorRange<const_inst_iterator> debugs1() const;
- // Iterators for debug 2 instructions (excluding OpLine & OpNoLine) contained
- // in this module. These are for layout section 7b.
- inline inst_iterator debug2_begin();
- inline inst_iterator debug2_end();
- inline IteratorRange<inst_iterator> debugs2();
- inline IteratorRange<const_inst_iterator> debugs2() const;
- // Iterators for debug 3 instructions (excluding OpLine & OpNoLine) contained
- // in this module. These are for layout section 7c.
- inline inst_iterator debug3_begin();
- inline inst_iterator debug3_end();
- inline IteratorRange<inst_iterator> debugs3();
- inline IteratorRange<const_inst_iterator> debugs3() const;
- // Iterators for debug info instructions (excluding OpLine & OpNoLine)
- // contained in this module. These are OpExtInst for DebugInfo extension
- // placed between section 9 and 10.
- inline inst_iterator ext_inst_debuginfo_begin();
- inline inst_iterator ext_inst_debuginfo_end();
- inline IteratorRange<inst_iterator> ext_inst_debuginfo();
- inline IteratorRange<const_inst_iterator> ext_inst_debuginfo() const;
- // Iterators for entry point instructions contained in this module
- inline IteratorRange<inst_iterator> entry_points();
- inline IteratorRange<const_inst_iterator> entry_points() const;
- // Iterators for execution_modes instructions contained in this module.
- inline inst_iterator execution_mode_begin();
- inline inst_iterator execution_mode_end();
- inline IteratorRange<inst_iterator> execution_modes();
- inline IteratorRange<const_inst_iterator> execution_modes() const;
- // Iterators for annotation instructions contained in this module.
- inline inst_iterator annotation_begin();
- inline inst_iterator annotation_end();
- IteratorRange<inst_iterator> annotations();
- IteratorRange<const_inst_iterator> annotations() const;
- // Iterators for extension instructions contained in this module.
- inline inst_iterator extension_begin();
- inline inst_iterator extension_end();
- IteratorRange<inst_iterator> extensions();
- IteratorRange<const_inst_iterator> extensions() const;
- // Iterators for types, constants and global variables instructions.
- inline inst_iterator types_values_begin();
- inline inst_iterator types_values_end();
- inline IteratorRange<inst_iterator> types_values();
- inline IteratorRange<const_inst_iterator> types_values() const;
- // Iterators for functions contained in this module.
- iterator begin() { return iterator(&functions_, functions_.begin()); }
- iterator end() { return iterator(&functions_, functions_.end()); }
- const_iterator begin() const { return cbegin(); }
- const_iterator end() const { return cend(); }
- inline const_iterator cbegin() const;
- inline const_iterator cend() const;
- // Invokes function |f| on all instructions in this module, and optionally on
- // the debug line instructions that precede them.
- void ForEachInst(const std::function<void(Instruction*)>& f,
- bool run_on_debug_line_insts = false);
- void ForEachInst(const std::function<void(const Instruction*)>& f,
- bool run_on_debug_line_insts = false) const;
- // Pushes the binary segments for this instruction into the back of *|binary|.
- // If |skip_nop| is true and this is a OpNop, do nothing.
- void ToBinary(std::vector<uint32_t>* binary, bool skip_nop) const;
- // Returns 1 more than the maximum Id value mentioned in the module.
- uint32_t ComputeIdBound() const;
- // Returns true if module has capability |cap|
- bool HasExplicitCapability(uint32_t cap);
- // Returns id for OpExtInst instruction for extension |extstr|.
- // Returns 0 if not found.
- uint32_t GetExtInstImportId(const char* extstr);
- // Sets the associated context for this module
- void SetContext(IRContext* c) { context_ = c; }
- // Gets the associated context for this module
- IRContext* context() const { return context_; }
- // Sets the trailing debug line info to |dbg_line_info|.
- void SetTrailingDbgLineInfo(std::vector<Instruction>&& dbg_line_info) {
- trailing_dbg_line_info_ = std::move(dbg_line_info);
- }
- std::vector<Instruction>& trailing_dbg_line_info() {
- return trailing_dbg_line_info_;
- }
- const std::vector<Instruction>& trailing_dbg_line_info() const {
- return trailing_dbg_line_info_;
- }
- private:
- ModuleHeader header_; // Module header
- // The following fields respect the "Logical Layout of a Module" in
- // Section 2.4 of the SPIR-V specification.
- IRContext* context_;
- InstructionList capabilities_;
- InstructionList extensions_;
- InstructionList ext_inst_imports_;
- // A module only has one memory model instruction.
- std::unique_ptr<Instruction> memory_model_;
- // A module can only have one optional sampled image addressing mode
- std::unique_ptr<Instruction> sampled_image_address_mode_;
- InstructionList entry_points_;
- InstructionList execution_modes_;
- InstructionList debugs1_;
- InstructionList debugs2_;
- InstructionList debugs3_;
- InstructionList ext_inst_debuginfo_;
- InstructionList annotations_;
- // Type declarations, constants, and global variable declarations.
- InstructionList types_values_;
- std::vector<std::unique_ptr<Function>> functions_;
- // If the module ends with Op*Line instruction, they will not be attached to
- // any instruction. We record them here, so they will not be lost.
- std::vector<Instruction> trailing_dbg_line_info_;
- // This module contains DebugScope/DebugNoScope or OpLine/OpNoLine.
- bool contains_debug_info_;
- };
- // Pretty-prints |module| to |str|. Returns |str|.
- std::ostream& operator<<(std::ostream& str, const Module& module);
- inline void Module::AddCapability(std::unique_ptr<Instruction> c) {
- capabilities_.push_back(std::move(c));
- }
- inline void Module::AddExtension(std::unique_ptr<Instruction> e) {
- extensions_.push_back(std::move(e));
- }
- inline void Module::AddExtInstImport(std::unique_ptr<Instruction> e) {
- ext_inst_imports_.push_back(std::move(e));
- }
- inline void Module::SetMemoryModel(std::unique_ptr<Instruction> m) {
- memory_model_ = std::move(m);
- }
- inline void Module::SetSampledImageAddressMode(std::unique_ptr<Instruction> m) {
- sampled_image_address_mode_ = std::move(m);
- }
- inline void Module::AddEntryPoint(std::unique_ptr<Instruction> e) {
- entry_points_.push_back(std::move(e));
- }
- inline void Module::AddExecutionMode(std::unique_ptr<Instruction> e) {
- execution_modes_.push_back(std::move(e));
- }
- inline void Module::AddDebug1Inst(std::unique_ptr<Instruction> d) {
- debugs1_.push_back(std::move(d));
- }
- inline void Module::AddDebug2Inst(std::unique_ptr<Instruction> d) {
- debugs2_.push_back(std::move(d));
- }
- inline void Module::AddDebug3Inst(std::unique_ptr<Instruction> d) {
- debugs3_.push_back(std::move(d));
- }
- inline void Module::AddExtInstDebugInfo(std::unique_ptr<Instruction> d) {
- ext_inst_debuginfo_.push_back(std::move(d));
- }
- inline void Module::AddAnnotationInst(std::unique_ptr<Instruction> a) {
- annotations_.push_back(std::move(a));
- }
- inline void Module::AddType(std::unique_ptr<Instruction> t) {
- types_values_.push_back(std::move(t));
- }
- inline void Module::AddGlobalValue(std::unique_ptr<Instruction> v) {
- types_values_.push_back(std::move(v));
- }
- inline void Module::AddFunctionDeclaration(std::unique_ptr<Function> f) {
- // function declarations must come before function definitions.
- functions_.emplace(functions_.begin(), std::move(f));
- }
- inline void Module::AddFunction(std::unique_ptr<Function> f) {
- functions_.emplace_back(std::move(f));
- }
- inline void Module::SetContainsDebugInfo() { contains_debug_info_ = true; }
- inline Module::inst_iterator Module::capability_begin() {
- return capabilities_.begin();
- }
- inline Module::inst_iterator Module::capability_end() {
- return capabilities_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::capabilities() {
- return make_range(capabilities_.begin(), capabilities_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::capabilities() const {
- return make_range(capabilities_.begin(), capabilities_.end());
- }
- inline Module::inst_iterator Module::ext_inst_import_begin() {
- return ext_inst_imports_.begin();
- }
- inline Module::inst_iterator Module::ext_inst_import_end() {
- return ext_inst_imports_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::ext_inst_imports() {
- return make_range(ext_inst_imports_.begin(), ext_inst_imports_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::ext_inst_imports()
- const {
- return make_range(ext_inst_imports_.begin(), ext_inst_imports_.end());
- }
- inline Module::inst_iterator Module::debug1_begin() { return debugs1_.begin(); }
- inline Module::inst_iterator Module::debug1_end() { return debugs1_.end(); }
- inline IteratorRange<Module::inst_iterator> Module::debugs1() {
- return make_range(debugs1_.begin(), debugs1_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::debugs1() const {
- return make_range(debugs1_.begin(), debugs1_.end());
- }
- inline Module::inst_iterator Module::debug2_begin() { return debugs2_.begin(); }
- inline Module::inst_iterator Module::debug2_end() { return debugs2_.end(); }
- inline IteratorRange<Module::inst_iterator> Module::debugs2() {
- return make_range(debugs2_.begin(), debugs2_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::debugs2() const {
- return make_range(debugs2_.begin(), debugs2_.end());
- }
- inline Module::inst_iterator Module::debug3_begin() { return debugs3_.begin(); }
- inline Module::inst_iterator Module::debug3_end() { return debugs3_.end(); }
- inline IteratorRange<Module::inst_iterator> Module::debugs3() {
- return make_range(debugs3_.begin(), debugs3_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::debugs3() const {
- return make_range(debugs3_.begin(), debugs3_.end());
- }
- inline Module::inst_iterator Module::ext_inst_debuginfo_begin() {
- return ext_inst_debuginfo_.begin();
- }
- inline Module::inst_iterator Module::ext_inst_debuginfo_end() {
- return ext_inst_debuginfo_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::ext_inst_debuginfo() {
- return make_range(ext_inst_debuginfo_.begin(), ext_inst_debuginfo_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::ext_inst_debuginfo()
- const {
- return make_range(ext_inst_debuginfo_.begin(), ext_inst_debuginfo_.end());
- }
- inline IteratorRange<Module::inst_iterator> Module::entry_points() {
- return make_range(entry_points_.begin(), entry_points_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::entry_points() const {
- return make_range(entry_points_.begin(), entry_points_.end());
- }
- inline Module::inst_iterator Module::execution_mode_begin() {
- return execution_modes_.begin();
- }
- inline Module::inst_iterator Module::execution_mode_end() {
- return execution_modes_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::execution_modes() {
- return make_range(execution_modes_.begin(), execution_modes_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::execution_modes()
- const {
- return make_range(execution_modes_.begin(), execution_modes_.end());
- }
- inline Module::inst_iterator Module::annotation_begin() {
- return annotations_.begin();
- }
- inline Module::inst_iterator Module::annotation_end() {
- return annotations_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::annotations() {
- return make_range(annotations_.begin(), annotations_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::annotations() const {
- return make_range(annotations_.begin(), annotations_.end());
- }
- inline Module::inst_iterator Module::extension_begin() {
- return extensions_.begin();
- }
- inline Module::inst_iterator Module::extension_end() {
- return extensions_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::extensions() {
- return make_range(extensions_.begin(), extensions_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::extensions() const {
- return make_range(extensions_.begin(), extensions_.end());
- }
- inline Module::inst_iterator Module::types_values_begin() {
- return types_values_.begin();
- }
- inline Module::inst_iterator Module::types_values_end() {
- return types_values_.end();
- }
- inline IteratorRange<Module::inst_iterator> Module::types_values() {
- return make_range(types_values_.begin(), types_values_.end());
- }
- inline IteratorRange<Module::const_inst_iterator> Module::types_values() const {
- return make_range(types_values_.begin(), types_values_.end());
- }
- inline Module::const_iterator Module::cbegin() const {
- return const_iterator(&functions_, functions_.cbegin());
- }
- inline Module::const_iterator Module::cend() const {
- return const_iterator(&functions_, functions_.cend());
- }
- } // namespace opt
- } // namespace spvtools
- #endif // SOURCE_OPT_MODULE_H_
|