158 lines
4.1 KiB
C
158 lines
4.1 KiB
C
|
// 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];
|
||
|
}
|