| // Copyright 2021 The Tint Authors. | 
 | // | 
 | // 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 SRC_TRANSFORM_SPIRV_H_ | 
 | #define SRC_TRANSFORM_SPIRV_H_ | 
 |  | 
 | #include <vector> | 
 |  | 
 | #include "src/transform/transform.h" | 
 |  | 
 | namespace tint { | 
 |  | 
 | // Forward declarations | 
 | class CloneContext; | 
 |  | 
 | namespace transform { | 
 |  | 
 | /// Spirv is a transform used to sanitize a Program for use with the Spirv | 
 | /// writer. Passing a non-sanitized Program to the Spirv writer will result in | 
 | /// undefined behavior. | 
 | class Spirv : public Transform { | 
 |  public: | 
 |   /// Constructor | 
 |   Spirv(); | 
 |   ~Spirv() override; | 
 |  | 
 |   /// Runs the transform on `program`, returning the transformation result. | 
 |   /// @param program the source program to transform | 
 |   /// @param data optional extra transform-specific input data | 
 |   /// @returns the transformation result | 
 |   Output Run(const Program* program, const DataMap& data = {}) override; | 
 |  | 
 |  private: | 
 |   /// Hoist entry point parameters, return values, and struct members out to | 
 |   /// global variables. | 
 |   void HandleEntryPointIOTypes(CloneContext& ctx) const; | 
 |   /// Change type of sample mask builtin variables to single element arrays. | 
 |   void HandleSampleMaskBuiltins(CloneContext& ctx) const; | 
 |   /// Add an empty shader entry point if none exist in the module. | 
 |   void AddEmptyEntryPoint(CloneContext& ctx) const; | 
 |  | 
 |   /// Recursively create module-scope input variables for `ty` and add | 
 |   /// function-scope variables for structs to `func`. | 
 |   /// | 
 |   /// For non-structures, create a module-scope input variable. | 
 |   /// For structures, recurse into members and then create a function-scope | 
 |   /// variable initialized using the variables created for its members. | 
 |   /// Return the symbol for the variable that was created. | 
 |   Symbol HoistToInputVariables(CloneContext& ctx, | 
 |                                const ast::Function* func, | 
 |                                type::Type* ty, | 
 |                                type::Type* declared_ty, | 
 |                                const ast::DecorationList& decorations) const; | 
 |  | 
 |   /// Recursively create module-scope output variables for `ty` and build a list | 
 |   /// of assignment instructions to write to them from `store_value`. | 
 |   /// | 
 |   /// For non-structures, create a module-scope output variable and generate the | 
 |   /// assignment instruction. | 
 |   /// For structures, recurse into members, tracking the chain of member | 
 |   /// accessors. | 
 |   /// Returns the list of variable assignments in `stores`. | 
 |   void HoistToOutputVariables(CloneContext& ctx, | 
 |                               const ast::Function* func, | 
 |                               type::Type* ty, | 
 |                               type::Type* declared_ty, | 
 |                               const ast::DecorationList& decorations, | 
 |                               std::vector<Symbol> member_accesses, | 
 |                               Symbol store_value, | 
 |                               ast::StatementList& stores) const; | 
 | }; | 
 |  | 
 | }  // namespace transform | 
 | }  // namespace tint | 
 |  | 
 | #endif  // SRC_TRANSFORM_SPIRV_H_ |