// RUN: %clang_cc1 -triple x86_64-unk-unk -emit-llvm -Os -o %t %s // RUN: FileCheck < %t %s struct s0 { unsigned int x[2] __attribute__((packed)); }; struct s1 { unsigned int x[2] __attribute__((packed)); unsigned int y; unsigned int z __attribute__((packed)); }; struct s2 { unsigned int x[2] __attribute__((packed)); unsigned int y __attribute__((packed)); unsigned int z __attribute__((packed)); }; struct __attribute__((packed)) s3 { unsigned int x[2]; unsigned int y; unsigned int z; }; // CHECK: @align0 ={{.*}} local_unnamed_addr global i32 1 int align0 = __alignof(struct s0); // CHECK: @align1 ={{.*}} local_unnamed_addr global i32 4 int align1 = __alignof(struct s1); // CHECK: @align2 ={{.*}} local_unnamed_addr global i32 1 int align2 = __alignof(struct s2); // CHECK: @align3 ={{.*}} local_unnamed_addr global i32 1 int align3 = __alignof(struct s3); // CHECK: @align0_x ={{.*}} local_unnamed_addr global i32 1 int align0_x = __alignof(((struct s0*) 0)->x); // // CHECK: @align1_x ={{.*}} local_unnamed_addr global i32 1 int align1_x = __alignof(((struct s1*) 0)->x); // CHECK: @align2_x ={{.*}} local_unnamed_addr global i32 1 int align2_x = __alignof(((struct s2*) 0)->x); // CHECK: @align3_x ={{.*}} local_unnamed_addr global i32 1 int align3_x = __alignof(((struct s3*) 0)->x); // CHECK: @align0_x0 ={{.*}} local_unnamed_addr global i32 4 int align0_x0 = __alignof(((struct s0*) 0)->x[0]); // CHECK: @align1_x0 ={{.*}} local_unnamed_addr global i32 4 int align1_x0 = __alignof(((struct s1*) 0)->x[0]); // CHECK: @align2_x0 ={{.*}} local_unnamed_addr global i32 4 int align2_x0 = __alignof(((struct s2*) 0)->x[0]); // CHECK: @align3_x0 ={{.*}} local_unnamed_addr global i32 4 int align3_x0 = __alignof(((struct s3*) 0)->x[0]); // CHECK-LABEL: define{{.*}} i32 @f0_a // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f0_b // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } int f0_a(struct s0 *a) { return a->x[1]; } int f0_b(struct s0 *a) { return *(a->x + 1); } // Note that 'y' still causes struct s1 to be four-byte aligned. // Note that we are incompatible with GCC on this example. // // CHECK-LABEL: define{{.*}} i32 @f1_a // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f1_b // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } // Note that we are incompatible with GCC on this example. // // CHECK-LABEL: define{{.*}} i32 @f1_c // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f1_d // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } int f1_a(struct s1 *a) { return a->x[1]; } int f1_b(struct s1 *a) { return *(a->x + 1); } int f1_c(struct s1 *a) { return a->y; } int f1_d(struct s1 *a) { return a->z; } // CHECK-LABEL: define{{.*}} i32 @f2_a // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f2_b // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f2_c // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f2_d // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } int f2_a(struct s2 *a) { return a->x[1]; } int f2_b(struct s2 *a) { return *(a->x + 1); } int f2_c(struct s2 *a) { return a->y; } int f2_d(struct s2 *a) { return a->z; } // CHECK-LABEL: define{{.*}} i32 @f3_a // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f3_b // CHECK: load i32, i32* %{{.*}}, align 4 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f3_c // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } // CHECK-LABEL: define{{.*}} i32 @f3_d // CHECK: load i32, i32* %{{.*}}, align 1 // CHECK: } int f3_a(struct s3 *a) { return a->x[1]; } int f3_b(struct s3 *a) { return *(a->x + 1); } int f3_c(struct s3 *a) { return a->y; } int f3_d(struct s3 *a) { return a->z; } // Verify we don't claim things are overaligned. // // CHECK-LABEL: define{{.*}} double @f4 // CHECK: load double, double* {{.*}}, align 8 // CHECK: } extern double g4[5] __attribute__((aligned(16))); double f4() { return g4[1]; }