mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-17 15:56:40 +00:00

When checking deduction consistency, a substitution can be incomplete such that only sugar parts refer to non-deduced template parameters. This would not otherwise lead to an inconsistent deduction, so this patch makes it so we canonicalize the types before substitution in order to avoid that possibility, for now. When we are able to produce substitution failure diagnostics for partial ordering, we might want to improve the TemplateInstantiator so that it does not fail in that case. This fixes a regression on top of #100692, which was reported on the PR. This was never released, so there are no release notes.
124 lines
3.8 KiB
C++
124 lines
3.8 KiB
C++
// RUN: %clang_cc1 -std=c++23 -verify %s
|
|
|
|
namespace t1 {
|
|
template<bool> struct enable_if { typedef void type; };
|
|
template <class T> class Foo {};
|
|
template <class X> constexpr bool check() { return true; }
|
|
template <class X, class Enable = void> struct Bar {};
|
|
|
|
namespace param {
|
|
template<class X> void func(Bar<X, typename enable_if<check<X>()>::type>) {}
|
|
// expected-note@-1 2{{candidate function}}
|
|
template<class T> void func(Bar<Foo<T>>) {}
|
|
// expected-note@-1 2{{candidate function}}
|
|
|
|
void g() {
|
|
func(Bar<Foo<int>>()); // expected-error {{call to 'func' is ambiguous}}
|
|
void (*ptr)(Bar<Foo<int>>){func};
|
|
// expected-error@-1 {{address of overloaded function 'func' is ambiguous}}
|
|
}
|
|
} // namespace param
|
|
namespace ret {
|
|
template<class X> Bar<X, typename enable_if<check<X>()>::type> func();
|
|
// expected-note@-1 {{candidate function}}
|
|
template<class T> Bar<Foo<T>> func();
|
|
// expected-note@-1 {{candidate function}}
|
|
|
|
void g() {
|
|
Bar<Foo<int>> (*ptr)(){func};
|
|
// expected-error@-1 {{address of overloaded function 'func' is ambiguous}}
|
|
}
|
|
} // namespace ret
|
|
namespace conv {
|
|
struct A {
|
|
template<class X> operator Bar<X, typename enable_if<check<X>()>::type>();
|
|
// expected-note@-1 {{candidate function}}
|
|
template<class T> operator Bar<Foo<T>>();
|
|
// expected-note@-1 {{candidate function}}
|
|
};
|
|
void g() {
|
|
Bar<Foo<int>> x = A();
|
|
// expected-error@-1 {{conversion from 'A' to 'Bar<Foo<int>>' is ambiguous}}
|
|
}
|
|
} // namespace conv
|
|
} // namespace t1
|
|
|
|
namespace t2 {
|
|
template <bool> struct enable_if;
|
|
template <> struct enable_if<true> {
|
|
typedef int type;
|
|
};
|
|
struct pair {
|
|
template <int = 0> pair(int);
|
|
template <class _U2, enable_if<__is_constructible(int &, _U2)>::type = 0>
|
|
pair(_U2 &&);
|
|
};
|
|
int test_test_i;
|
|
void test() { pair{test_test_i}; }
|
|
} // namespace t2
|
|
|
|
namespace t3 {
|
|
template <class _Tp> void to_address(_Tp);
|
|
template <class _Pointer> auto to_address(_Pointer __p) -> decltype(__p);
|
|
|
|
template <class _CharT> struct basic_string_view {
|
|
basic_string_view(_CharT);
|
|
|
|
template <class _It> requires requires(_It __i) { to_address(__i); }
|
|
basic_string_view(_It);
|
|
};
|
|
void operatorsv() { basic_string_view(0); }
|
|
} // namespace t3
|
|
|
|
namespace func_pointer {
|
|
template <class> struct __promote {
|
|
using type = float;
|
|
};
|
|
template <class> class complex {};
|
|
|
|
namespace ret {
|
|
template <class _Tp> complex<_Tp> pow(const complex<_Tp> &) {};
|
|
template <class _Tp> complex<typename __promote<_Tp>::type> pow(_Tp) = delete;
|
|
complex<float> (*ptr)(const complex<float> &){pow};
|
|
} // namespace ret
|
|
namespace param {
|
|
template <class _Tp> void pow(const complex<_Tp> &, complex<_Tp>) {};
|
|
template <class _Tp> void pow(_Tp, complex<typename __promote<_Tp>::type>) = delete;
|
|
void (*ptr)(const complex<float> &, complex<float>){pow};
|
|
} // namespace param
|
|
} // namespace func_pointer
|
|
|
|
namespace static_vs_nonstatic {
|
|
namespace implicit_obj_param {
|
|
struct A {
|
|
template <class... Args>
|
|
static void f(int a, Args... args) {}
|
|
template <class... Args>
|
|
void f(Args... args) = delete;
|
|
};
|
|
void g(){
|
|
A::f(0);
|
|
}
|
|
} // namespace implicit_obj_param
|
|
namespace explicit_obj_param {
|
|
struct A {
|
|
template <class... Args>
|
|
static void f(int, Args... args) {}
|
|
template <class... Args>
|
|
void f(this A *, Args... args) = delete;
|
|
};
|
|
void g(){
|
|
A::f(0);
|
|
}
|
|
} // namespace explicit_obj_param
|
|
} // namespace static_vs_nonstatic
|
|
|
|
namespace incomplete_on_sugar {
|
|
template <unsigned P, class T> void f(T[P]) = delete;
|
|
template <unsigned P> void f(int[][P]);
|
|
void test() {
|
|
int array[1][8];
|
|
f<8>(array);
|
|
}
|
|
} // namespace incomplete_on_sugar
|