Andy Kaylor a304e09fe7
[CIR] Handle expression with cleanups (#151600)
This adds code to handle expressions with cleanup, including
materializing a temporary object for the expression.
2025-08-01 11:38:35 -07:00

98 lines
2.9 KiB
C++

// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fclangir -emit-cir %s -o %t.cir
// RUN: FileCheck --input-file=%t.cir %s
struct Struk {
~Struk();
};
// CHECK: !rec_Struk = !cir.record<struct "Struk" padded {!u8i}>
// CHECK: cir.func{{.*}} @_ZN5StrukD1Ev(!cir.ptr<!rec_Struk>)
void test_cleanup() {
Struk s;
}
// CHECK: cir.func{{.*}} @_Z12test_cleanupv()
// CHECK: %[[S_ADDR:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["s"]
// CHECK: cir.call @_ZN5StrukD1Ev(%[[S_ADDR]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: cir.return
void test_cleanup_ifelse(bool b) {
if (b) {
Struk s;
} else {
Struk s;
}
}
// CHECK: cir.func{{.*}} @_Z19test_cleanup_ifelseb(%arg0: !cir.bool
// CHECK: cir.scope {
// CHECK: %[[B:.*]] = cir.load{{.*}} %0 : !cir.ptr<!cir.bool>
// CHECK: cir.if %[[B]] {
// CHECK: %[[S:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["s"]
// CHECK: cir.call @_ZN5StrukD1Ev(%[[S]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: } else {
// CHECK: %[[S_TOO:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["s"]
// CHECK: cir.call @_ZN5StrukD1Ev(%[[S_TOO]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: }
// CHECK: }
// CHECK: cir.return
void test_cleanup_for() {
for (int i = 0; i < 10; i++) {
Struk s;
}
}
// CHECK: cir.func{{.*}} @_Z16test_cleanup_forv()
// CHECK: cir.scope {
// CHECK: cir.for : cond {
// CHECK: } body {
// CHECK: cir.scope {
// CHECK: %[[S:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["s"]
// CHECK: cir.call @_ZN5StrukD1Ev(%[[S]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: }
// CHECK: cir.yield
// CHECK: } step {
// CHECK: }
// CHECK: }
// CHECK: cir.return
void test_cleanup_nested() {
Struk outer;
{
Struk middle;
{
Struk inner;
}
}
}
// CHECK: cir.func{{.*}} @_Z19test_cleanup_nestedv()
// CHECK: %[[OUTER:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["outer"]
// CHECK: cir.scope {
// CHECK: %[[MIDDLE:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["middle"]
// CHECK: cir.scope {
// CHECK: %[[INNER:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>, ["inner"]
// CHECK: cir.call @_ZN5StrukD1Ev(%[[INNER]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: }
// CHECK: cir.call @_ZN5StrukD1Ev(%[[MIDDLE]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: }
// CHECK: cir.call @_ZN5StrukD1Ev(%[[OUTER]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: cir.return
void use_ref(const Struk &);
void test_expr_with_cleanup() {
use_ref(Struk{});
}
// CHECK: cir.func{{.*}} @_Z22test_expr_with_cleanupv()
// CHECK: cir.scope {
// CHECK: %[[S:.*]] = cir.alloca !rec_Struk, !cir.ptr<!rec_Struk>
// CHECK: cir.call @_Z7use_refRK5Struk(%[[S]])
// CHECK: cir.call @_ZN5StrukD1Ev(%[[S]]) nothrow : (!cir.ptr<!rec_Struk>) -> ()
// CHECK: }
// CHECK: cir.return