| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 |
- // 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<bool(Instruction*)>& handle_load,
- const std::unordered_set<uint32_t>& 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<uint32_t>& 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<uint32_t> 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<uint32_t, std::unordered_set<uint32_t>>
- var_ids_to_entry_fn_for_volatile_semantics_;
- };
- } // namespace opt
- } // namespace spvtools
- #endif // SOURCE_OPT_SPREAD_VOLATILE_SEMANTICS_H_
|