blob: fa4fa223bc2d9ea39e26c20ece71eee32bfba61b [file] [log] [blame] [edit]
// RUN: %clang_cc1 -triple arm64e-apple-ios18 -fptrauth-calls -fptrauth-intrinsics -fobjc-arc -emit-llvm -o - %s | FileCheck %s
// CHECK: %[[STRUCT_S0:.*]] = type { i32, i32, ptr }
// CHECK: %[[STRUCT_S1:.*]] = type { ptr, ptr }
// This struct isn't POD because it has an address-discriminated ptrauth
// field.
typedef struct {
int f0, f1;
int * __ptrauth(1,1,50) f2;
} S0;
// This struct isn't POD because it has an address-discriminated ptrauth
// field and an ARC ObjC pointer field.
typedef struct {
id f0;
int * __ptrauth(1,1,50) f1;
} S1;
// CHECK: define void @compound_literal_assignment0(ptr noundef %[[P:.*]])
// CHECK: %[[P_ADDR:.*]] = alloca ptr, align 8
// CHECK-NEXT: %[[_COMPOUNDLITERAL:.*]] = alloca %[[STRUCT_S0]], align 8
// CHECK-NEXT: store ptr %[[P]], ptr %[[P_ADDR]], align 8
// CHECK-NEXT: %[[V0:.*]] = load ptr, ptr %[[P_ADDR]], align 8
// CHECK-NEXT: %[[F0:.*]] = getelementptr inbounds nuw %[[STRUCT_S0]], ptr %[[_COMPOUNDLITERAL]], i32 0, i32 0
// CHECK-NEXT: %[[V1:.*]] = load ptr, ptr %[[P_ADDR]], align 8
// CHECK-NEXT: %[[F1:.*]] = getelementptr inbounds nuw %[[STRUCT_S0]], ptr %[[V1]], i32 0, i32 1
// CHECK-NEXT: %[[V2:.*]] = load i32, ptr %[[F1]], align 4
// CHECK-NEXT: store i32 %[[V2]], ptr %[[F0]], align 8
// CHECK-NEXT: %[[F11:.*]] = getelementptr inbounds nuw %[[STRUCT_S0]], ptr %[[_COMPOUNDLITERAL]], i32 0, i32 1
// CHECK-NEXT: %[[V3:.*]] = load ptr, ptr %[[P_ADDR]], align 8
// CHECK-NEXT: %[[F02:.*]] = getelementptr inbounds nuw %[[STRUCT_S0]], ptr %[[V3]], i32 0, i32 0
// CHECK-NEXT: %[[V4:.*]] = load i32, ptr %[[F02]], align 8
// CHECK-NEXT: store i32 %[[V4]], ptr %[[F11]], align 4
// CHECK-NEXT: %[[F2:.*]] = getelementptr inbounds nuw %[[STRUCT_S0]], ptr %[[_COMPOUNDLITERAL]], i32 0, i32 2
// CHECK-NEXT: store ptr null, ptr %[[F2]], align 8
// CHECK-NEXT: call void @__copy_assignment_8_8_t0w8_pa1_50_8(ptr %[[V0]], ptr %[[_COMPOUNDLITERAL]])
// CHECK-NEXT: ret void
void compound_literal_assignment0(S0 *p) {
*p = (S0){.f0 = p->f1, .f1 = p->f0};
}
// CHECK: define void @compound_literal_assignment1(ptr noundef %[[P:.*]])
// CHECK: %[[P_ADDR:.*]] = alloca ptr, align 8
// CHECK-NEXT: %[[_COMPOUNDLITERAL:.*]] = alloca %[[STRUCT_S1]], align 8
// CHECK-NEXT: store ptr %[[P]], ptr %[[P_ADDR]], align 8
// CHECK-NEXT: %[[V0:.*]] = load ptr, ptr %[[P_ADDR]], align 8
// CHECK-NEXT: %[[F0:.*]] = getelementptr inbounds nuw %[[STRUCT_S1]], ptr %[[_COMPOUNDLITERAL]], i32 0, i32 0
// CHECK-NEXT: store ptr null, ptr %[[F0]], align 8
// CHECK-NEXT: %[[F1:.*]] = getelementptr inbounds nuw %[[STRUCT_S1]], ptr %[[_COMPOUNDLITERAL]], i32 0, i32 1
// CHECK-NEXT: store ptr null, ptr %[[F1]], align 8
// CHECK-NEXT: call void @__copy_assignment_8_8_s0_pa1_50_8(ptr %[[V0]], ptr %[[_COMPOUNDLITERAL]])
// CHECK-NEXT: call void @__destructor_8_s0(ptr %[[_COMPOUNDLITERAL]])
// CHECK-NEXT: ret void
void compound_literal_assignment1(S1 *p) {
*p = (S1){};
}