// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -std=c++20 -fclangir -mconstructor-aliases -emit-cir %s -o %t.cir // RUN: FileCheck --check-prefix=CIR --input-file=%t.cir %s // RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -std=c++20 -fclangir -mconstructor-aliases -emit-llvm %s -o %t-cir.ll // RUN: FileCheck --check-prefix=LLVM --input-file=%t-cir.ll %s // RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -std=c++20 -mconstructor-aliases -emit-llvm %s -o %t.ll // RUN: FileCheck --check-prefix=OGCG --input-file=%t.ll %s typedef __typeof(sizeof(int)) size_t; struct SizedDelete { void operator delete(void*, size_t); int member; }; void test_sized_delete(SizedDelete *x) { delete x; } // SizedDelete::operator delete(void*, unsigned long) // CIR: cir.func private @_ZN11SizedDeletedlEPvm(!cir.ptr {llvm.noundef}, !u64i {llvm.noundef}) // LLVM: declare void @_ZN11SizedDeletedlEPvm(ptr noundef, i64 noundef) // CIR: cir.func {{.*}} @_Z17test_sized_deleteP11SizedDelete // CIR: %[[X:.*]] = cir.load{{.*}} %{{.*}} // CIR: %[[NULL:.*]] = cir.const #cir.ptr : !cir.ptr // CIR: %[[NOT_NULL:.*]] = cir.cmp ne %[[X]], %[[NULL]] : !cir.ptr // CIR: cir.if %[[NOT_NULL]] { // CIR: cir.cleanup.scope { // CIR: cir.yield // CIR: } cleanup normal { // CIR: %[[X_CAST:.*]] = cir.cast bitcast %[[X]] : !cir.ptr -> !cir.ptr // CIR: %[[OBJ_SIZE:.*]] = cir.const #cir.int<4> : !u64i // CIR: cir.call @_ZN11SizedDeletedlEPvm(%[[X_CAST]], %[[OBJ_SIZE]]) nothrow : (!cir.ptr {llvm.noundef}, !u64i {llvm.noundef}) -> () // CIR: cir.yield // CIR: } // CIR: } // LLVM: define dso_local void @_Z17test_sized_deleteP11SizedDelete // LLVM: %[[X:.*]] = load ptr, ptr %{{.*}} // LLVM: %[[NOT_NULL:.*]] = icmp ne ptr %[[X]], null // LLVM: br i1 %[[NOT_NULL]], label %[[DELETE_NOTNULL:.*]], label %[[DELETE_END:.*]] // LLVM: [[DELETE_NOTNULL]]: // LLVM: br label %[[ENTER_CLEANUP_SCOPE:.*]] // LLVM: [[ENTER_CLEANUP_SCOPE]]: // LLVM: br label %[[NORMAL_CLEANUP:.*]] // LLVM: [[NORMAL_CLEANUP]]: // LLVM: call void @_ZN11SizedDeletedlEPvm(ptr noundef %[[X]], i64 noundef 4) // LLVM: br label %[[EXIT_CLEANUP:.*]] // LLVM: [[EXIT_CLEANUP]]: // LLVM: br label %[[EXIT_CLEANUP_SCOPE:.*]] // LLVM: [[EXIT_CLEANUP_SCOPE]]: // LLVM: br label %[[DELETE_END]] // LLVM: [[DELETE_END]]: // LLVM: ret void // OGCG: define dso_local void @_Z17test_sized_deleteP11SizedDelete // OGCG: %[[X:.*]] = load ptr, ptr %{{.*}} // OGCG: %[[ISNULL:.*]] = icmp eq ptr %[[X]], null // OGCG: br i1 %[[ISNULL]], label %{{.*}}, label %[[DELETE_NOTNULL:.*]] // OGCG: [[DELETE_NOTNULL]]: // OGCG: call void @_ZN11SizedDeletedlEPvm(ptr noundef %[[X]], i64 noundef 4) // This function is declared below the call in OGCG. // OGCG: declare void @_ZN11SizedDeletedlEPvm(ptr noundef, i64 noundef) struct Contents { ~Contents() {} }; struct Container { Contents *contents; ~Container(); }; Container::~Container() { delete contents; } // Contents::~Contents() // CIR: cir.func {{.*}} @_ZN8ContentsD2Ev // LLVM: define linkonce_odr void @_ZN8ContentsD2Ev // operator delete(void*, unsigned long) // CIR: cir.func {{.*}} @_ZdlPvm(!cir.ptr {llvm.noundef}, !u64i {llvm.noundef}) // LLVM: declare void @_ZdlPvm(ptr noundef, i64 noundef) // Container::~Container() // CIR: cir.func {{.*}} @_ZN9ContainerD2Ev // CIR: %[[THIS:.*]] = cir.load %{{.*}} // CIR: %[[CONTENTS_PTR_ADDR:.*]] = cir.get_member %[[THIS]][0] {name = "contents"} : !cir.ptr -> !cir.ptr> // CIR: %[[CONTENTS_PTR:.*]] = cir.load{{.*}} %[[CONTENTS_PTR_ADDR]] // CIR: %[[NULL:.*]] = cir.const #cir.ptr : !cir.ptr // CIR: %[[NOT_NULL:.*]] = cir.cmp ne %[[CONTENTS_PTR]], %[[NULL]] : !cir.ptr // CIR: cir.if %[[NOT_NULL]] { // CIR: cir.cleanup.scope { // CIR: cir.call @_ZN8ContentsD2Ev(%[[CONTENTS_PTR]]) nothrow : (!cir.ptr {llvm.align = 1 : i64, llvm.dereferenceable = 1 : i64, llvm.nonnull, llvm.noundef}) -> () // CIR: cir.yield // CIR: } cleanup normal { // CIR: %[[CONTENTS_CAST:.*]] = cir.cast bitcast %[[CONTENTS_PTR]] : !cir.ptr -> !cir.ptr // CIR: %[[OBJ_SIZE:.*]] = cir.const #cir.int<1> : !u64i // CIR: cir.call @_ZdlPvm(%[[CONTENTS_CAST]], %[[OBJ_SIZE]]) nothrow {builtin} : (!cir.ptr {llvm.noundef}, !u64i {llvm.noundef}) -> () // CIR: cir.yield // CIR: } // CIR: } // LLVM: define dso_local void @_ZN9ContainerD2Ev // LLVM: %[[THIS:.*]] = load ptr, ptr %{{.*}} // LLVM: %[[CONTENTS_PTR_ADDR:.*]] = getelementptr inbounds nuw %struct.Container, ptr %[[THIS]], i32 0, i32 0 // LLVM: %[[CONTENTS_PTR:.*]] = load ptr, ptr %[[CONTENTS_PTR_ADDR]] // LLVM: %[[NOT_NULL:.*]] = icmp ne ptr %[[CONTENTS_PTR]], null // LLVM: br i1 %[[NOT_NULL]], label %[[DELETE_NOTNULL:.*]], label %[[DELETE_END:.*]] // LLVM: [[DELETE_NOTNULL]]: // LLVM: br label %[[ENTER_CLEANUP_SCOPE:.*]] // LLVM: [[ENTER_CLEANUP_SCOPE]]: // LLVM: call void @_ZN8ContentsD2Ev(ptr noundef nonnull align 1 dereferenceable(1) %[[CONTENTS_PTR]]) // LLVM: br label %[[NORMAL_CLEANUP:.*]] // LLVM: [[NORMAL_CLEANUP]]: // LLVM: call void @_ZdlPvm(ptr noundef %[[CONTENTS_PTR]], i64 noundef 1) // LLVM: br label %[[EXIT_CLEANUP:.*]] // LLVM: [[EXIT_CLEANUP]]: // LLVM: br label %[[EXIT_CLEANUP_SCOPE:.*]] // LLVM: [[EXIT_CLEANUP_SCOPE]]: // LLVM: br label %[[DELETE_END]] // LLVM: [[DELETE_END]]: // LLVM: ret void // OGCG: define dso_local void @_ZN9ContainerD2Ev // OGCG: %[[THIS:.*]] = load ptr, ptr %{{.*}} // OGCG: %[[CONTENTS:.*]] = getelementptr inbounds nuw %struct.Container, ptr %[[THIS]], i32 0, i32 0 // OGCG: %[[CONTENTS_PTR:.*]] = load ptr, ptr %[[CONTENTS]] // OGCG: %[[ISNULL:.*]] = icmp eq ptr %[[CONTENTS_PTR]], null // OGCG: br i1 %[[ISNULL]], label %{{.*}}, label %[[DELETE_NOTNULL:.*]] // OGCG: [[DELETE_NOTNULL]]: // OGCG: call void @_ZN8ContentsD2Ev(ptr noundef nonnull align 1 dereferenceable(1) %[[CONTENTS_PTR]]) // OGCG: call void @_ZdlPvm(ptr noundef %[[CONTENTS_PTR]], i64 noundef 1) // These functions are declared/defined below the calls in OGCG. // OGCG: define linkonce_odr void @_ZN8ContentsD2Ev // OGCG: declare void @_ZdlPvm(ptr noundef, i64 noundef) struct StructWithVirtualDestructor { virtual ~StructWithVirtualDestructor(); }; void destroy(StructWithVirtualDestructor *x) { delete x; } // CIR: cir.func {{.*}} @_Z7destroyP27StructWithVirtualDestructor(%[[X_ARG:.*]]: !cir.ptr {{.*}}) // CIR: %[[X_ADDR:.*]] = cir.alloca !cir.ptr // CIR: cir.store %[[X_ARG]], %[[X_ADDR]] // CIR: %[[X:.*]] = cir.load{{.*}} %[[X_ADDR]] // CIR: %[[NULL:.*]] = cir.const #cir.ptr : !cir.ptr // CIR: %[[NOT_NULL:.*]] = cir.cmp ne %[[X]], %[[NULL]] : !cir.ptr // CIR: cir.if %[[NOT_NULL]] { // CIR: %[[VTABLE_PTR:.*]] = cir.vtable.get_vptr %[[X]] : !cir.ptr -> !cir.ptr // CIR: %[[VTABLE:.*]] = cir.load{{.*}} %[[VTABLE_PTR]] : !cir.ptr, !cir.vptr // CIR: %[[DTOR_FN_ADDR_PTR:.*]] = cir.vtable.get_virtual_fn_addr %[[VTABLE]][1] // CIR: %[[DTOR_FN_ADDR:.*]] = cir.load{{.*}} %[[DTOR_FN_ADDR_PTR]] // CIR: cir.call %[[DTOR_FN_ADDR]](%[[X]]) // CIR: } // LLVM: define {{.*}} void @_Z7destroyP27StructWithVirtualDestructor(ptr {{.*}} %[[X_ARG:.*]]) // LLVM: %[[X_ADDR:.*]] = alloca ptr // LLVM: store ptr %[[X_ARG]], ptr %[[X_ADDR]] // LLVM: %[[X:.*]] = load ptr, ptr %[[X_ADDR]] // LLVM: %[[NOT_NULL:.*]] = icmp ne ptr %[[X]], null // LLVM: br i1 %[[NOT_NULL]], label %[[DELETE_NOTNULL:.*]], label %[[DELETE_END:.*]] // LLVM: [[DELETE_NOTNULL]]: // LLVM: %[[VTABLE:.*]] = load ptr, ptr %[[X]] // LLVM: %[[DTOR_FN_ADDR_PTR:.*]] = getelementptr inbounds ptr, ptr %[[VTABLE]], i32 1 // LLVM: %[[DTOR_FN_ADDR:.*]] = load ptr, ptr %[[DTOR_FN_ADDR_PTR]] // LLVM: call void %[[DTOR_FN_ADDR]](ptr {{.*}} %[[X]]) // LLVM: [[DELETE_END]]: // LLVM: ret void // OGCG: define {{.*}} void @_Z7destroyP27StructWithVirtualDestructor(ptr {{.*}} %[[X_ARG:.*]]) // OGCG: %[[X_ADDR:.*]] = alloca ptr // OGCG: store ptr %[[X_ARG]], ptr %[[X_ADDR]] // OGCG: %[[X:.*]] = load ptr, ptr %[[X_ADDR]] // OGCG: %[[ISNULL:.*]] = icmp eq ptr %[[X]], null // OGCG: br i1 %[[ISNULL]], label %{{.*}}, label %[[DELETE_NOTNULL:.*]] // OGCG: [[DELETE_NOTNULL]]: // OGCG: %[[VTABLE:.*]] = load ptr, ptr %[[X]] // OGCG: %[[DTOR_FN_ADDR_PTR:.*]] = getelementptr inbounds ptr, ptr %[[VTABLE]], i64 1 // OGCG: %[[DTOR_FN_ADDR:.*]] = load ptr, ptr %[[DTOR_FN_ADDR_PTR]] // OGCG: call void %[[DTOR_FN_ADDR]](ptr {{.*}} %[[X]])