// Copyright (c) 2022 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_OPT_SPREAD_VOLATILE_SEMANTICS_H_ #define SOURCE_OPT_SPREAD_VOLATILE_SEMANTICS_H_ #include "source/opt/pass.h" namespace spvtools { namespace opt { // See optimizer.hpp for documentation. class SpreadVolatileSemantics : public Pass { public: SpreadVolatileSemantics() {} const char* name() const override { return "spread-volatile-semantics"; } Status Process() override; IRContext::Analysis GetPreservedAnalyses() override { return IRContext::kAnalysisDefUse | IRContext::kAnalysisDecorations | IRContext::kAnalysisInstrToBlockMapping; } private: // Returns true if it does not have an execution model. Linkage shaders do not // have an execution model. bool HasNoExecutionModel() { return get_module()->entry_points().empty() && context()->get_feature_mgr()->HasCapability( spv::Capability::Linkage); } // Iterates interface variables and spreads the Volatile semantics if it has // load instructions for the Volatile semantics. Pass::Status SpreadVolatileSemanticsToVariables( const bool is_vk_memory_model_enabled); // Returns whether |var_id| is the result id of a target builtin variable for // the volatile semantics for |execution_model| based on the Vulkan spec // VUID-StandaloneSpirv-VulkanMemoryModel-04678 or // VUID-StandaloneSpirv-VulkanMemoryModel-04679. bool IsTargetForVolatileSemantics(uint32_t var_id, spv::ExecutionModel execution_model); // Collects interface variables that need the volatile semantics. // |is_vk_memory_model_enabled| is true if VulkanMemoryModel capability is // enabled. void CollectTargetsForVolatileSemantics( const bool is_vk_memory_model_enabled); // Reports an error if an interface variable is used by two entry points and // it needs the Volatile decoration for one but not for another. Returns true // if the error must be reported. bool HasInterfaceInConflictOfVolatileSemantics(); // Returns whether the variable whose result is |var_id| is used by a // non-volatile load or a pointer to it is used by a non-volatile load in // |entry_point| or not. bool IsTargetUsedByNonVolatileLoadInEntryPoint(uint32_t var_id, Instruction* entry_point); // Visits load instructions of pointers to variable whose result id is // |var_id| if the load instructions are in reachable functions from entry // points. |handle_load| is a function to do some actions for the load // instructions. Finishes the traversal and returns false if |handle_load| // returns false for a load instruction. Otherwise, returns true after running // |handle_load| for all the load instructions. bool VisitLoadsOfPointersToVariableInEntries( uint32_t var_id, const std::function& handle_load, const std::unordered_set& function_ids); // Sets Memory Operands of OpLoad instructions that load |var| or pointers // of |var| as Volatile if the function id of the OpLoad instruction is // included in |entry_function_ids|. void SetVolatileForLoadsInEntries( Instruction* var, const std::unordered_set& entry_function_ids); // Adds OpDecorate Volatile for |var| if it does not exist. void DecorateVarWithVolatile(Instruction* var); // Returns a set of entry function ids to spread the volatile semantics for // the variable with the result id |var_id|. std::unordered_set EntryFunctionsToSpreadVolatileSemanticsForVar( uint32_t var_id) { auto itr = var_ids_to_entry_fn_for_volatile_semantics_.find(var_id); if (itr == var_ids_to_entry_fn_for_volatile_semantics_.end()) return {}; return itr->second; } // Specifies that we have to spread the volatile semantics for the // variable with the result id |var_id| for the entry point |entry_point|. void MarkVolatileSemanticsForVariable(uint32_t var_id, Instruction* entry_point); // Result ids of variables to entry function ids for the volatile semantics // spread. std::unordered_map> var_ids_to_entry_fn_for_volatile_semantics_; }; } // namespace opt } // namespace spvtools #endif // SOURCE_OPT_SPREAD_VOLATILE_SEMANTICS_H_