summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRichard Smith <richard-llvm@metafoo.co.uk>2019-10-06 18:50:40 +0000
committerRichard Smith <richard-llvm@metafoo.co.uk>2019-10-06 18:50:40 +0000
commit1c3c183da3437d9580a66c719a1cb54522d530ee (patch)
treef8dd946dfae80a58fb17727fbbf311c35f1be2c3
parentc059e0ac05e9795b065bbbe18159f52fdf1f7086 (diff)
downloadclang-1c3c183da3437d9580a66c719a1cb54522d530ee.tar.gz
Implements CWG 1601 in [over.ics.rank/4.2]
Summary: The overload resolution for enums with a fixed underlying type has changed in the C++14 standard. This patch implements the new rule. Patch by Mark de Wever! Reviewers: rsmith Reviewed By: rsmith Subscribers: cfe-commits Tags: #clang Differential Revision: https://reviews.llvm.org/D65695 git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@373866 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r--lib/Sema/SemaOverload.cpp42
-rw-r--r--test/CXX/drs/dr16xx.cpp12
-rw-r--r--test/CXX/drs/dr6xx.cpp15
-rwxr-xr-xwww/cxx_dr_status.html2
4 files changed, 65 insertions, 6 deletions
diff --git a/lib/Sema/SemaOverload.cpp b/lib/Sema/SemaOverload.cpp
index 3db8e3d2e5..80bd3562bc 100644
--- a/lib/Sema/SemaOverload.cpp
+++ b/lib/Sema/SemaOverload.cpp
@@ -3765,6 +3765,34 @@ isBetterReferenceBindingKind(const StandardConversionSequence &SCS1,
!SCS2.IsLvalueReference && SCS2.BindsToFunctionLvalue);
}
+enum class FixedEnumPromotion {
+ None,
+ ToUnderlyingType,
+ ToPromotedUnderlyingType
+};
+
+/// Returns kind of fixed enum promotion the \a SCS uses.
+static FixedEnumPromotion
+getFixedEnumPromtion(Sema &S, const StandardConversionSequence &SCS) {
+
+ if (SCS.Second != ICK_Integral_Promotion)
+ return FixedEnumPromotion::None;
+
+ QualType FromType = SCS.getFromType();
+ if (!FromType->isEnumeralType())
+ return FixedEnumPromotion::None;
+
+ EnumDecl *Enum = FromType->getAs<EnumType>()->getDecl();
+ if (!Enum->isFixed())
+ return FixedEnumPromotion::None;
+
+ QualType UnderlyingType = Enum->getIntegerType();
+ if (S.Context.hasSameType(SCS.getToType(1), UnderlyingType))
+ return FixedEnumPromotion::ToUnderlyingType;
+
+ return FixedEnumPromotion::ToPromotedUnderlyingType;
+}
+
/// CompareStandardConversionSequences - Compare two standard
/// conversion sequences to determine whether one is better than the
/// other or if they are indistinguishable (C++ 13.3.3.2p3).
@@ -3806,6 +3834,20 @@ CompareStandardConversionSequences(Sema &S, SourceLocation Loc,
? ImplicitConversionSequence::Better
: ImplicitConversionSequence::Worse;
+ // C++14 [over.ics.rank]p4b2:
+ // This is retroactively applied to C++11 by CWG 1601.
+ //
+ // A conversion that promotes an enumeration whose underlying type is fixed
+ // to its underlying type is better than one that promotes to the promoted
+ // underlying type, if the two are different.
+ FixedEnumPromotion FEP1 = getFixedEnumPromtion(S, SCS1);
+ FixedEnumPromotion FEP2 = getFixedEnumPromtion(S, SCS2);
+ if (FEP1 != FixedEnumPromotion::None && FEP2 != FixedEnumPromotion::None &&
+ FEP1 != FEP2)
+ return FEP1 == FixedEnumPromotion::ToUnderlyingType
+ ? ImplicitConversionSequence::Better
+ : ImplicitConversionSequence::Worse;
+
// C++ [over.ics.rank]p4b2:
//
// If class B is derived directly or indirectly from class A,
diff --git a/test/CXX/drs/dr16xx.cpp b/test/CXX/drs/dr16xx.cpp
index b5047e8fe2..44d5b8f3f5 100644
--- a/test/CXX/drs/dr16xx.cpp
+++ b/test/CXX/drs/dr16xx.cpp
@@ -23,6 +23,18 @@ namespace std {
} // std
#endif
+namespace dr1601 { // dr1601: 10
+enum E : char { e };
+#if __cplusplus < 201103L
+ // expected-error@-2 {{enumeration types with a fixed underlying type are a C++11 extension}}
+#endif
+void f(char);
+void f(int);
+void g() {
+ f(e);
+}
+} // namespace dr1601
+
namespace dr1611 { // dr1611: dup 1658
struct A { A(int); };
struct B : virtual A { virtual void f() = 0; };
diff --git a/test/CXX/drs/dr6xx.cpp b/test/CXX/drs/dr6xx.cpp
index 530c88f86f..31e3571f50 100644
--- a/test/CXX/drs/dr6xx.cpp
+++ b/test/CXX/drs/dr6xx.cpp
@@ -987,14 +987,19 @@ namespace dr684 { // dr684: sup 1454
}
#endif
-#if __cplusplus >= 201103L
namespace dr685 { // dr685: yes
enum E : long { e };
+#if __cplusplus < 201103L
+ // expected-error@-2 {{enumeration types with a fixed underlying type are a C++11 extension}}
+#endif
void f(int);
int f(long);
int a = f(e);
enum G : short { g };
+#if __cplusplus < 201103L
+ // expected-error@-2 {{enumeration types with a fixed underlying type are a C++11 extension}}
+#endif
int h(short);
void h(long);
int b = h(g);
@@ -1007,11 +1012,11 @@ namespace dr685 { // dr685: yes
void j(long); // expected-note {{candidate}}
int d = j(g); // expected-error {{ambiguous}}
- int k(short); // expected-note {{candidate}}
- void k(int); // expected-note {{candidate}}
- int x = k(g); // expected-error {{ambiguous}}
+ // Valid per dr1601
+ int k(short);
+ void k(int);
+ int x = k(g);
}
-#endif
namespace dr686 { // dr686: yes
void f() {
diff --git a/www/cxx_dr_status.html b/www/cxx_dr_status.html
index 40af4732ef..83567e29f4 100755
--- a/www/cxx_dr_status.html
+++ b/www/cxx_dr_status.html
@@ -9421,7 +9421,7 @@ and <I>POD class</I></td>
<td><a href="http://wg21.link/cwg1601">1601</a></td>
<td>C++14</td>
<td>Promotion of enumeration with fixed underlying type</td>
- <td class="none" align="center">Unknown</td>
+ <td class="svn" align="center">SVN</td>
</tr>
<tr class="open" id="1602">
<td><a href="http://wg21.link/cwg1602">1602</a></td>