/* Test semantics of #pragma pack. Contributed by Mike Coleman */ /* { dg-do compile { target { ! default_packed } } } */ /* Mainly we're just testing whether pushing and popping seem to be working correctly, and verifying the (alignment == 1) case, which is really the only reason anyone would use this pragma anyway. */ #include /* generalized compile-time test expression */ #define test(n, expr) int test_##n [(expr) ? 1 : -1] /* Round V down to multiple of A */ #define floor(v,a) ((v) / (a) * (a)) /* Offset of field with alignment A in structure S after a field P of type PT */ #define offset(s,p,pt,a) \ floor ((offsetof(struct s, p) + sizeof (pt) + (a) - 1), a) /* regular minimum */ #define min(a,b) ((a) < (b) ? (a) : (b)) /* Check that field A (type AT) followed by field B (type BT) are packed according to P */ #define test_pack(n, a, at, b, bt, p) \ test(n, offsetof (struct SNAME, b) \ == min (offset (SNAME,a,at,__alignof__(bt)), \ offset (SNAME,a,at,p))) /* Test offset of field F in structs s1 and s2 are the same. */ #define test_offset(n, s1, s2, f) \ test (n, (offsetof(struct s1, f) == offsetof(struct s2, f))) #define SNAME s0 #include "pack-test-1.h" #undef SNAME #define SNAME s1 #pragma pack(push, p1, 1) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 1); test_pack(1, f2, short, f3, double, 1); test_pack(2, f4, int, f5, double, 1); } #undef SNAME #define SNAME s2 #pragma pack(push, p2, 2) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 2); test_pack(1, f2, short, f3, double, 2); test_pack(2, f4, int, f5, double, 2); } #undef SNAME #define SNAME s3 #pragma pack(push, p3, 4) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 4); test_pack(1, f2, short, f3, double, 4); test_pack(2, f4, int, f5, double, 4); } #undef SNAME #define SNAME s4 #pragma pack(pop) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 2); test_pack(1, f2, short, f3, double, 2); test_pack(2, f4, int, f5, double, 2); } #undef SNAME #define SNAME s5 #pragma pack(pop, p2) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 1); test_pack(1, f2, short, f3, double, 1); test_pack(2, f4, int, f5, double, 1); } #undef SNAME #define SNAME s6 #pragma pack(pop, p1) #include "pack-test-1.h" void SNAME() { test_offset (0, s0, SNAME, f0); test_offset (1, s0, SNAME, f1); test_offset (2, s0, SNAME, f2); test_offset (3, s0, SNAME, f3); test_offset (4, s0, SNAME, f4); test_offset (5, s0, SNAME, f5); } #undef SNAME #define SNAME s7 #pragma pack(1) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 1); test_pack(1, f2, short, f3, double, 1); test_pack(2, f4, int, f5, double, 1); } #undef SNAME #define SNAME s8 #pragma pack(push, p2, 2) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 2); test_pack(1, f2, short, f3, double, 2); test_pack(2, f4, int, f5, double, 2); } #undef SNAME #define SNAME s9 #pragma pack(pop) #include "pack-test-1.h" void SNAME() { test_pack(0, f0, char, f1, double, 1); test_pack(1, f2, short, f3, double, 1); test_pack(2, f4, int, f5, double, 1); } #undef SNAME #define SNAME s10 #pragma pack() #include "pack-test-1.h" void SNAME() { test_offset (0, s0, SNAME, f0); test_offset (1, s0, SNAME, f1); test_offset (2, s0, SNAME, f2); test_offset (3, s0, SNAME, f3); test_offset (4, s0, SNAME, f4); test_offset (5, s0, SNAME, f5); }