// 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 struct S { void foo() { void *p = new C(); delete (C *)p; } }; S 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 dereferenceable(8) %[[CURELEM]]) // X86-NEXT: call x86_thiscallcc void @"??1Parrot@@UAE@XZ"(ptr noundef nonnull align 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 dereferenceable(8) %[[LTHIS]]) // X86-NEXT: call x86_thiscallcc void @"??1Parrot@@UAE@XZ"(ptr noundef nonnull align 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 dereferenceable(16) %arraydestroy.element) // X86-NEXT: call x86_thiscallcc void @"??1Derived@@UAE@XZ"(ptr noundef nonnull align 4 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 dereferenceable(16) %this1) // X86-NEXT: call x86_thiscallcc void @"??1Derived@@UAE@XZ"(ptr noundef nonnull align 4 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"