diff options
| author | Richard Smith <richard-llvm@metafoo.co.uk> | 2016-03-25 00:08:53 +0000 |
|---|---|---|
| committer | Richard Smith <richard-llvm@metafoo.co.uk> | 2016-03-25 00:08:53 +0000 |
| commit | 95853077c3f3f959af93a6b50d751834aea1f8d6 (patch) | |
| tree | 50f618d38ddadfe341b39fd692db62fa3c2e63d3 | |
| parent | e0451c9378337a7387dc08d549a0a4e33a470c58 (diff) | |
| download | bcm5719-llvm-95853077c3f3f959af93a6b50d751834aea1f8d6.tar.gz bcm5719-llvm-95853077c3f3f959af93a6b50d751834aea1f8d6.zip | |
Fix nondeterminism in computation of builtin operator overload sets.
llvm-svn: 264363
| -rw-r--r-- | clang/lib/Sema/SemaOverload.cpp | 7 | ||||
| -rw-r--r-- | clang/test/SemaCXX/diagnostic-order.cpp | 45 |
2 files changed, 45 insertions, 7 deletions
diff --git a/clang/lib/Sema/SemaOverload.cpp b/clang/lib/Sema/SemaOverload.cpp index cf7fb557a75..e9154bc12a5 100644 --- a/clang/lib/Sema/SemaOverload.cpp +++ b/clang/lib/Sema/SemaOverload.cpp @@ -6818,7 +6818,8 @@ namespace { /// enumeration types. class BuiltinCandidateTypeSet { /// TypeSet - A set of types. - typedef llvm::SmallPtrSet<QualType, 8> TypeSet; + typedef llvm::SetVector<QualType, SmallVector<QualType, 8>, + llvm::SmallPtrSet<QualType, 8>> TypeSet; /// PointerTypes - The set of pointer types that will be used in the /// built-in candidates. @@ -6917,7 +6918,7 @@ BuiltinCandidateTypeSet::AddPointerWithMoreQualifiedTypeVariants(QualType Ty, const Qualifiers &VisibleQuals) { // Insert this type. - if (!PointerTypes.insert(Ty).second) + if (!PointerTypes.insert(Ty)) return false; QualType PointeeTy; @@ -6985,7 +6986,7 @@ bool BuiltinCandidateTypeSet::AddMemberPointerWithMoreQualifiedTypeVariants( QualType Ty) { // Insert this type. - if (!MemberPointerTypes.insert(Ty).second) + if (!MemberPointerTypes.insert(Ty)) return false; const MemberPointerType *PointerTy = Ty->getAs<MemberPointerType>(); diff --git a/clang/test/SemaCXX/diagnostic-order.cpp b/clang/test/SemaCXX/diagnostic-order.cpp index f0899018f7f..b3b270bfc93 100644 --- a/clang/test/SemaCXX/diagnostic-order.cpp +++ b/clang/test/SemaCXX/diagnostic-order.cpp @@ -3,9 +3,9 @@ // Ensure that the diagnostics we produce for this situation appear in a // deterministic order. This requires ADL to provide lookup results in a // deterministic order. -template<typename T> struct Error { typedef typename T::error error; }; -struct X { template<typename T> friend typename Error<T>::error f(X, T); }; -struct Y { template<typename T> friend typename Error<T>::error f(T, Y); }; +template<typename T, typename> struct Error { typedef typename T::error error; }; +struct X { template<typename T> friend typename Error<X, T>::error f(X, T); }; +struct Y { template<typename T> friend typename Error<Y, T>::error f(T, Y); }; void g() { f(X(), Y()); @@ -15,6 +15,43 @@ void g() { // order below is source order, which seems best). The crucial fact is that // there is one single order that is stable across multiple runs of clang. // -// CHECK: no type named 'error' in 'Y' // CHECK: no type named 'error' in 'X' +// CHECK: no type named 'error' in 'Y' // CHECK: no matching function for call to 'f' + + +struct Oper { + template<typename T, typename U = typename Error<Oper, T>::error> operator T(); + + operator int*(); + operator float*(); + operator X*(); + operator Y*(); + + operator int(*[1])(); + operator int(*[2])(); + operator int(*[3])(); + operator int(*[4])(); + operator int(*[5])(); + operator int(*[6])(); + operator int(*[7])(); + operator int(*[8])(); + operator float(*[1])(); + operator float(*[2])(); + operator float(*[3])(); + operator float(*[4])(); + operator float(*[5])(); + operator float(*[6])(); + operator float(*[7])(); + operator float(*[8])(); +}; +int *p = Oper() + 0; + +// CHECK: no type named 'error' in 'Oper' +// CHECK: in instantiation of template class 'Error<Oper, int *>' +// CHECK: no type named 'error' in 'Oper' +// CHECK: in instantiation of template class 'Error<Oper, float *>' +// CHECK: no type named 'error' in 'Oper' +// CHECK: in instantiation of template class 'Error<Oper, X *>' +// CHECK: no type named 'error' in 'Oper' +// CHECK: in instantiation of template class 'Error<Oper, Y *>' |

