summaryrefslogtreecommitdiffstats
path: root/clang/test/CodeGenCXX/arm-vaarg.cpp
diff options
context:
space:
mode:
authorTim Northover <tnorthover@apple.com>2013-06-21 23:05:33 +0000
committerTim Northover <tnorthover@apple.com>2013-06-21 23:05:33 +0000
commit1711cc930bda8d27e87a2092bd220c18e4600c98 (patch)
treed729b08354346fbf0054495aaa45d9a9f1a62883 /clang/test/CodeGenCXX/arm-vaarg.cpp
parent97c6c5bd9869a3c929cef0880425c198b70b6d46 (diff)
downloadbcm5719-llvm-1711cc930bda8d27e87a2092bd220c18e4600c98.tar.gz
bcm5719-llvm-1711cc930bda8d27e87a2092bd220c18e4600c98.zip
Teach ARM va_arg to ignore empty structs.
Empty structs are ignored for parameter passing purposes, but va_arg was incrementing the pointer anyway which could lead to va_list getting out of sync. llvm-svn: 184605
Diffstat (limited to 'clang/test/CodeGenCXX/arm-vaarg.cpp')
-rw-r--r--clang/test/CodeGenCXX/arm-vaarg.cpp27
1 files changed, 27 insertions, 0 deletions
diff --git a/clang/test/CodeGenCXX/arm-vaarg.cpp b/clang/test/CodeGenCXX/arm-vaarg.cpp
new file mode 100644
index 00000000000..9850fb342fa
--- /dev/null
+++ b/clang/test/CodeGenCXX/arm-vaarg.cpp
@@ -0,0 +1,27 @@
+// RUN: %clang_cc1 -triple armv7-apple-ios -emit-llvm -o - %s | FileCheck %s
+struct Empty {};
+
+Empty emptyvar;
+
+int take_args(int a, ...) {
+ __builtin_va_list l;
+ __builtin_va_start(l, a);
+// CHECK: call void @llvm.va_start
+
+ emptyvar = __builtin_va_arg(l, Empty);
+// CHECK: load i8**
+// CHECK-NOT: getelementptr
+// CHECK: [[EMPTY_PTR:%[a-zA-Z0-9._]+]] = bitcast i8* {{%[a-zA-Z0-9._]+}} to %struct.Empty*
+
+ // It's conceivable that EMPTY_PTR may not actually be a valid pointer
+ // (e.g. it's at the very bottom of the stack and the next page is
+ // invalid). This doesn't matter provided it's never loaded (there's no
+ // well-defined way to tell), but it becomes a problem if we do try to use it.
+// CHECK-NOT: load %struct.Empty* [[EMPTY_PTR]]
+
+ int i = __builtin_va_arg(l, int);
+// CHECK: load i32*
+
+ __builtin_va_end(l);
+ return i;
+}
OpenPOWER on IntegriCloud