mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-07-04 04:06:01 +00:00
Avoid cloning parameters until we know we are going to rewrite the function. Change-Id: I0b0e2513d8652a0f2e561419848f77875d67591b Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/44600 Commit-Queue: James Price <jrprice@google.com> Reviewed-by: Ben Clayton <bclayton@google.com>
401 lines
16 KiB
C++
401 lines
16 KiB
C++
// 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 <unordered_set>
|
|
#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;
|
|
}
|
|
|
|
// Find location-decorated parameters and build a struct to hold them.
|
|
ast::StructMemberList struct_members;
|
|
std::unordered_set<ast::Variable*> builtins;
|
|
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.
|
|
builtins.insert(param);
|
|
continue;
|
|
} else if (auto* loc = deco->As<ast::LocationDecoration>()) {
|
|
// Create a struct member with the location decoration.
|
|
struct_members.push_back(ctx.dst->Member(
|
|
ctx.src->Symbols().NameFor(param->symbol()),
|
|
ctx.Clone(param->type()), ast::DecorationList{ctx.Clone(loc)}));
|
|
} else {
|
|
TINT_ICE(ctx.dst->Diagnostics())
|
|
<< "Unsupported entry point parameter decoration";
|
|
}
|
|
}
|
|
|
|
if (struct_members.empty()) {
|
|
// Nothing to do.
|
|
continue;
|
|
}
|
|
|
|
ast::VariableList new_parameters;
|
|
ast::StatementList new_body;
|
|
|
|
// 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 : func->params()) {
|
|
if (builtins.count(param)) {
|
|
// Keep any builtin-decorated parameters unchanged.
|
|
new_parameters.push_back(ctx.Clone(param));
|
|
continue;
|
|
}
|
|
|
|
auto name = ctx.src->Symbols().NameFor(param->symbol());
|
|
|
|
// 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().Register(name);
|
|
auto* func_const =
|
|
ctx.dst->Const(func_const_symbol, ctx.Clone(param->type()),
|
|
ctx.dst->MemberAccessor(struct_param_symbol, name));
|
|
|
|
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
|