// RUN: %clang_cc1 -triple armv7a-linux-gnueabi -emit-llvm -o - -x c %s | FileCheck %s --check-prefixes=CHECK,C // RUN: %clang_cc1 -triple armv7a-linux-gnueabi -emit-llvm -o - %s | FileCheck %s --check-prefixes=CHECK,CXX // RUN: %clang_cc1 -triple armv7a-linux-gnueabi -emit-llvm -o - %s -fclang-abi-compat=19 | FileCheck %s --check-prefixes=CHECK,CXXCLANG19 // RUN: %clang_cc1 -triple thumbv7k-apple-watchos2.0 -target-abi aapcs16 -emit-llvm -o - %s | FileCheck %s --check-prefixes=CHECK,WATCHOS // Empty structs are ignored for PCS purposes on WatchOS and in C mode // elsewhere. In C++ mode they consume a register slot though. Functions are // slightly bigger than minimal to make confirmation against actual GCC // behaviour easier. #if __cplusplus #define EXTERNC extern "C" #else #define EXTERNC #endif struct Empty {}; // C: define{{.*}} i32 @empty_arg(i32 noundef %a) // CXX: define{{.*}} i32 @empty_arg(i8 %e.coerce, i32 noundef %a) // CXXCLANG19: define{{.*}} i32 @empty_arg(i32 noundef %a) // WATCHOS: define{{.*}} i32 @empty_arg(i32 noundef %a) EXTERNC int empty_arg(struct Empty e, int a) { return a; } // C: define{{.*}} void @empty_ret() // CXX: define{{.*}} void @empty_ret() // CXXCLANG19: define{{.*}} void @empty_ret() // WATCHOS: define{{.*}} void @empty_ret() EXTERNC struct Empty empty_ret(void) { struct Empty e; return e; } // However, what counts as "empty" is a baroque mess. This is super-empty, it's // ignored even in C++ mode. It also has sizeof == 0, violating C++, but that's // legacy for you: struct SuperEmpty { int arr[0]; }; // C: define{{.*}} i32 @super_empty_arg(i32 noundef %a) // CXX: define{{.*}} i32 @super_empty_arg(i32 noundef %a) // CXXCLANG19: define{{.*}} i32 @super_empty_arg(i32 noundef %a) // WATCHOS: define{{.*}} i32 @super_empty_arg(i32 noundef %a) EXTERNC int super_empty_arg(struct SuperEmpty e, int a) { return a; } struct SortOfEmpty { struct SuperEmpty e; }; // C: define{{.*}} i32 @sort_of_empty_arg(i32 noundef %a) // CXX: define{{.*}} i32 @sort_of_empty_arg(i8 %e.coerce, i32 noundef %a) // CXXCLANG19: define{{.*}} i32 @sort_of_empty_arg(i32 noundef %a) // WATCHOS: define{{.*}} i32 @sort_of_empty_arg(i32 noundef %a) EXTERNC int sort_of_empty_arg(struct Empty e, int a) { return a; } // C: define{{.*}} void @sort_of_empty_ret() // CXX: define{{.*}} void @sort_of_empty_ret() // CXXCLANG19: define{{.*}} void @sort_of_empty_ret() // WATCHOS: define{{.*}} void @sort_of_empty_ret() EXTERNC struct SortOfEmpty sort_of_empty_ret(void) { struct SortOfEmpty e; return e; } #include // va_arg matches the above rules, consuming an incoming argument in cases // where one would be passed, and not doing so when the argument should be // ignored. EXTERNC int empty_arg_variadic(int a, ...) { // CHECK-LABEL: @empty_arg_variadic( // C: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // C-NOT: {{ getelementptr }} // CXX: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // CXX: %argp.next2 = getelementptr inbounds i8, ptr %argp.cur1, i32 4 // CXXCLANG19: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // CXXCLANG19-NOT: {{ getelementptr }} // WATCHOS: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // WATCHOS-NOT: {{ getelementptr }} va_list vl; va_start(vl, a); struct Empty b = va_arg(vl, struct Empty); int c = va_arg(vl, int); va_end(vl); return c; } EXTERNC int super_empty_arg_variadic(int a, ...) { // CHECK-LABEL: @super_empty_arg_variadic( // C: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // C-NOT: {{ getelementptr }} // CXX: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // CXX-NOT: {{ getelementptr }} // CXXCLANG19: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // CXXCLANG19-NOT: {{ getelementptr }} // WATCHOS: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // WATCHOS-NOT: {{ getelementptr }} va_list vl; va_start(vl, a); struct SuperEmpty b = va_arg(vl, struct SuperEmpty); int c = va_arg(vl, int); va_end(vl); return c; } EXTERNC int sort_of_empty_arg_variadic(int a, ...) { // CHECK-LABEL: @sort_of_empty_arg_variadic( // C: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // C-NOT: {{ getelementptr }} // CXX: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // CXX-NOT: {{ getelementptr }} // CXXCLANG19: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // CXXCLANG19-NOT: {{ getelementptr }} // WATCHOS: %argp.next = getelementptr inbounds i8, ptr %argp.cur, i32 4 // WATCHOS-NOT: {{ getelementptr }} va_list vl; va_start(vl, a); struct SortOfEmpty b = va_arg(vl, struct SortOfEmpty); int c = va_arg(vl, int); va_end(vl); return c; }