Now also mark the this pointer dead_on_return for classes with a non-zero number of base classes. We saw a limited number of failures internally due to this change, so it doesn't seem like there are too many problems with real world deployment.
349 lines
19 KiB
C++
349 lines
19 KiB
C++
// RUN: %clang_cc1 -emit-llvm -fms-extensions %s -triple=x86_64-pc-windows-msvc -o - | FileCheck --check-prefixes=X64,CHECK %s
|
|
// RUN: %clang_cc1 -emit-llvm -fms-extensions %s -triple=i386-pc-windows-msvc -o - | FileCheck --check-prefixes=X86,CHECK %s
|
|
// RUN: %clang_cc1 -emit-llvm -fms-extensions %s -triple=x86_64-pc-windows-msvc -fclang-abi-compat=21 -o - | FileCheck --check-prefixes=CLANG21 %s
|
|
|
|
struct Bird {
|
|
virtual ~Bird();
|
|
};
|
|
|
|
struct Parrot : public Bird {
|
|
// X64: @[[ParrotVtable:[0-9]+]] = private unnamed_addr constant { [2 x ptr] } { [2 x ptr] [ptr @"??_R4Parrot@@6B@", ptr @"??_EParrot@@UEAAPEAXI@Z"] }, comdat($"??_7Parrot@@6B@")
|
|
// X86: @[[ParrotVtable:[0-9]+]] = private unnamed_addr constant { [2 x ptr] } { [2 x ptr] [ptr @"??_R4Parrot@@6B@", ptr @"??_EParrot@@UAEPAXI@Z"] }, comdat($"??_7Parrot@@6B@")
|
|
// CLANG21: @[[ParrotVtable:[0-9]+]] = private unnamed_addr constant { [2 x ptr] } { [2 x ptr] [ptr @"??_R4Parrot@@6B@", ptr @"??_GParrot@@UEAAPEAXI@Z"] }, comdat($"??_7Parrot@@6B@")
|
|
// X64: @[[Bird:[0-9]+]] = private unnamed_addr constant { [2 x ptr] } { [2 x ptr] [ptr @"??_R4Bird@@6B@", ptr @"??_EBird@@UEAAPEAXI@Z"] }, comdat($"??_7Bird@@6B@")
|
|
// X86: @[[Bird:[0-9]+]] = private unnamed_addr constant { [2 x ptr] } { [2 x ptr] [ptr @"??_R4Bird@@6B@", ptr @"??_EBird@@UAEPAXI@Z"] }, comdat($"??_7Bird@@6B@")
|
|
// CLANG21: @[[Bird:[0-9]+]] = private unnamed_addr constant { [2 x ptr] } { [2 x ptr] [ptr @"??_R4Bird@@6B@", ptr @"??_GBird@@UEAAPEAXI@Z"] }, comdat($"??_7Bird@@6B@")
|
|
virtual ~Parrot() {}
|
|
};
|
|
|
|
Bird::~Bird() {}
|
|
|
|
// For the weird bird we first emit scalar deleting destructor, then find out
|
|
// that we need vector deleting destructor and remove the alias.
|
|
struct JustAWeirdBird {
|
|
virtual ~JustAWeirdBird() {}
|
|
|
|
bool doSmth(int n) {
|
|
JustAWeirdBird *c = new JustAWeirdBird[n];
|
|
|
|
delete[] c;
|
|
return true;
|
|
}
|
|
};
|
|
|
|
int i = 0;
|
|
struct HasOperatorDelete : public Bird{
|
|
~HasOperatorDelete() { }
|
|
void operator delete(void *p) { i-=2; }
|
|
void operator delete[](void *p) { i--; }
|
|
};
|
|
|
|
struct AllocatedAsArray : public Bird {
|
|
|
|
};
|
|
|
|
// Vector deleting dtor for Bird is an alias because no new Bird[] expressions
|
|
// in the TU.
|
|
// X64: @"??_EBird@@UEAAPEAXI@Z" = weak dso_local unnamed_addr alias ptr (ptr, i32), ptr @"??_GBird@@UEAAPEAXI@Z"
|
|
// X86: @"??_EBird@@UAEPAXI@Z" = weak dso_local unnamed_addr alias ptr (ptr, i32), ptr @"??_GBird@@UAEPAXI@Z"
|
|
// No scalar destructor for Parrot.
|
|
// CHECK-NOT: @"??_GParrot"
|
|
// No vector destructor definition for Bird.
|
|
// CHECK-NOT: define{{.*}}@"??_EBird"
|
|
// No scalar deleting dtor for JustAWeirdBird.
|
|
// CHECK-NOT: @"??_GJustAWeirdBird"
|
|
// CLANG21-NOT: @"??_E
|
|
|
|
void dealloc(Bird *p) {
|
|
delete[] p;
|
|
}
|
|
|
|
Bird* alloc() {
|
|
Parrot* P = new Parrot[38];
|
|
return P;
|
|
}
|
|
|
|
|
|
template<class C>
|
|
struct S {
|
|
void foo() { void *p = new C(); delete (C *)p; }
|
|
};
|
|
|
|
S<AllocatedAsArray[1][3]> sp;
|
|
|
|
void bar() {
|
|
dealloc(alloc());
|
|
|
|
JustAWeirdBird B;
|
|
B.doSmth(38);
|
|
|
|
Bird *p = new HasOperatorDelete[2];
|
|
dealloc(p);
|
|
|
|
sp.foo();
|
|
}
|
|
|
|
// CHECK-LABEL: define dso_local void @{{.*}}dealloc{{.*}}(
|
|
// CHECK-SAME: ptr noundef %[[PTR:.*]])
|
|
// CHECK: entry:
|
|
// CHECK-NEXT: %[[PTRADDR:.*]] = alloca ptr
|
|
// CHECK-NEXT: store ptr %[[PTR]], ptr %[[PTRADDR]]
|
|
// CHECK-NEXT: %[[LPTR:.*]] = load ptr, ptr %[[PTRADDR]]
|
|
// CHECK-NEXT: %[[ISNULL:.*]] = icmp eq ptr %[[LPTR]], null
|
|
// CHECK-NEXT: br i1 %[[ISNULL]], label %delete.end, label %delete.notnull
|
|
// CHECK: delete.notnull:
|
|
// X64-NEXT: %[[COOKIEGEP:.*]] = getelementptr inbounds i8, ptr %[[LPTR]], i64 -8
|
|
// X86-NEXT: %[[COOKIEGEP:.*]] = getelementptr inbounds i8, ptr %[[LPTR]], i32 -4
|
|
// X64-NEXT: %[[HOWMANY:.*]] = load i64, ptr %[[COOKIEGEP]]
|
|
// X86-NEXT: %[[HOWMANY:.*]] = load i32, ptr %[[COOKIEGEP]]
|
|
// X64-NEXT: %[[ISNOELEM:.*]] = icmp eq i64 %2, 0
|
|
// X86-NEXT: %[[ISNOELEM:.*]] = icmp eq i32 %2, 0
|
|
// CHECK-NEXT: br i1 %[[ISNOELEM]], label %vdtor.nocall, label %vdtor.call
|
|
// CHECK: vdtor.nocall:
|
|
// X64-NEXT: %[[HOWMANYBYTES:.*]] = mul i64 8, %[[HOWMANY]]
|
|
// X86-NEXT: %[[HOWMANYBYTES:.*]] = mul i32 4, %[[HOWMANY]]
|
|
// X64-NEXT: %[[ADDCOOKIESIZE:.*]] = add i64 %[[HOWMANYBYTES]], 8
|
|
// X86-NEXT: %[[ADDCOOKIESIZE:.*]] = add i32 %[[HOWMANYBYTES]], 4
|
|
// X64-NEXT: call void @"??_V@YAXPEAX_K@Z"(ptr noundef %[[COOKIEGEP]], i64 noundef %[[ADDCOOKIESIZE]])
|
|
// X86-NEXT: call void @"??_V@YAXPAXI@Z"(ptr noundef %[[COOKIEGEP]], i32 noundef %[[ADDCOOKIESIZE]])
|
|
// CHECK-NEXT: br label %delete.end
|
|
// CHECK: vdtor.call:
|
|
// CHECK-NEXT: %[[VTABLE:.*]] = load ptr, ptr %[[LPTR]]
|
|
// CHECK-NEXT: %[[FPGEP:.*]] = getelementptr inbounds ptr, ptr %[[VTABLE]], i64 0
|
|
// CHECK-NEXT: %[[FPLOAD:.*]] = load ptr, ptr %[[FPGEP]]
|
|
// X64-NEXT: %[[CALL:.*]] = call noundef ptr %[[FPLOAD]](ptr noundef nonnull align 8 dereferenceable(8) %[[LPTR]], i32 noundef 3)
|
|
// X86-NEXT: %[[CALL:.*]] = call x86_thiscallcc noundef ptr %[[FPLOAD]](ptr noundef nonnull align 4 dereferenceable(4) %[[LPTR]], i32 noundef 3)
|
|
// CHECK-NEXT: br label %delete.end
|
|
// CHECK: delete.end:
|
|
// CHECK-NEXT: ret void
|
|
|
|
// Normal loop over the array elements for clang21 ABI
|
|
// CLANG21-LABEL: define dso_local void @"?dealloc@@YAXPEAUBird@@@Z"
|
|
// CLANG21: %p.addr = alloca ptr
|
|
// CLANG21-NEXT: store ptr %p, ptr %p.addr
|
|
// CLANG21-NEXT: %0 = load ptr, ptr %p.addr
|
|
// CLANG21-NEXT: %isnull = icmp eq ptr %0, null
|
|
// CLANG21-NEXT: br i1 %isnull, label %delete.end2, label %delete.notnull
|
|
// CLANG21: delete.notnull:
|
|
// CLANG21-NEXT: %1 = getelementptr inbounds i8, ptr %0, i64 -8
|
|
// CLANG21-NEXT: %2 = load i64, ptr %1
|
|
// CLANG21-NEXT: %delete.end = getelementptr inbounds %struct.Bird, ptr %0, i64 %2
|
|
// CLANG21-NEXT: %arraydestroy.isempty = icmp eq ptr %0, %delete.end
|
|
// CLANG21-NEXT: br i1 %arraydestroy.isempty, label %arraydestroy.done1, label %arraydestroy.body
|
|
// CLANG21: arraydestroy.body:
|
|
// CLANG21-NEXT: %arraydestroy.elementPast = phi ptr [ %delete.end, %delete.notnull ], [ %arraydestroy.element, %arraydestroy.body ]
|
|
// CLANG21-NEXT: %arraydestroy.element = getelementptr inbounds %struct.Bird, ptr %arraydestroy.elementPast, i64 -1
|
|
// CLANG21-NEXT: call void @"??1Bird@@UEAA@XZ"(ptr noundef nonnull align 8 dead_on_return(8) dereferenceable(8) %arraydestroy.element)
|
|
// CLANG21-NEXT: %arraydestroy.done = icmp eq ptr %arraydestroy.element, %0
|
|
// CLANG21-NEXT: br i1 %arraydestroy.done, label %arraydestroy.done1, label %arraydestroy.body
|
|
// CLANG21: arraydestroy.done1:
|
|
// CLANG21-NEXT: %3 = mul i64 8, %2
|
|
// CLANG21-NEXT: %4 = add i64 %3, 8
|
|
// CLANG21-NEXT: call void @"??_V@YAXPEAX_K@Z"(ptr noundef %1, i64 noundef %4)
|
|
// CLANG21-NEXT: br label %delete.end2
|
|
|
|
// Definition of S::foo, check that it has vector deleting destructor call
|
|
// X64-LABEL: define linkonce_odr dso_local void @"?foo@?$S@$$BY102UAllocatedAsArray@@@@QEAAXXZ"
|
|
// X86-LABEL: define linkonce_odr dso_local x86_thiscallcc void @"?foo@?$S@$$BY102UAllocatedAsArray@@@@QAEXXZ"
|
|
// X64: %[[NEWCALL:.*]] = call noalias noundef nonnull ptr @"??_U@YAPEAX_K@Z"(i64 noundef 32)
|
|
// X86: %[[NEWCALL:.*]] = call noalias noundef nonnull ptr @"??_U@YAPAXI@Z"(i32 noundef 16)
|
|
// X64: %[[ARR:.*]] = getelementptr inbounds i8, ptr %[[NEWCALL]], i64 8
|
|
// X86: %[[ARR:.*]] = getelementptr inbounds i8, ptr %[[NEWCALL]], i32 4
|
|
// CHECK: store ptr %[[ARR]], ptr %[[DP:.*]]
|
|
// CHECK: %[[DEL_PTR:.*]] = load ptr, ptr %[[DP:.*]]
|
|
// CHECK: delete.notnull:
|
|
// X64-NEXT: %[[COOKIEGEP:.*]] = getelementptr inbounds i8, ptr %[[DEL_PTR]], i64 -8
|
|
// X86-NEXT: %[[COOKIEGEP:.*]] = getelementptr inbounds i8, ptr %[[DEL_PTR]], i32 -4
|
|
// X64-NEXT: %[[HOWMANY:.*]] = load i64, ptr %[[COOKIEGEP]]
|
|
// X86-NEXT: %[[HOWMANY:.*]] = load i32, ptr %[[COOKIEGEP]]
|
|
// X64-NEXT: %[[ISNOELEM:.*]] = icmp eq i64 %[[HOWMANY]], 0
|
|
// X86-NEXT: %[[ISNOELEM:.*]] = icmp eq i32 %[[HOWMANY]], 0
|
|
// CHECK-NEXT: br i1 %[[ISNOELEM]], label %vdtor.nocall, label %vdtor.call
|
|
// CHECK: vdtor.nocall: ; preds = %delete.notnull
|
|
// X64-NEXT: %[[HOWMANYBYTES:.*]] = mul i64 8, %[[HOWMANY]]
|
|
// X86-NEXT: %[[HOWMANYBYTES:.*]] = mul i32 4, %[[HOWMANY]]
|
|
// X64-NEXT: %[[ADDCOOKIESIZE:.*]] = add i64 %[[HOWMANYBYTES]], 8
|
|
// X86-NEXT: %[[ADDCOOKIESIZE:.*]] = add i32 %[[HOWMANYBYTES]], 4
|
|
// X64-NEXT: call void @"??_V@YAXPEAX_K@Z"(ptr noundef %[[COOKIEGEP]], i64 noundef %[[ADDCOOKIESIZE]])
|
|
// X86-NEXT: call void @"??_V@YAXPAXI@Z"(ptr noundef %[[COOKIEGEP]], i32 noundef %[[ADDCOOKIESIZE]])
|
|
// CHECK-NEXT: br label %delete.end
|
|
// CHECK: vdtor.call: ; preds = %delete.notnull
|
|
// CHECK-NEXT: %[[VTABLE:.*]] = load ptr, ptr %[[DEL_PTR]]
|
|
// CHECK-NEXT: %[[FPGEP:.*]] = getelementptr inbounds ptr, ptr %[[VTABLE]], i64 0
|
|
// CHECK-NEXT: %[[FPLOAD:.*]] = load ptr, ptr %[[FPGEP]]
|
|
// X64-NEXT: %[[CALL:.*]] = call noundef ptr %[[FPLOAD]](ptr noundef nonnull align 8 dereferenceable(8) %[[DEL_PTR]], i32 noundef 3)
|
|
// X86-NEXT: %[[CALL:.*]] = call x86_thiscallcc noundef ptr %[[FPLOAD]](ptr noundef nonnull align 4 dereferenceable(4) %[[DEL_PTR]], i32 noundef 3)
|
|
// CHECK-NEXT: br label %delete.end
|
|
// CHECK: delete.end:
|
|
// CHECK-NEXT: ret void
|
|
|
|
// Vector dtor definition for Parrot.
|
|
// X64-LABEL: define weak dso_local noundef ptr @"??_EParrot@@UEAAPEAXI@Z"(
|
|
// X64-SAME: ptr {{.*}} %[[THIS:.*]], i32 {{.*}} %[[IMPLICIT_PARAM:.*]]) unnamed_addr
|
|
// X86-LABEL: define weak dso_local x86_thiscallcc noundef ptr @"??_EParrot@@UAEPAXI@Z"(
|
|
// X86-SAME: ptr noundef nonnull align 4 dereferenceable(4) %[[THIS:.*]], i32 noundef %[[IMPLICIT_PARAM:.*]]) unnamed_addr
|
|
// CHECK: entry:
|
|
// CHECK-NEXT: %[[RET:.*]] = alloca ptr
|
|
// CHECK-NEXT: %[[IPADDR:.*]] = alloca i32
|
|
// CHECK-NEXT: %[[THISADDR:.*]] = alloca ptr
|
|
// CHECK-NEXT: store i32 %[[IMPLICIT_PARAM]], ptr %[[IPADDR]]
|
|
// CHECK-NEXT: store ptr %[[THIS]], ptr %[[THISADDR]]
|
|
// CHECK-NEXT: %[[LTHIS:.*]] = load ptr, ptr %[[THISADDR]]
|
|
// CHECK-NEXT: store ptr %[[LTHIS]], ptr %[[RET]]
|
|
// CHECK-NEXT: %[[LIP:.*]] = load i32, ptr %[[IPADDR]]
|
|
// CHECK-NEXT: %[[SECONDBIT:.*]] = and i32 %[[LIP]], 2
|
|
// CHECK-NEXT: %[[ISSECONDBITZERO:.*]] = icmp eq i32 %[[SECONDBIT]], 0
|
|
// CHECK-NEXT: br i1 %[[ISSECONDBITZERO:.*]], label %dtor.scalar, label %dtor.vector
|
|
// CHECK: dtor.vector:
|
|
// X64-NEXT: %[[COOKIEGEP:.*]] = getelementptr inbounds i8, ptr %[[LTHIS]], i64 -8
|
|
// X86-NEXT: %[[COOKIEGEP:.*]] = getelementptr inbounds i8, ptr %[[LTHIS]], i32 -4
|
|
// X64-NEXT: %[[HOWMANY:.*]] = load i64, ptr %[[COOKIEGEP]]
|
|
// X86-NEXT: %[[HOWMANY:.*]] = load i32, ptr %[[COOKIEGEP]]
|
|
// X64-NEXT: %[[END:.*]] = getelementptr inbounds %struct.Parrot, ptr %[[LTHIS]], i64 %[[HOWMANY]]
|
|
// X86-NEXT: %[[END:.*]] = getelementptr inbounds %struct.Parrot, ptr %[[LTHIS]], i32 %[[HOWMANY]]
|
|
// CHECK-NEXT: br label %arraydestroy.body
|
|
// CHECK: arraydestroy.body:
|
|
// CHECK-NEXT: %[[PASTELEM:.*]] = phi ptr [ %delete.end, %dtor.vector ], [ %arraydestroy.element, %arraydestroy.body ]
|
|
// X64-NEXT: %[[CURELEM:.*]] = getelementptr inbounds %struct.Parrot, ptr %[[PASTELEM]], i64 -1
|
|
// X86-NEXT: %[[CURELEM:.*]] = getelementptr inbounds %struct.Parrot, ptr %[[PASTELEM]], i32 -1
|
|
// X64-NEXT: call void @"??1Parrot@@UEAA@XZ"(ptr noundef nonnull align 8 dead_on_return(8) dereferenceable(8) %[[CURELEM]])
|
|
// X86-NEXT: call x86_thiscallcc void @"??1Parrot@@UAE@XZ"(ptr noundef nonnull align 4 dead_on_return(4) dereferenceable(4) %[[CURELEM]])
|
|
// CHECK-NEXT: %[[DONE:.*]] = icmp eq ptr %[[CURELEM]], %[[LTHIS]]
|
|
// CHECK-NEXT: br i1 %[[DONE]], label %arraydestroy.done3, label %arraydestroy.body
|
|
// CHECK: arraydestroy.done3:
|
|
// CHECK-NEXT: br label %dtor.vector.cont
|
|
// CHECK: dtor.vector.cont:
|
|
// CHECK-NEXT: %[[FIRSTBIT:.*]] = and i32 %[[LIP]], 1
|
|
// CHECK-NEXT: %[[ISFIRSTBITZERO:.*]] = icmp eq i32 %[[FIRSTBIT]], 0
|
|
// CHECK-NEXT: br i1 %[[ISFIRSTBITZERO]], label %dtor.continue, label %dtor.call_delete_after_array_destroy
|
|
// CHECK: dtor.call_delete_after_array_destroy:
|
|
// X64-NEXT: %[[ARRSZ:.*]] = mul i64 8, %[[HOWMANY]]
|
|
// X86-NEXT: %[[ARRSZ:.*]] = mul i32 4, %[[HOWMANY]]
|
|
// X64-NEXT: %[[TOTALSZ:.*]] = add i64 %[[ARRSZ]], 8
|
|
// X86-NEXT: %[[TOTALSZ:.*]] = add i32 %[[ARRSZ]], 4
|
|
// X64-NEXT: call void @"??_V@YAXPEAX_K@Z"(ptr noundef %[[COOKIEGEP]], i64 noundef %[[TOTALSZ]])
|
|
// X86-NEXT: call void @"??_V@YAXPAXI@Z"(ptr noundef %[[COOKIEGEP]], i32 noundef %[[TOTALSZ]])
|
|
// CHECK-NEXT: br label %dtor.continue
|
|
// CHECK: dtor.scalar:
|
|
// X64-NEXT: call void @"??1Parrot@@UEAA@XZ"(ptr noundef nonnull align 8 dead_on_return(8) dereferenceable(8) %[[LTHIS]])
|
|
// X86-NEXT: call x86_thiscallcc void @"??1Parrot@@UAE@XZ"(ptr noundef nonnull align 4 dead_on_return(4) dereferenceable(4) %[[LTHIS]])
|
|
// CHECK-NEXT: %[[FIRSTBIT:.*]] = and i32 %[[LIP]], 1
|
|
// CHECK-NEXT: %[[ISFIRSTBITZERO:.*]] = icmp eq i32 %[[FIRSTBIT]], 0
|
|
// CHECK-NEXT: br i1 %[[ISFIRSTBITZERO]], label %dtor.continue, label %dtor.call_delete
|
|
// CHECK: dtor.call_delete:
|
|
// X64-NEXT: call void @"??3@YAXPEAX_K@Z"(ptr noundef %[[LTHIS]], i64 noundef 8)
|
|
// X86-NEXT: call void @"??3@YAXPAXI@Z"(ptr noundef %[[LTHIS]], i32 noundef 4)
|
|
// CHECK-NEXT: br label %dtor.continue
|
|
// CHECK: dtor.continue:
|
|
// CHECK-NEXT: %[[LOADRET:.*]] = load ptr, ptr %[[RET]]
|
|
// CHECK-NEXT: ret ptr %[[LOADRET]]
|
|
|
|
// X64: define weak dso_local noundef ptr @"??_EJustAWeirdBird@@UEAAPEAXI@Z"(
|
|
// X64-SAME: ptr noundef nonnull align 8 dereferenceable(8) %this, i32 noundef %should_call_delete)
|
|
// CLANG21: define linkonce_odr dso_local noundef ptr @"??_GJustAWeirdBird@@UEAAPEAXI@Z"(
|
|
// X86: define weak dso_local x86_thiscallcc noundef ptr @"??_EJustAWeirdBird@@UAEPAXI@Z"(
|
|
// X86-SAME: ptr noundef nonnull align 4 dereferenceable(4) %this, i32 noundef %should_call_delete) unnamed_addr
|
|
|
|
// X64-LABEL: define weak dso_local noundef ptr @"??_EHasOperatorDelete@@UEAAPEAXI@Z"
|
|
// X86-LABEL: define weak dso_local x86_thiscallcc noundef ptr @"??_EHasOperatorDelete@@UAEPAXI@Z"
|
|
// CLANG21: define linkonce_odr dso_local noundef ptr @"??_GHasOperatorDelete@@UEAAPEAXI@Z"
|
|
// CHECK: dtor.call_delete_after_array_destroy:
|
|
// CHECK-NEXT: %[[SHOULD_CALL_GLOB_DELETE:.*]] = and i32 %should_call_delete2, 4
|
|
// CHECK-NEXT: %[[CHK:.*]] = icmp eq i32 %[[SHOULD_CALL_GLOB_DELETE]], 0
|
|
// CHECK-NEXT: br i1 %[[CHK]], label %dtor.call_class_delete_after_array_destroy, label %dtor.call_glob_delete_after_array_destroy
|
|
// CHECK: dtor.call_class_delete_after_array_destroy:
|
|
// X64-NEXT: call void @"??_VHasOperatorDelete@@SAXPEAX@Z"(ptr noundef %2)
|
|
// X86-NEXT: call void @"??_VHasOperatorDelete@@SAXPAX@Z"
|
|
// CHECK-NEXT: br label %dtor.continue
|
|
// CHECK: dtor.call_glob_delete_after_array_destroy:
|
|
// X64-NEXT: %[[ARRSZ:.*]] = mul i64 8, %[[COOKIE:.*]]
|
|
// X86-NEXT: %[[ARRSZ:.*]] = mul i32 4, %[[COOKIE:.*]]
|
|
// X64-NEXT: %[[TOTALSZ:.*]] = add i64 %[[ARRSZ]], 8
|
|
// X86-NEXT: %[[TOTALSZ:.*]] = add i32 %[[ARRSZ]], 4
|
|
// X64-NEXT: call void @"??_V@YAXPEAX_K@Z"(ptr noundef %2, i64 noundef %[[TOTALSZ]])
|
|
// X86-NEXT: call void @"??_V@YAXPAXI@Z"(ptr noundef %2, i32 noundef %[[TOTALSZ]])
|
|
// CHECK-NEXT: br label %dtor.continue
|
|
|
|
|
|
|
|
struct BaseDelete1 {
|
|
void operator delete[](void *);
|
|
};
|
|
struct BaseDelete2 {
|
|
void operator delete[](void *);
|
|
};
|
|
struct BaseDestructor {
|
|
BaseDestructor() {}
|
|
virtual ~BaseDestructor() = default;
|
|
};
|
|
|
|
struct Derived : BaseDelete1, BaseDelete2, BaseDestructor {
|
|
Derived() {}
|
|
};
|
|
|
|
void foobartest() {
|
|
Derived *a = new Derived[10]();
|
|
::delete[] a;
|
|
}
|
|
|
|
// X64-LABEL: define weak dso_local noundef ptr @"??_EDerived@@UEAAPEAXI@Z"(ptr {{.*}} %this, i32 noundef %should_call_delete)
|
|
// X86-LABEL: define weak dso_local x86_thiscallcc noundef ptr @"??_EDerived@@UAEPAXI@Z"(ptr {{.*}} %this, i32 noundef %should_call_delete)
|
|
// CHECK: %retval = alloca ptr
|
|
// CHECK-NEXT: %should_call_delete.addr = alloca i32, align 4
|
|
// CHECK-NEXT: %this.addr = alloca ptr
|
|
// CHECK-NEXT: store i32 %should_call_delete, ptr %should_call_delete.addr, align 4
|
|
// CHECK-NEXT: store ptr %this, ptr %this.addr
|
|
// CHECK-NEXT: %this1 = load ptr, ptr %this.addr
|
|
// CHECK-NEXT: store ptr %this1, ptr %retval
|
|
// CHECK-NEXT: %should_call_delete2 = load i32, ptr %should_call_delete.addr, align 4
|
|
// CHECK-NEXT: %0 = and i32 %should_call_delete2, 2
|
|
// CHECK-NEXT: %1 = icmp eq i32 %0, 0
|
|
// CHECK-NEXT: br i1 %1, label %dtor.scalar, label %dtor.vector
|
|
// CHECK: dtor.vector:
|
|
// X64-NEXT: %2 = getelementptr inbounds i8, ptr %this1, i64 -8
|
|
// X86-NEXT: %2 = getelementptr inbounds i8, ptr %this1, i32 -4
|
|
// X64-NEXT: %3 = load i64, ptr %2
|
|
// X86-NEXT: %3 = load i32, ptr %2
|
|
// X64-NEXT: %delete.end = getelementptr inbounds %struct.Derived, ptr %this1, i64 %3
|
|
// X86-NEXT: %delete.end = getelementptr inbounds %struct.Derived, ptr %this1, i32 %3
|
|
// CHECK-NEXT: br label %arraydestroy.body
|
|
// CHECK: arraydestroy.body:
|
|
// CHECK-NEXT: %arraydestroy.elementPast = phi ptr [ %delete.end, %dtor.vector ], [ %arraydestroy.element, %arraydestroy.body ]
|
|
// X64-NEXT: %arraydestroy.element = getelementptr inbounds %struct.Derived, ptr %arraydestroy.elementPast, i64 -1
|
|
// X86-NEXT: %arraydestroy.element = getelementptr inbounds %struct.Derived, ptr %arraydestroy.elementPast, i32 -1
|
|
// X64-NEXT: call void @"??1Derived@@UEAA@XZ"(ptr noundef nonnull align 8 dead_on_return(16) dereferenceable(16) %arraydestroy.element)
|
|
// X86-NEXT: call x86_thiscallcc void @"??1Derived@@UAE@XZ"(ptr noundef nonnull align 4 dead_on_return(8) dereferenceable(8) %arraydestroy.element)
|
|
// CHECK-NEXT: %arraydestroy.done = icmp eq ptr %arraydestroy.element, %this1
|
|
// CHECK-NEXT: br i1 %arraydestroy.done, label %arraydestroy.done3, label %arraydestroy.body
|
|
// CHECK: arraydestroy.done3:
|
|
// CHECK-NEXT: br label %dtor.vector.cont
|
|
// CHECK: dtor.vector.cont:
|
|
// CHECK-NEXT: %4 = and i32 %should_call_delete2, 1
|
|
// CHECK-NEXT: %5 = icmp eq i32 %4, 0
|
|
// CHECK-NEXT: br i1 %5, label %dtor.continue, label %dtor.call_delete_after_array_destroy
|
|
// CHECK: dtor.call_delete_after_array_destroy:
|
|
// X64-NEXT: %[[ARRSZ:.*]] = mul i64 16, %[[COOKIE:.*]]
|
|
// X86-NEXT: %[[ARRSZ:.*]] = mul i32 8, %[[COOKIE:.*]]
|
|
// X64-NEXT: %[[TOTALSZ:.*]] = add i64 %[[ARRSZ]], 8
|
|
// X86-NEXT: %[[TOTALSZ:.*]] = add i32 %[[ARRSZ]], 4
|
|
// X64-NEXT: call void @"??_V@YAXPEAX_K@Z"(ptr noundef %2, i64 noundef %[[TOTALSZ]])
|
|
// X86-NEXT: call void @"??_V@YAXPAXI@Z"(ptr noundef %2, i32 noundef %[[TOTALSZ]])
|
|
// CHECK-NEXT: br label %dtor.continue
|
|
// CHECK: dtor.scalar:
|
|
// X64-NEXT: call void @"??1Derived@@UEAA@XZ"(ptr noundef nonnull align 8 dead_on_return(16) dereferenceable(16) %this1)
|
|
// X86-NEXT: call x86_thiscallcc void @"??1Derived@@UAE@XZ"(ptr noundef nonnull align 4 dead_on_return(8) dereferenceable(8) %this1)
|
|
// CHECK-NEXT: %8 = and i32 %should_call_delete2, 1
|
|
// CHECK-NEXT: %9 = icmp eq i32 %8, 0
|
|
// CHECK-NEXT: br i1 %9, label %dtor.continue, label %dtor.call_delete
|
|
// CHECK: dtor.call_delete:
|
|
// X64-NEXT: call void @"??3@YAXPEAX_K@Z"(ptr noundef %this1, i64 noundef 16)
|
|
// X86-NEXT: call void @"??3@YAXPAXI@Z"(ptr noundef %this1, i32 noundef 8)
|
|
// CHECK-NEXT: br label %dtor.continue
|
|
// CHECK: dtor.continue:
|
|
// CHECK-NEXT: %10 = load ptr, ptr %retval
|
|
// CHECK-NEXT: ret ptr %10
|
|
|
|
// X64: define weak dso_local noundef ptr @"??_EAllocatedAsArray@@UEAAPEAXI@Z"
|
|
// X86: define weak dso_local x86_thiscallcc noundef ptr @"??_EAllocatedAsArray@@UAEPAXI@Z"
|
|
// CLANG21: define linkonce_odr dso_local noundef ptr @"??_GAllocatedAsArray@@UEAAPEAXI@Z"
|