spirv-reader: dereference the base pointer for access chain, if needed
Bug: tint:737 Change-Id: I36588ee54ea014e55d3dcc9a54e8d3835c1e83bc Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/53540 Auto-Submit: David Neto <dneto@google.com> Kokoro: Kokoro <noreply+kokoro@google.com> Commit-Queue: James Price <jrprice@google.com> Reviewed-by: James Price <jrprice@google.com>
This commit is contained in:
parent
3a472bc5c2
commit
9ef9a10a23
|
@ -3837,6 +3837,9 @@ TypedExpression FunctionEmitter::MakeAccessChain(
|
||||||
// for the base, and then bury that inside nested indexing expressions.
|
// for the base, and then bury that inside nested indexing expressions.
|
||||||
if (!current_expr) {
|
if (!current_expr) {
|
||||||
current_expr = MakeOperand(inst, 0);
|
current_expr = MakeOperand(inst, 0);
|
||||||
|
if (current_expr.type->Is<Pointer>()) {
|
||||||
|
current_expr = Dereference(current_expr);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
const auto constants = constant_mgr_->GetOperandConstants(&inst);
|
const auto constants = constant_mgr_->GetOperandConstants(&inst);
|
||||||
|
|
||||||
|
|
|
@ -820,6 +820,81 @@ TEST_F(SpvParserMemoryTest, EmitStatement_AccessChain_InvalidPointeeType) {
|
||||||
"%60: %60 = OpTypePointer Workgroup %55"));
|
"%60: %60 = OpTypePointer Workgroup %55"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserMemoryTest, EmitStatement_AccessChain_DereferenceBase) {
|
||||||
|
// The base operand to OpAccessChain may have to be dereferenced first.
|
||||||
|
// crbug.com/tint/737
|
||||||
|
const std::string assembly = Preamble() + R"(
|
||||||
|
%void = OpTypeVoid
|
||||||
|
%voidfn = OpTypeFunction %void
|
||||||
|
|
||||||
|
%uint = OpTypeInt 32 0
|
||||||
|
%v2uint = OpTypeVector %uint 2
|
||||||
|
%elem_ty = OpTypePointer Private %uint
|
||||||
|
%vec_ty = OpTypePointer Private %v2uint
|
||||||
|
|
||||||
|
%ptrfn = OpTypeFunction %void %vec_ty
|
||||||
|
|
||||||
|
%uint_0 = OpConstant %uint 0
|
||||||
|
|
||||||
|
; The shortest way to make a pointer example is as a function parameter.
|
||||||
|
%200 = OpFunction %void None %ptrfn
|
||||||
|
%1 = OpFunctionParameter %vec_ty
|
||||||
|
%entry = OpLabel
|
||||||
|
%2 = OpAccessChain %elem_ty %1 %uint_0
|
||||||
|
%3 = OpLoad %uint %2
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%main_entry = OpLabel
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)";
|
||||||
|
auto p = parser(test::Assemble(assembly));
|
||||||
|
std::cout << assembly << std::endl;
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModule());
|
||||||
|
const auto got = p->program().to_str();
|
||||||
|
const std::string expected = R"(Module{
|
||||||
|
Function x_200 -> __void
|
||||||
|
(
|
||||||
|
VariableConst{
|
||||||
|
x_1
|
||||||
|
none
|
||||||
|
undefined
|
||||||
|
__ptr_private__vec_2__u32
|
||||||
|
}
|
||||||
|
)
|
||||||
|
{
|
||||||
|
VariableDeclStatement{
|
||||||
|
VariableConst{
|
||||||
|
x_3
|
||||||
|
none
|
||||||
|
undefined
|
||||||
|
__u32
|
||||||
|
{
|
||||||
|
MemberAccessor[not set]{
|
||||||
|
UnaryOp[not set]{
|
||||||
|
indirection
|
||||||
|
Identifier[not set]{x_1}
|
||||||
|
}
|
||||||
|
Identifier[not set]{x}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Return{}
|
||||||
|
}
|
||||||
|
Function main -> __void
|
||||||
|
StageDecoration{vertex}
|
||||||
|
()
|
||||||
|
{
|
||||||
|
Return{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
EXPECT_EQ(got, expected) << got;
|
||||||
|
}
|
||||||
|
|
||||||
std::string OldStorageBufferPreamble() {
|
std::string OldStorageBufferPreamble() {
|
||||||
return Preamble() + R"(
|
return Preamble() + R"(
|
||||||
OpName %myvar "myvar"
|
OpName %myvar "myvar"
|
||||||
|
|
Loading…
Reference in New Issue