writer/spirv: Generate load of atomic value arguments
Fixed: tint:926 Change-Id: Ia27abe605ebfb46a7524b50500ecebd6e4656d1d Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/55883 Reviewed-by: David Neto <dneto@google.com> Commit-Queue: David Neto <dneto@google.com> Auto-Submit: Ben Clayton <bclayton@google.com> Kokoro: Kokoro <noreply+kokoro@google.com>
This commit is contained in:
parent
9d3e2acd35
commit
07b59ca230
|
@ -2970,6 +2970,10 @@ bool Builder::GenerateAtomicIntrinsic(ast::CallExpression* call,
|
|||
if (value_id == 0) {
|
||||
return false;
|
||||
}
|
||||
value_id = GenerateLoadIfNeeded(TypeOf(call->params().back()), value_id);
|
||||
if (value_id == 0) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
Operand pointer = Operand::Int(pointer_id);
|
||||
|
|
|
@ -1849,8 +1849,10 @@ TEST_F(IntrinsicBuilderTest, Call_AtomicStore) {
|
|||
// [[binding(1), group(2)]] var<storage, read_write> b : S;
|
||||
//
|
||||
// fn a_func() {
|
||||
// let u : u32 = atomicStore(&b.u, 1u);
|
||||
// let i : i32 = atomicStore(&b.i, 2);
|
||||
// var u = 1u;
|
||||
// var i = 2;
|
||||
// atomicStore(&b.u, u);
|
||||
// atomicStore(&b.i, i);
|
||||
// }
|
||||
auto* s = Structure("S",
|
||||
{
|
||||
|
@ -1866,10 +1868,12 @@ TEST_F(IntrinsicBuilderTest, Call_AtomicStore) {
|
|||
|
||||
Func("a_func", ast::VariableList{}, ty.void_(),
|
||||
ast::StatementList{
|
||||
Decl(Var("u", nullptr, Expr(1u))),
|
||||
Decl(Var("i", nullptr, Expr(2))),
|
||||
create<ast::CallStatement>(
|
||||
Call("atomicStore", AddressOf(MemberAccessor("b", "u")), 1u)),
|
||||
Call("atomicStore", AddressOf(MemberAccessor("b", "u")), "u")),
|
||||
create<ast::CallStatement>(
|
||||
Call("atomicStore", AddressOf(MemberAccessor("b", "i")), 2)),
|
||||
Call("atomicStore", AddressOf(MemberAccessor("b", "i")), "i")),
|
||||
},
|
||||
ast::DecorationList{Stage(ast::PipelineStage::kFragment)});
|
||||
|
||||
|
@ -1886,19 +1890,27 @@ TEST_F(IntrinsicBuilderTest, Call_AtomicStore) {
|
|||
%1 = OpVariable %2 StorageBuffer
|
||||
%7 = OpTypeVoid
|
||||
%6 = OpTypeFunction %7
|
||||
%11 = OpConstant %4 1
|
||||
%12 = OpConstant %4 0
|
||||
%14 = OpTypePointer StorageBuffer %4
|
||||
%18 = OpTypePointer StorageBuffer %5
|
||||
%20 = OpConstant %5 2
|
||||
%10 = OpConstant %4 1
|
||||
%12 = OpTypePointer Function %4
|
||||
%13 = OpConstantNull %4
|
||||
%14 = OpConstant %5 2
|
||||
%16 = OpTypePointer Function %5
|
||||
%17 = OpConstantNull %5
|
||||
%19 = OpConstant %4 0
|
||||
%21 = OpTypePointer StorageBuffer %4
|
||||
%26 = OpTypePointer StorageBuffer %5
|
||||
)";
|
||||
auto got_types = DumpInstructions(b.types());
|
||||
EXPECT_EQ(expected_types, got_types);
|
||||
|
||||
auto* expected_instructions = R"(%15 = OpAccessChain %14 %1 %12
|
||||
OpAtomicStore %15 %11 %12 %11
|
||||
%19 = OpAccessChain %18 %1 %11
|
||||
OpAtomicStore %19 %11 %12 %20
|
||||
auto* expected_instructions = R"(OpStore %11 %10
|
||||
OpStore %15 %14
|
||||
%22 = OpAccessChain %21 %1 %19
|
||||
%23 = OpLoad %4 %11
|
||||
OpAtomicStore %22 %10 %19 %23
|
||||
%27 = OpAccessChain %26 %1 %10
|
||||
%28 = OpLoad %5 %15
|
||||
OpAtomicStore %27 %10 %19 %28
|
||||
)";
|
||||
auto got_instructions = DumpInstructions(b.functions()[0].instructions());
|
||||
EXPECT_EQ(expected_instructions, got_instructions);
|
||||
|
@ -1916,7 +1928,8 @@ TEST_P(Intrinsic_Builtin_AtomicRMW_i32, Test) {
|
|||
// [[binding(1), group(2)]] var<storage, read_write> b : S;
|
||||
//
|
||||
// fn a_func() {
|
||||
// let x : i32 = atomicOP(&b.v);
|
||||
// var v = 10;
|
||||
// let x : i32 = atomicOP(&b.v, v);
|
||||
// }
|
||||
auto* s = Structure("S",
|
||||
{
|
||||
|
@ -1931,9 +1944,10 @@ TEST_P(Intrinsic_Builtin_AtomicRMW_i32, Test) {
|
|||
|
||||
Func("a_func", ast::VariableList{}, ty.void_(),
|
||||
ast::StatementList{
|
||||
Decl(Const(
|
||||
"x", ty.i32(),
|
||||
Call(GetParam().name, AddressOf(MemberAccessor("b", "v")), 10))),
|
||||
Decl(Var("v", nullptr, Expr(10))),
|
||||
Decl(Const("x", ty.i32(),
|
||||
Call(GetParam().name, AddressOf(MemberAccessor("b", "v")),
|
||||
"v"))),
|
||||
},
|
||||
ast::DecorationList{Stage(ast::PipelineStage::kFragment)});
|
||||
|
||||
|
@ -1949,17 +1963,22 @@ TEST_P(Intrinsic_Builtin_AtomicRMW_i32, Test) {
|
|||
%1 = OpVariable %2 StorageBuffer
|
||||
%6 = OpTypeVoid
|
||||
%5 = OpTypeFunction %6
|
||||
%10 = OpTypeInt 32 0
|
||||
%11 = OpConstant %10 1
|
||||
%12 = OpConstant %10 0
|
||||
%14 = OpTypePointer StorageBuffer %4
|
||||
%16 = OpConstant %4 10
|
||||
%9 = OpConstant %4 10
|
||||
%11 = OpTypePointer Function %4
|
||||
%12 = OpConstantNull %4
|
||||
%14 = OpTypeInt 32 0
|
||||
%15 = OpConstant %14 1
|
||||
%16 = OpConstant %14 0
|
||||
%18 = OpTypePointer StorageBuffer %4
|
||||
)";
|
||||
auto got_types = DumpInstructions(b.types());
|
||||
EXPECT_EQ(expected_types, got_types);
|
||||
|
||||
std::string expected_instructions = "%15 = OpAccessChain %14 %1 %12\n";
|
||||
expected_instructions += "%9 = " + GetParam().op + " %4 %15 %11 %12 %16\n";
|
||||
std::string expected_instructions = R"(OpStore %10 %9
|
||||
%19 = OpAccessChain %18 %1 %16
|
||||
%20 = OpLoad %4 %10
|
||||
)";
|
||||
expected_instructions += "%13 = " + GetParam().op + " %4 %19 %15 %16 %20\n";
|
||||
|
||||
auto got_instructions = DumpInstructions(b.functions()[0].instructions());
|
||||
EXPECT_EQ(expected_instructions, got_instructions);
|
||||
|
@ -1986,7 +2005,8 @@ TEST_P(Intrinsic_Builtin_AtomicRMW_u32, Test) {
|
|||
// [[binding(1), group(2)]] var<storage, read_write> b : S;
|
||||
//
|
||||
// fn a_func() {
|
||||
// let x : u32 = atomicOP(&b.v);
|
||||
// var v = 10u;
|
||||
// let x : u32 = atomicOP(&b.v, v);
|
||||
// }
|
||||
auto* s = Structure("S",
|
||||
{
|
||||
|
@ -2001,9 +2021,10 @@ TEST_P(Intrinsic_Builtin_AtomicRMW_u32, Test) {
|
|||
|
||||
Func("a_func", ast::VariableList{}, ty.void_(),
|
||||
ast::StatementList{
|
||||
Decl(Var("v", nullptr, Expr(10u))),
|
||||
Decl(Const("x", ty.u32(),
|
||||
Call(GetParam().name, AddressOf(MemberAccessor("b", "v")),
|
||||
10u))),
|
||||
"v"))),
|
||||
},
|
||||
ast::DecorationList{Stage(ast::PipelineStage::kFragment)});
|
||||
|
||||
|
@ -2019,16 +2040,21 @@ TEST_P(Intrinsic_Builtin_AtomicRMW_u32, Test) {
|
|||
%1 = OpVariable %2 StorageBuffer
|
||||
%6 = OpTypeVoid
|
||||
%5 = OpTypeFunction %6
|
||||
%10 = OpConstant %4 1
|
||||
%11 = OpConstant %4 0
|
||||
%13 = OpTypePointer StorageBuffer %4
|
||||
%15 = OpConstant %4 10
|
||||
%9 = OpConstant %4 10
|
||||
%11 = OpTypePointer Function %4
|
||||
%12 = OpConstantNull %4
|
||||
%14 = OpConstant %4 1
|
||||
%15 = OpConstant %4 0
|
||||
%17 = OpTypePointer StorageBuffer %4
|
||||
)";
|
||||
auto got_types = DumpInstructions(b.types());
|
||||
EXPECT_EQ(expected_types, got_types);
|
||||
|
||||
std::string expected_instructions = "%14 = OpAccessChain %13 %1 %11\n";
|
||||
expected_instructions += "%9 = " + GetParam().op + " %4 %14 %10 %11 %15\n";
|
||||
std::string expected_instructions = R"(OpStore %10 %9
|
||||
%18 = OpAccessChain %17 %1 %15
|
||||
%19 = OpLoad %4 %10
|
||||
)";
|
||||
expected_instructions += "%13 = " + GetParam().op + " %4 %18 %14 %15 %19\n";
|
||||
|
||||
auto got_instructions = DumpInstructions(b.functions()[0].instructions());
|
||||
EXPECT_EQ(expected_instructions, got_instructions);
|
||||
|
@ -2054,8 +2080,10 @@ TEST_F(IntrinsicBuilderTest, Call_AtomicExchange) {
|
|||
// [[binding(1), group(2)]] var<storage, read_write> b : S;
|
||||
//
|
||||
// fn a_func() {
|
||||
// let u : u32 = atomicExchange(&b.u, 10u);
|
||||
// let i : i32 = atomicExchange(&b.i, 10);
|
||||
// var u = 10u;
|
||||
// var i = 10;
|
||||
// let r : u32 = atomicExchange(&b.u, u);
|
||||
// let s : i32 = atomicExchange(&b.i, i);
|
||||
// }
|
||||
auto* s = Structure("S",
|
||||
{
|
||||
|
@ -2071,12 +2099,14 @@ TEST_F(IntrinsicBuilderTest, Call_AtomicExchange) {
|
|||
|
||||
Func("a_func", ast::VariableList{}, ty.void_(),
|
||||
ast::StatementList{
|
||||
Decl(Const("u", ty.u32(),
|
||||
Decl(Var("u", nullptr, Expr(10u))),
|
||||
Decl(Var("i", nullptr, Expr(10))),
|
||||
Decl(Const("r", ty.u32(),
|
||||
Call("atomicExchange",
|
||||
AddressOf(MemberAccessor("b", "u")), 10u))),
|
||||
Decl(Const("i", ty.i32(),
|
||||
AddressOf(MemberAccessor("b", "u")), "u"))),
|
||||
Decl(Const("s", ty.i32(),
|
||||
Call("atomicExchange",
|
||||
AddressOf(MemberAccessor("b", "i")), 10))),
|
||||
AddressOf(MemberAccessor("b", "i")), "i"))),
|
||||
},
|
||||
ast::DecorationList{Stage(ast::PipelineStage::kFragment)});
|
||||
|
||||
|
@ -2093,20 +2123,28 @@ TEST_F(IntrinsicBuilderTest, Call_AtomicExchange) {
|
|||
%1 = OpVariable %2 StorageBuffer
|
||||
%7 = OpTypeVoid
|
||||
%6 = OpTypeFunction %7
|
||||
%11 = OpConstant %4 1
|
||||
%12 = OpConstant %4 0
|
||||
%14 = OpTypePointer StorageBuffer %4
|
||||
%16 = OpConstant %4 10
|
||||
%19 = OpTypePointer StorageBuffer %5
|
||||
%21 = OpConstant %5 10
|
||||
%10 = OpConstant %4 10
|
||||
%12 = OpTypePointer Function %4
|
||||
%13 = OpConstantNull %4
|
||||
%14 = OpConstant %5 10
|
||||
%16 = OpTypePointer Function %5
|
||||
%17 = OpConstantNull %5
|
||||
%19 = OpConstant %4 1
|
||||
%20 = OpConstant %4 0
|
||||
%22 = OpTypePointer StorageBuffer %4
|
||||
%27 = OpTypePointer StorageBuffer %5
|
||||
)";
|
||||
auto got_types = DumpInstructions(b.types());
|
||||
EXPECT_EQ(expected_types, got_types);
|
||||
|
||||
auto* expected_instructions = R"(%15 = OpAccessChain %14 %1 %12
|
||||
%10 = OpAtomicExchange %4 %15 %11 %12 %16
|
||||
%20 = OpAccessChain %19 %1 %11
|
||||
%17 = OpAtomicExchange %5 %20 %11 %12 %21
|
||||
auto* expected_instructions = R"(OpStore %11 %10
|
||||
OpStore %15 %14
|
||||
%23 = OpAccessChain %22 %1 %20
|
||||
%24 = OpLoad %4 %11
|
||||
%18 = OpAtomicExchange %4 %23 %19 %20 %24
|
||||
%28 = OpAccessChain %27 %1 %19
|
||||
%29 = OpLoad %5 %15
|
||||
%25 = OpAtomicExchange %5 %28 %19 %20 %29
|
||||
)";
|
||||
auto got_instructions = DumpInstructions(b.functions()[0].instructions());
|
||||
EXPECT_EQ(expected_instructions, got_instructions);
|
||||
|
|
|
@ -0,0 +1,14 @@
|
|||
[[block]] struct DrawIndirectArgs {
|
||||
vertexCount : atomic<u32>;
|
||||
};
|
||||
[[group(0), binding(5)]] var<storage, read_write> drawOut : DrawIndirectArgs;
|
||||
|
||||
var<private> cubeVerts : u32 = 0u;
|
||||
|
||||
[[stage(compute)]]
|
||||
fn computeMain([[builtin(global_invocation_id)]] global_id : vec3<u32>) {
|
||||
// Increment cubeVerts based on some criteria...
|
||||
|
||||
// This fails SPIR-V validation
|
||||
let firstVertex : u32 = atomicAdd(&drawOut.vertexCount, cubeVerts);
|
||||
}
|
|
@ -0,0 +1,15 @@
|
|||
RWByteAddressBuffer drawOut : register(u5, space0);
|
||||
static uint cubeVerts = 0u;
|
||||
|
||||
struct tint_symbol_1 {
|
||||
uint3 global_id : SV_DispatchThreadID;
|
||||
};
|
||||
|
||||
[numthreads(1, 1, 1)]
|
||||
void computeMain(tint_symbol_1 tint_symbol) {
|
||||
const uint3 global_id = tint_symbol.global_id;
|
||||
uint atomic_result = 0u;
|
||||
drawOut.InterlockedAdd(0u, 0u, atomic_result);
|
||||
const uint firstVertex = atomic_result;
|
||||
return;
|
||||
}
|
|
@ -0,0 +1,17 @@
|
|||
SKIP: FAILED
|
||||
|
||||
|
||||
[[block]]
|
||||
struct DrawIndirectArgs {
|
||||
vertexCount : atomic<u32>;
|
||||
};
|
||||
|
||||
[[group(0), binding(5)]] var<storage, read_write> drawOut : DrawIndirectArgs;
|
||||
|
||||
[[stage(compute)]]
|
||||
fn computeMain([[builtin(global_invocation_id)]] global_id : vec3<u32>) {
|
||||
[[internal(disable_validation__function_var_storage_class)]] var<private> tint_symbol_1 : u32 = 0u;
|
||||
let firstVertex : u32 = atomicAdd(&(drawOut.vertexCount), tint_symbol_1);
|
||||
}
|
||||
|
||||
Failed to generate: error: unknown type in EmitType: __atomic__u32
|
|
@ -0,0 +1,41 @@
|
|||
; SPIR-V
|
||||
; Version: 1.3
|
||||
; Generator: Google Tint Compiler; 0
|
||||
; Bound: 21
|
||||
; Schema: 0
|
||||
OpCapability Shader
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpEntryPoint GLCompute %computeMain "computeMain"
|
||||
OpExecutionMode %computeMain LocalSize 1 1 1
|
||||
OpName %DrawIndirectArgs "DrawIndirectArgs"
|
||||
OpMemberName %DrawIndirectArgs 0 "vertexCount"
|
||||
OpName %drawOut "drawOut"
|
||||
OpName %cubeVerts "cubeVerts"
|
||||
OpName %tint_symbol "tint_symbol"
|
||||
OpName %computeMain "computeMain"
|
||||
OpDecorate %DrawIndirectArgs Block
|
||||
OpMemberDecorate %DrawIndirectArgs 0 Offset 0
|
||||
OpDecorate %drawOut DescriptorSet 0
|
||||
OpDecorate %drawOut Binding 5
|
||||
OpDecorate %tint_symbol BuiltIn GlobalInvocationId
|
||||
%uint = OpTypeInt 32 0
|
||||
%DrawIndirectArgs = OpTypeStruct %uint
|
||||
%_ptr_StorageBuffer_DrawIndirectArgs = OpTypePointer StorageBuffer %DrawIndirectArgs
|
||||
%drawOut = OpVariable %_ptr_StorageBuffer_DrawIndirectArgs StorageBuffer
|
||||
%uint_0 = OpConstant %uint 0
|
||||
%_ptr_Private_uint = OpTypePointer Private %uint
|
||||
%cubeVerts = OpVariable %_ptr_Private_uint Private %uint_0
|
||||
%v3uint = OpTypeVector %uint 3
|
||||
%_ptr_Input_v3uint = OpTypePointer Input %v3uint
|
||||
%tint_symbol = OpVariable %_ptr_Input_v3uint Input
|
||||
%void = OpTypeVoid
|
||||
%11 = OpTypeFunction %void
|
||||
%uint_1 = OpConstant %uint 1
|
||||
%_ptr_StorageBuffer_uint = OpTypePointer StorageBuffer %uint
|
||||
%computeMain = OpFunction %void None %11
|
||||
%14 = OpLabel
|
||||
%19 = OpAccessChain %_ptr_StorageBuffer_uint %drawOut %uint_0
|
||||
%20 = OpLoad %uint %cubeVerts
|
||||
%15 = OpAtomicIAdd %uint %19 %uint_1 %uint_0 %20
|
||||
OpReturn
|
||||
OpFunctionEnd
|
|
@ -0,0 +1,13 @@
|
|||
[[block]]
|
||||
struct DrawIndirectArgs {
|
||||
vertexCount : atomic<u32>;
|
||||
};
|
||||
|
||||
[[group(0), binding(5)]] var<storage, read_write> drawOut : DrawIndirectArgs;
|
||||
|
||||
var<private> cubeVerts : u32 = 0u;
|
||||
|
||||
[[stage(compute)]]
|
||||
fn computeMain([[builtin(global_invocation_id)]] global_id : vec3<u32>) {
|
||||
let firstVertex : u32 = atomicAdd(&(drawOut.vertexCount), cubeVerts);
|
||||
}
|
Loading…
Reference in New Issue