From b687f3ba4cdc3fd6e4fa9a89addc1efa61761eeb Mon Sep 17 00:00:00 2001 From: Logan Chien Date: Sat, 20 Oct 2012 06:11:33 +0000 Subject: [PATCH] Fix __builtin_va_arg assertion failure in ARM AAPCS. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@166369 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/Sema/SemaExpr.cpp | 10 ++++++ test/SemaCXX/builtins-arm.cpp | 6 ++++ test/SemaCXX/builtins-va_arg.cpp | 52 ++++++++++++++++++++++++++++++++ 3 files changed, 68 insertions(+) create mode 100644 test/SemaCXX/builtins-arm.cpp create mode 100644 test/SemaCXX/builtins-va_arg.cpp diff --git a/lib/Sema/SemaExpr.cpp b/lib/Sema/SemaExpr.cpp index a71da15322..6a6fe897c5 100644 --- a/lib/Sema/SemaExpr.cpp +++ b/lib/Sema/SemaExpr.cpp @@ -9643,6 +9643,16 @@ ExprResult Sema::BuildVAArgExpr(SourceLocation BuiltinLoc, if (Result.isInvalid()) return ExprError(); E = Result.take(); + } else if (VaListType->isRecordType() && getLangOpts().CPlusPlus) { + // If va_list is a record type and we are compiling in C++ mode, + // check the argument using reference binding. + InitializedEntity Entity + = InitializedEntity::InitializeParameter(Context, + Context.getLValueReferenceType(VaListType), false); + ExprResult Init = PerformCopyInitialization(Entity, SourceLocation(), E); + if (Init.isInvalid()) + return ExprError(); + E = Init.takeAs(); } else { // Otherwise, the va_list argument must be an l-value because // it is modified by va_arg. diff --git a/test/SemaCXX/builtins-arm.cpp b/test/SemaCXX/builtins-arm.cpp new file mode 100644 index 0000000000..8a0cf8102b --- /dev/null +++ b/test/SemaCXX/builtins-arm.cpp @@ -0,0 +1,6 @@ +// RUN: %clang_cc1 -triple armv7 -fsyntax-only -verify %s + +// va_list on ARM AAPCS is struct { void* __ap }. +int test1(const __builtin_va_list &ap) { + return __builtin_va_arg(ap, int); // expected-error {{binding of reference to type '__builtin_va_list' to a value of type 'const __builtin_va_list' drops qualifiers}} +} diff --git a/test/SemaCXX/builtins-va_arg.cpp b/test/SemaCXX/builtins-va_arg.cpp new file mode 100644 index 0000000000..4f549c8db6 --- /dev/null +++ b/test/SemaCXX/builtins-va_arg.cpp @@ -0,0 +1,52 @@ +// RUN: %clang_cc1 %s -ffreestanding +// RUN: %clang_cc1 %s -ffreestanding -triple i686-unknown-linux +// RUN: %clang_cc1 %s -ffreestanding -triple x86_64-unknown-linux +// RUN: %clang_cc1 %s -ffreestanding -triple mips-unknown-linux +// RUN: %clang_cc1 %s -ffreestanding -triple mipsel-unknown-linux +// RUN: %clang_cc1 %s -ffreestanding -triple armv7-unknown-linux-gnueabi +// RUN: %clang_cc1 %s -ffreestanding -triple thumbv7-unknown-linux-gnueabi + +#include "stdarg.h" + +int int_accumulator = 0; +double double_accumulator = 0; + +int test_vprintf(const char *fmt, va_list ap) { + char ch; + int result = 0; + while (*fmt != '\0') { + ch = *fmt++; + if (ch != '%') { + continue; + } + + ch = *fmt++; + switch (ch) { + case 'd': + int_accumulator += va_arg(ap, int); + result++; + break; + + case 'f': + double_accumulator += va_arg(ap, double); + result++; + break; + + default: + break; + } + + if (ch == '0') { + break; + } + } + return result; +} + +int test_printf(const char *fmt, ...) { + va_list ap; + va_start(ap, fmt); + int result = test_vprintf(fmt, ap); + va_end(ap); + return result; +} -- 2.40.0