diff options
-rw-r--r-- | lib/Sema/SemaTemplateInstantiateDecl.cpp | 16 | ||||
-rw-r--r-- | test/SemaCXX/libstdcxx_common_type_hack.cpp | 33 | ||||
-rw-r--r-- | www/cxx_status.html | 7 |
3 files changed, 53 insertions, 3 deletions
diff --git a/lib/Sema/SemaTemplateInstantiateDecl.cpp b/lib/Sema/SemaTemplateInstantiateDecl.cpp index ee9d529c69..19c46ab9c9 100644 --- a/lib/Sema/SemaTemplateInstantiateDecl.cpp +++ b/lib/Sema/SemaTemplateInstantiateDecl.cpp @@ -158,6 +158,22 @@ Decl *TemplateDeclInstantiator::InstantiateTypedefNameDecl(TypedefNameDecl *D, SemaRef.MarkDeclarationsReferencedInType(D->getLocation(), DI->getType()); } + // HACK: g++ has a bug where it gets the value kind of ?: wrong. + // libstdc++ relies upon this bug in its implementation of common_type. + // If we happen to be processing that implementation, fake up the g++ ?: + // semantics. See LWG issue 2141 for more information on the bug. + const DecltypeType *DT = DI->getType()->getAs<DecltypeType>(); + CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D->getDeclContext()); + if (DT && RD && isa<ConditionalOperator>(DT->getUnderlyingExpr()) && + DT->isReferenceType() && + RD->getEnclosingNamespaceContext() == SemaRef.getStdNamespace() && + RD->getIdentifier() && RD->getIdentifier()->isStr("common_type") && + D->getIdentifier() && D->getIdentifier()->isStr("type") && + SemaRef.getSourceManager().isInSystemHeader(D->getLocStart())) + // Fold it to the (non-reference) type which g++ would have produced. + DI = SemaRef.Context.getTrivialTypeSourceInfo( + DI->getType().getNonReferenceType()); + // Create the new typedef TypedefNameDecl *Typedef; if (IsTypeAlias) diff --git a/test/SemaCXX/libstdcxx_common_type_hack.cpp b/test/SemaCXX/libstdcxx_common_type_hack.cpp new file mode 100644 index 0000000000..e9cb22f9da --- /dev/null +++ b/test/SemaCXX/libstdcxx_common_type_hack.cpp @@ -0,0 +1,33 @@ +// RUN: %clang_cc1 -fsyntax-only %s -std=c++11 -verify + +// This is a test for an egregious hack in Clang that works around +// an issue with GCC's <type_traits> implementation. std::common_type +// relies on pre-standard rules for decltype(), in which it doesn't +// produce reference types so frequently. + +#ifdef BE_THE_HEADER + +#pragma GCC system_header +namespace std { + template<typename T> T &&declval(); + + template<typename...Ts> struct common_type {}; + template<typename A, typename B> struct common_type<A, B> { + // Under the rules in the standard, this always produces a + // reference type. + typedef decltype(true ? declval<A>() : declval<B>()) type; + }; +} + +#else + +#define BE_THE_HEADER +#include "libstdcxx_common_type_hack.cpp" + +using T = int; +using T = std::common_type<int, int>::type; + +using U = int; // expected-note {{here}} +using U = decltype(true ? std::declval<int>() : std::declval<int>()); // expected-error {{different types}} + +#endif diff --git a/www/cxx_status.html b/www/cxx_status.html index dbf5797c05..b4821a1fd8 100644 --- a/www/cxx_status.html +++ b/www/cxx_status.html @@ -40,10 +40,11 @@ <p>You can use Clang in C++11 mode either with <a href="http://libcxx.llvm.org/">libc++</a> or with gcc's libstdc++. -Patches are needed to make <a href="libstdc++4.4-clang0x.patch">libstdc++-4.4</a>, +Patches are needed to make <a href="libstdc++4.4-clang0x.patch">libstdc++-4.4</a> +work with Clang in C++11 mode. Patches are also needed to make <a href="libstdc++4.6-clang11.patch">libstdc++-4.6</a>, -and <a href="libstdc++4.7-clang11.patch">libstdc++-4.7</a> work with Clang in -C++11 mode.</p> +and <a href="libstdc++4.7-clang11.patch">libstdc++-4.7</a> work with Clang +releases prior to version 3.2 in C++11 mode.</p> <table width="689" border="1" cellspacing="0"> <tr> |