transform/InlinePtrLets: Fix ICE for lets in for-loops

For loop initializers and continuing statements do not have a BlockStatement as their parent.
Handle removal of these statements with a new Transform::RemoveStatement() helper

Fixed: tint:990
Change-Id: I24e7b18dcf71d3ef0a4d3ee68b9f68518e0eb5e8
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/58063
Kokoro: Kokoro <noreply+kokoro@google.com>
Reviewed-by: James Price <jrprice@google.com>
This commit is contained in:
Ben Clayton 2021-07-15 22:20:29 +00:00
parent 96a6e7e95d
commit 8e38fad091
9 changed files with 95 additions and 3 deletions

View File

@ -513,12 +513,15 @@ class CloneContext {
/// Reports an internal compiler error if the cast failed. /// Reports an internal compiler error if the cast failed.
template <typename TO, typename FROM> template <typename TO, typename FROM>
TO* CheckedCast(FROM* obj) { TO* CheckedCast(FROM* obj) {
if (TO* cast = As<TO>(obj)) { if (obj == nullptr) {
return nullptr;
}
if (TO* cast = obj->template As<TO>()) {
return cast; return cast;
} }
TINT_ICE(Clone, Diagnostics()) TINT_ICE(Clone, Diagnostics())
<< "Cloned object was not of the expected type\n" << "Cloned object was not of the expected type\n"
<< "got: " << (obj ? obj->TypeInfo().name : "<null>") << "\n" << "got: " << obj->TypeInfo().name << "\n"
<< "expected: " << TypeInfo::Of<TO>().name; << "expected: " << TypeInfo::Of<TO>().name;
return nullptr; return nullptr;
} }

View File

@ -177,7 +177,7 @@ void InlinePointerLets::Run(CloneContext& ctx, const DataMap&, DataMap&) {
ptr_lets.emplace(let->variable(), std::move(ptr_let)); ptr_lets.emplace(let->variable(), std::move(ptr_let));
// As the original `let` declaration will be fully inlined, there's no // As the original `let` declaration will be fully inlined, there's no
// need for the original declaration to exist. Remove it. // need for the original declaration to exist. Remove it.
ctx.Remove(block->statements(), let); RemoveStatement(ctx, let);
} }
} }

View File

@ -19,6 +19,8 @@
#include "src/program_builder.h" #include "src/program_builder.h"
#include "src/sem/atomic_type.h" #include "src/sem/atomic_type.h"
#include "src/sem/block_statement.h"
#include "src/sem/for_loop_statement.h"
#include "src/sem/reference_type.h" #include "src/sem/reference_type.h"
TINT_INSTANTIATE_TYPEINFO(tint::transform::Transform); TINT_INSTANTIATE_TYPEINFO(tint::transform::Transform);
@ -84,6 +86,21 @@ ast::DecorationList Transform::RemoveDecorations(
return new_decorations; return new_decorations;
} }
void Transform::RemoveStatement(CloneContext& ctx, ast::Statement* stmt) {
auto* sem = ctx.src->Sem().Get(stmt);
if (auto* block = tint::As<sem::BlockStatement>(sem->Parent())) {
ctx.Remove(block->Declaration()->statements(), stmt);
return;
}
if (tint::Is<sem::ForLoopStatement>(sem->Parent())) {
ctx.Replace(stmt, static_cast<ast::Expression*>(nullptr));
return;
}
TINT_ICE(Transform, ctx.dst->Diagnostics())
<< "unable to remove statement from parent of type "
<< sem->TypeInfo().name;
}
ast::Type* Transform::CreateASTTypeFor(CloneContext& ctx, const sem::Type* ty) { ast::Type* Transform::CreateASTTypeFor(CloneContext& ctx, const sem::Type* ty) {
if (ty->Is<sem::Void>()) { if (ty->Is<sem::Void>()) {
return ctx.dst->create<ast::Void>(); return ctx.dst->create<ast::Void>();

View File

@ -195,6 +195,13 @@ class Transform : public Castable<Transform> {
const ast::DecorationList& in, const ast::DecorationList& in,
std::function<bool(const ast::Decoration*)> should_remove); std::function<bool(const ast::Decoration*)> should_remove);
/// Removes the statement `stmt` from the transformed program.
/// RemoveStatement handles edge cases, like statements in the initializer and
/// continuing of for-loops.
/// @param ctx the clone context
/// @param stmt the statement to remove when the program is cloned
static void RemoveStatement(CloneContext& ctx, ast::Statement* stmt);
/// CreateASTTypeFor constructs new ast::Type nodes that reconstructs the /// CreateASTTypeFor constructs new ast::Type nodes that reconstructs the
/// semantic type `ty`. /// semantic type `ty`.
/// @param ctx the clone context /// @param ctx the clone context

4
test/bug/tint/990.wgsl Normal file
View File

@ -0,0 +1,4 @@
fn f() {
var i : i32;
for (let p = &i;;) {}
}

View File

@ -0,0 +1,12 @@
[numthreads(1, 1, 1)]
void unused_entry_point() {
return;
}
void f() {
int i = 0;
{
for(; ; ) {
}
}
}

View File

@ -0,0 +1,9 @@
#include <metal_stdlib>
using namespace metal;
void f() {
int i = 0;
for(; ; ) {
}
}

View File

@ -0,0 +1,35 @@
; SPIR-V
; Version: 1.3
; Generator: Google Tint Compiler; 0
; Bound: 15
; Schema: 0
OpCapability Shader
OpMemoryModel Logical GLSL450
OpEntryPoint GLCompute %unused_entry_point "unused_entry_point"
OpExecutionMode %unused_entry_point LocalSize 1 1 1
OpName %unused_entry_point "unused_entry_point"
OpName %f "f"
OpName %i "i"
%void = OpTypeVoid
%1 = OpTypeFunction %void
%int = OpTypeInt 32 1
%_ptr_Function_int = OpTypePointer Function %int
%10 = OpConstantNull %int
%unused_entry_point = OpFunction %void None %1
%4 = OpLabel
OpReturn
OpFunctionEnd
%f = OpFunction %void None %1
%6 = OpLabel
%i = OpVariable %_ptr_Function_int Function %10
OpBranch %11
%11 = OpLabel
OpLoopMerge %12 %13 None
OpBranch %14
%14 = OpLabel
OpBranch %13
%13 = OpLabel
OpBranch %11
%12 = OpLabel
OpReturn
OpFunctionEnd

View File

@ -0,0 +1,5 @@
fn f() {
var i : i32;
for(let p = &(i); ; ;) {
}
}