| // Copyright 2020 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. |
| |
| #include "src/transform/msl.h" |
| |
| #include <utility> |
| #include <vector> |
| |
| #include "src/program_builder.h" |
| #include "src/semantic/variable.h" |
| |
| namespace tint { |
| namespace transform { |
| namespace { |
| const char* kReservedKeywords[] = {"access", |
| "alignas", |
| "alignof", |
| "and", |
| "and_eq", |
| "array", |
| "array_ref", |
| "as_type", |
| "asm", |
| "atomic", |
| "atomic_bool", |
| "atomic_int", |
| "atomic_uint", |
| "auto", |
| "bitand", |
| "bitor", |
| "bool", |
| "bool2", |
| "bool3", |
| "bool4", |
| "break", |
| "buffer", |
| "case", |
| "catch", |
| "char", |
| "char16_t", |
| "char2", |
| "char3", |
| "char32_t", |
| "char4", |
| "class", |
| "compl", |
| "const", |
| "const_cast", |
| "const_reference", |
| "constant", |
| "constexpr", |
| "continue", |
| "decltype", |
| "default", |
| "delete", |
| "depth2d", |
| "depth2d_array", |
| "depth2d_ms", |
| "depth2d_ms_array", |
| "depthcube", |
| "depthcube_array", |
| "device", |
| "discard_fragment", |
| "do", |
| "double", |
| "dynamic_cast", |
| "else", |
| "enum", |
| "explicit", |
| "extern", |
| "false", |
| "final", |
| "float", |
| "float2", |
| "float2x2", |
| "float2x3", |
| "float2x4", |
| "float3", |
| "float3x2", |
| "float3x3", |
| "float3x4", |
| "float4", |
| "float4x2", |
| "float4x3", |
| "float4x4", |
| "for", |
| "fragment", |
| "friend", |
| "goto", |
| "half", |
| "half2", |
| "half2x2", |
| "half2x3", |
| "half2x4", |
| "half3", |
| "half3x2", |
| "half3x3", |
| "half3x4", |
| "half4", |
| "half4x2", |
| "half4x3", |
| "half4x4", |
| "if", |
| "imageblock", |
| "inline", |
| "int", |
| "int16_t", |
| "int2", |
| "int3", |
| "int32_t", |
| "int4", |
| "int64_t", |
| "int8_t", |
| "kernel", |
| "long", |
| "long2", |
| "long3", |
| "long4", |
| "main", |
| "metal", |
| "mutable", |
| "namespace", |
| "new", |
| "noexcept", |
| "not", |
| "not_eq", |
| "nullptr", |
| "operator", |
| "or", |
| "or_eq", |
| "override", |
| "packed_bool2", |
| "packed_bool3", |
| "packed_bool4", |
| "packed_char2", |
| "packed_char3", |
| "packed_char4", |
| "packed_float2", |
| "packed_float3", |
| "packed_float4", |
| "packed_half2", |
| "packed_half3", |
| "packed_half4", |
| "packed_int2", |
| "packed_int3", |
| "packed_int4", |
| "packed_short2", |
| "packed_short3", |
| "packed_short4", |
| "packed_uchar2", |
| "packed_uchar3", |
| "packed_uchar4", |
| "packed_uint2", |
| "packed_uint3", |
| "packed_uint4", |
| "packed_ushort2", |
| "packed_ushort3", |
| "packed_ushort4", |
| "patch_control_point", |
| "private", |
| "protected", |
| "ptrdiff_t", |
| "public", |
| "r16snorm", |
| "r16unorm", |
| "r8unorm", |
| "reference", |
| "register", |
| "reinterpret_cast", |
| "return", |
| "rg11b10f", |
| "rg16snorm", |
| "rg16unorm", |
| "rg8snorm", |
| "rg8unorm", |
| "rgb10a2", |
| "rgb9e5", |
| "rgba16snorm", |
| "rgba16unorm", |
| "rgba8snorm", |
| "rgba8unorm", |
| "sampler", |
| "short", |
| "short2", |
| "short3", |
| "short4", |
| "signed", |
| "size_t", |
| "sizeof", |
| "srgba8unorm", |
| "static", |
| "static_assert", |
| "static_cast", |
| "struct", |
| "switch", |
| "template", |
| "texture", |
| "texture1d", |
| "texture1d_array", |
| "texture2d", |
| "texture2d_array", |
| "texture2d_ms", |
| "texture2d_ms_array", |
| "texture3d", |
| "texture_buffer", |
| "texturecube", |
| "texturecube_array", |
| "this", |
| "thread", |
| "thread_local", |
| "threadgroup", |
| "threadgroup_imageblock", |
| "throw", |
| "true", |
| "try", |
| "typedef", |
| "typeid", |
| "typename", |
| "uchar", |
| "uchar2", |
| "uchar3", |
| "uchar4", |
| "uint", |
| "uint16_t", |
| "uint2", |
| "uint3", |
| "uint32_t", |
| "uint4", |
| "uint64_t", |
| "uint8_t", |
| "ulong2", |
| "ulong3", |
| "ulong4", |
| "uniform", |
| "union", |
| "unsigned", |
| "ushort", |
| "ushort2", |
| "ushort3", |
| "ushort4", |
| "using", |
| "vec", |
| "vertex", |
| "virtual", |
| "void", |
| "volatile", |
| "wchar_t", |
| "while", |
| "xor", |
| "xor_eq"}; |
| } // namespace |
| |
| Msl::Msl() = default; |
| Msl::~Msl() = default; |
| |
| Transform::Output Msl::Run(const Program* in) { |
| ProgramBuilder out; |
| CloneContext ctx(&out, in); |
| RenameReservedKeywords(&ctx, kReservedKeywords); |
| HandleEntryPointIOTypes(ctx); |
| ctx.Clone(); |
| |
| return Output{Program(std::move(out))}; |
| } |
| |
| void Msl::HandleEntryPointIOTypes(CloneContext& ctx) const { |
| // Collect location-decorated entry point parameters into a struct. |
| // Insert function-scope const declarations to replace those parameters. |
| // |
| // Before: |
| // ``` |
| // [[stage(fragment)]] |
| // fn frag_main([[builtin(frag_coord)]] coord : vec4<f32>, |
| // [[location(1)]] loc1 : f32, |
| // [[location(2)]] loc2 : vec4<u32>) -> void { |
| // var col : f32 = (coord.x * loc1); |
| // } |
| // ``` |
| // |
| // After: |
| // ``` |
| // struct frag_main_in { |
| // [[location(1)]] loc1 : f32; |
| // [[location(2)]] loc2 : vec4<u32> |
| // }; |
| |
| // [[stage(fragment)]] |
| // fn frag_main([[builtin(frag_coord)]] coord : vec4<f32>, |
| // in : frag_main_in) -> void { |
| // const loc1 : f32 = in.loc1; |
| // const loc2 : vec4<u32> = in.loc2; |
| // var col : f32 = (coord.x * loc1); |
| // } |
| // ``` |
| |
| for (auto* func : ctx.src->AST().Functions()) { |
| if (!func->IsEntryPoint()) { |
| continue; |
| } |
| |
| std::vector<ast::Variable*> worklist; |
| ast::StructMemberList struct_members; |
| ast::VariableList new_parameters; |
| ast::StatementList new_body; |
| |
| // Find location-decorated parameters. |
| for (auto* param : func->params()) { |
| // TODO(jrprice): Handle structs (collate members into a single struct). |
| if (param->decorations().size() != 1) { |
| TINT_ICE(ctx.dst->Diagnostics()) << "Unsupported entry point parameter"; |
| } |
| |
| auto* deco = param->decorations()[0]; |
| if (auto* builtin = deco->As<ast::BuiltinDecoration>()) { |
| // Keep any builtin-decorated parameters unchanged. |
| new_parameters.push_back(ctx.Clone(param)); |
| } else if (auto* loc = deco->As<ast::LocationDecoration>()) { |
| // Create a struct member with the location decoration. |
| struct_members.push_back( |
| ctx.dst->Member(param->symbol().to_str(), ctx.Clone(param->type()), |
| ast::DecorationList{ctx.Clone(loc)})); |
| worklist.push_back(param); |
| } else { |
| TINT_ICE(ctx.dst->Diagnostics()) |
| << "Unsupported entry point parameter decoration"; |
| } |
| } |
| |
| if (worklist.empty()) { |
| // Nothing to do. |
| continue; |
| } |
| |
| // Create a struct type to hold all of the user-defined input parameters. |
| auto* in_struct = ctx.dst->create<type::Struct>( |
| ctx.dst->Symbols().New(), |
| ctx.dst->create<ast::Struct>(struct_members, ast::DecorationList{})); |
| ctx.dst->AST().AddConstructedType(in_struct); |
| |
| // Create a new function parameter using this struct type. |
| auto struct_param_symbol = ctx.dst->Symbols().New(); |
| auto* struct_param = |
| ctx.dst->Var(struct_param_symbol, in_struct, ast::StorageClass::kNone); |
| new_parameters.push_back(struct_param); |
| |
| // Replace the original parameters with function-scope constants. |
| for (auto* param : worklist) { |
| // Create a function-scope const to replace the parameter. |
| // Initialize it with the value extracted from the struct parameter. |
| auto func_const_symbol = ctx.dst->Symbols().New(); |
| auto* func_const = |
| ctx.dst->Const(func_const_symbol, ctx.Clone(param->type()), |
| ctx.dst->MemberAccessor(struct_param_symbol, |
| param->symbol().to_str())); |
| |
| new_body.push_back(ctx.dst->WrapInStatement(func_const)); |
| |
| // Replace all uses of the function parameter with the function const. |
| for (auto* user : ctx.src->Sem().Get(param)->Users()) { |
| ctx.Replace<ast::Expression>(user->Declaration(), |
| ctx.dst->Expr(func_const_symbol)); |
| } |
| } |
| |
| // Copy over the rest of the function body unchanged. |
| for (auto* stmt : func->body()->list()) { |
| new_body.push_back(ctx.Clone(stmt)); |
| } |
| |
| // Rewrite the function header with the new parameters. |
| auto* new_func = ctx.dst->create<ast::Function>( |
| func->source(), ctx.Clone(func->symbol()), new_parameters, |
| ctx.Clone(func->return_type()), |
| ctx.dst->create<ast::BlockStatement>(new_body), |
| ctx.Clone(func->decorations())); |
| ctx.Replace(func, new_func); |
| } |
| } |
| |
| } // namespace transform |
| } // namespace tint |