tint: Clean up AddSpirvBlockAttribute
* Walk the sem type list, instead of all AST nodes to just find the types. * Walk the AST global list, instead of all declarations, then filtering to globals. * Use the ast::IsHostSharable() helper where possible. * Only lookup the sem node once instead of twice, per global. * Use the new utils::Hashmap and utils::Hashset containers instead of std::unordered_map and std::unordered_set. Change-Id: Idb47c855ae9dfd27515774a89cedb7ed06e07035 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/100140 Reviewed-by: Dan Sinclair <dsinclair@chromium.org> Kokoro: Kokoro <noreply+kokoro@google.com> Commit-Queue: Ben Clayton <bclayton@google.com>
This commit is contained in:
parent
bace05e499
commit
cdcc85973b
|
@ -14,13 +14,12 @@
|
||||||
|
|
||||||
#include "src/tint/transform/add_spirv_block_attribute.h"
|
#include "src/tint/transform/add_spirv_block_attribute.h"
|
||||||
|
|
||||||
#include <unordered_map>
|
|
||||||
#include <unordered_set>
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include "src/tint/program_builder.h"
|
#include "src/tint/program_builder.h"
|
||||||
#include "src/tint/sem/variable.h"
|
#include "src/tint/sem/variable.h"
|
||||||
#include "src/tint/utils/map.h"
|
#include "src/tint/utils/hashmap.h"
|
||||||
|
#include "src/tint/utils/hashset.h"
|
||||||
|
|
||||||
TINT_INSTANTIATE_TYPEINFO(tint::transform::AddSpirvBlockAttribute);
|
TINT_INSTANTIATE_TYPEINFO(tint::transform::AddSpirvBlockAttribute);
|
||||||
TINT_INSTANTIATE_TYPEINFO(tint::transform::AddSpirvBlockAttribute::SpirvBlockAttribute);
|
TINT_INSTANTIATE_TYPEINFO(tint::transform::AddSpirvBlockAttribute::SpirvBlockAttribute);
|
||||||
|
@ -35,59 +34,64 @@ void AddSpirvBlockAttribute::Run(CloneContext& ctx, const DataMap&, DataMap&) co
|
||||||
auto& sem = ctx.src->Sem();
|
auto& sem = ctx.src->Sem();
|
||||||
|
|
||||||
// Collect the set of structs that are nested in other types.
|
// Collect the set of structs that are nested in other types.
|
||||||
std::unordered_set<const sem::Struct*> nested_structs;
|
utils::Hashset<const sem::Struct*, 8> nested_structs;
|
||||||
for (auto* node : ctx.src->ASTNodes().Objects()) {
|
for (auto* ty : ctx.src->Types()) {
|
||||||
if (auto* arr = sem.Get<sem::Array>(node->As<ast::Array>())) {
|
Switch(
|
||||||
if (auto* nested_str = arr->ElemType()->As<sem::Struct>()) {
|
ty,
|
||||||
nested_structs.insert(nested_str);
|
[&](const sem::Array* arr) {
|
||||||
}
|
if (auto* nested_str = arr->ElemType()->As<sem::Struct>()) {
|
||||||
} else if (auto* str = sem.Get<sem::Struct>(node->As<ast::Struct>())) {
|
nested_structs.Add(nested_str);
|
||||||
for (auto* member : str->Members()) {
|
|
||||||
if (auto* nested_str = member->Type()->As<sem::Struct>()) {
|
|
||||||
nested_structs.insert(nested_str);
|
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
}
|
[&](const sem::Struct* str) {
|
||||||
|
for (auto* member : str->Members()) {
|
||||||
|
if (auto* nested_str = member->Type()->As<sem::Struct>()) {
|
||||||
|
nested_structs.Add(nested_str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// A map from a type in the source program to a block-decorated wrapper that
|
// A map from a type in the source program to a block-decorated wrapper that contains it in the
|
||||||
// contains it in the destination program.
|
// destination program.
|
||||||
std::unordered_map<const sem::Type*, const ast::Struct*> wrapper_structs;
|
utils::Hashmap<const sem::Type*, const ast::Struct*, 8> wrapper_structs;
|
||||||
|
|
||||||
// Process global 'var' declarations that are buffers.
|
// Process global 'var' declarations that are buffers.
|
||||||
for (auto* var : ctx.src->AST().Globals<ast::Var>()) {
|
for (auto* global : ctx.src->AST().GlobalVariables()) {
|
||||||
auto* sem_var = sem.Get<sem::GlobalVariable>(var);
|
auto* var = sem.Get(global);
|
||||||
if (var->declared_storage_class != ast::StorageClass::kStorage &&
|
if (!ast::IsHostShareable(var->StorageClass())) {
|
||||||
var->declared_storage_class != ast::StorageClass::kUniform &&
|
// Not declared in a host-sharable storage class
|
||||||
var->declared_storage_class != ast::StorageClass::kPushConstant) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* ty = sem.Get(var->type);
|
auto* ty = var->Type()->UnwrapRef();
|
||||||
auto* str = ty->As<sem::Struct>();
|
auto* str = ty->As<sem::Struct>();
|
||||||
if (!str || nested_structs.count(str)) {
|
bool needs_wrapping = !str || // Type is not a structure
|
||||||
|
nested_structs.Contains(str); // Structure is nested by another type
|
||||||
|
|
||||||
|
if (needs_wrapping) {
|
||||||
const char* kMemberName = "inner";
|
const char* kMemberName = "inner";
|
||||||
|
|
||||||
// This is a non-struct or a struct that is nested somewhere else, so we
|
// This is a non-struct or a struct that is nested somewhere else, so we
|
||||||
// need to wrap it first.
|
// need to wrap it first.
|
||||||
auto* wrapper = utils::GetOrCreate(wrapper_structs, ty, [&]() {
|
auto* wrapper = wrapper_structs.GetOrCreate(ty, [&] {
|
||||||
auto* block = ctx.dst->ASTNodes().Create<SpirvBlockAttribute>(
|
auto* block = ctx.dst->ASTNodes().Create<SpirvBlockAttribute>(
|
||||||
ctx.dst->ID(), ctx.dst->AllocateNodeID());
|
ctx.dst->ID(), ctx.dst->AllocateNodeID());
|
||||||
auto wrapper_name = ctx.src->Symbols().NameFor(var->symbol) + "_block";
|
auto wrapper_name = ctx.src->Symbols().NameFor(global->symbol) + "_block";
|
||||||
auto* ret = ctx.dst->create<ast::Struct>(
|
auto* ret = ctx.dst->create<ast::Struct>(
|
||||||
ctx.dst->Symbols().New(wrapper_name),
|
ctx.dst->Symbols().New(wrapper_name),
|
||||||
utils::Vector{ctx.dst->Member(kMemberName, CreateASTTypeFor(ctx, ty))},
|
utils::Vector{ctx.dst->Member(kMemberName, CreateASTTypeFor(ctx, ty))},
|
||||||
utils::Vector{block});
|
utils::Vector{block});
|
||||||
ctx.InsertBefore(ctx.src->AST().GlobalDeclarations(), var, ret);
|
ctx.InsertBefore(ctx.src->AST().GlobalDeclarations(), global, ret);
|
||||||
return ret;
|
return ret;
|
||||||
});
|
});
|
||||||
ctx.Replace(var->type, ctx.dst->ty.Of(wrapper));
|
ctx.Replace(global->type, ctx.dst->ty.Of(wrapper));
|
||||||
|
|
||||||
// Insert a member accessor to get the original type from the wrapper at
|
// Insert a member accessor to get the original type from the wrapper at
|
||||||
// any usage of the original variable.
|
// any usage of the original variable.
|
||||||
for (auto* user : sem_var->Users()) {
|
for (auto* user : var->Users()) {
|
||||||
ctx.Replace(user->Declaration(),
|
ctx.Replace(user->Declaration(),
|
||||||
ctx.dst->MemberAccessor(ctx.Clone(var->symbol), kMemberName));
|
ctx.dst->MemberAccessor(ctx.Clone(global->symbol), kMemberName));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Add a block attribute to this struct directly.
|
// Add a block attribute to this struct directly.
|
||||||
|
|
|
@ -163,7 +163,40 @@ fn main() {
|
||||||
EXPECT_EQ(expect, str(got));
|
EXPECT_EQ(expect, str(got));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(AddSpirvBlockAttributeTest, BasicStruct) {
|
TEST_F(AddSpirvBlockAttributeTest, BasicStruct_AccessRoot) {
|
||||||
|
auto* src = R"(
|
||||||
|
struct S {
|
||||||
|
f : f32,
|
||||||
|
};
|
||||||
|
|
||||||
|
@group(0) @binding(0)
|
||||||
|
var<uniform> u : S;
|
||||||
|
|
||||||
|
@fragment
|
||||||
|
fn main() {
|
||||||
|
let f = u;
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
auto* expect = R"(
|
||||||
|
@internal(spirv_block)
|
||||||
|
struct S {
|
||||||
|
f : f32,
|
||||||
|
}
|
||||||
|
|
||||||
|
@group(0) @binding(0) var<uniform> u : S;
|
||||||
|
|
||||||
|
@fragment
|
||||||
|
fn main() {
|
||||||
|
let f = u;
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
|
||||||
|
auto got = Run<AddSpirvBlockAttribute>(src);
|
||||||
|
|
||||||
|
EXPECT_EQ(expect, str(got));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(AddSpirvBlockAttributeTest, BasicStruct_AccessField) {
|
||||||
auto* src = R"(
|
auto* src = R"(
|
||||||
struct S {
|
struct S {
|
||||||
f : f32,
|
f : f32,
|
||||||
|
|
Loading…
Reference in New Issue