| // RUN: %clang_cc1 %s -fsyntax-only -std=c++23 -verify |
| |
| namespace issue1 { |
| template<class T, class U = T> class B {}; |
| template<template<class> class P, class T> void f(P<T>); |
| // expected-note@-1 {{deduced type 'B<[...], (default) int>' of 1st parameter does not match adjusted type 'B<[...], float>' of argument [with P = B, T = int]}} |
| |
| void g() { |
| f(B<int>()); |
| f(B<int,float>()); // expected-error {{no matching function for call}} |
| } |
| } // namespace issue1 |
| |
| namespace issue2 { |
| template<typename> struct match; |
| |
| template<template<typename> class t,typename T> struct match<t<T>>; |
| |
| template<template<typename,typename> class t,typename T0,typename T1> |
| struct match<t<T0,T1>> {}; |
| |
| template<typename,typename = void> struct other {}; |
| template struct match<other<void,void>>; |
| } // namespace issue2 |
| |
| namespace type { |
| template<class T1, class T2 = float> struct A; |
| |
| template<class T3> struct B; |
| template<template<class T4 > class TT1, class T5 > struct B<TT1<T5 >> ; |
| template<template<class T6, class T7> class TT2, class T8, class T9> struct B<TT2<T8, T9>> {}; |
| template struct B<A<int>>; |
| } // namespace type |
| |
| namespace value { |
| template<class T1, int V1 = 1> struct A; |
| |
| template<class T2> struct B; |
| template<template<class T3 > class TT1, class T4 > struct B<TT1<T4 >> ; |
| template<template<class T5, int V2> class TT2, class T6, int V3> struct B<TT2<T6, V3>> {}; |
| template struct B<A<int>>; |
| } // namespace value |
| |
| namespace templ { |
| template <class T1> struct A; |
| |
| template<class T2, template <class T3> class T4 = A> struct B {}; |
| |
| template<class T5> struct C; |
| |
| template<template<class T6> class TT1, class T7> struct C<TT1<T7>>; |
| |
| template<template<class T8, template <class T9> class> class TT2, |
| class T10, template <class T11> class TT3> |
| struct C<TT2<T10, TT3>> {}; |
| |
| template struct C<B<int>>; |
| } // namespace templ |
| |
| namespace class_template { |
| template <class T1, class T2 = float> struct A; |
| |
| template <class T3> struct B; |
| |
| template <template <class T4> class TT1, class T5> struct B<TT1<T5>>; |
| |
| template <class T6, class T7> struct B<A<T6, T7>> {}; |
| |
| template struct B<A<int>>; |
| } // namespace class_template |
| |
| namespace class_template_func { |
| template <class T1, class T2 = float> struct A {}; |
| |
| template <template <class T4> class TT1, class T5> void f(TT1<T5>); |
| template <class T6, class T7> void f(A<T6, T7>) {}; |
| |
| void g() { |
| f(A<int>()); |
| } |
| } // namespace class_template_func |
| |
| namespace type_pack1 { |
| template<class T2> struct A; |
| template<template<class ...T3s> class TT1, class T4> struct A<TT1<T4>> ; |
| template<template<class T5 > class TT2, class T6> struct A<TT2<T6>> {}; |
| |
| template<class T1> struct B; |
| template struct A<B<char>>; |
| } // namespace type_pack1 |
| |
| namespace type_pack2 { |
| template<class T2> struct A; |
| template<template<class ...T3s> class TT1, class ...T4> struct A<TT1<T4...>> ; |
| template<template<class T5 > class TT2, class ...T6> struct A<TT2<T6...>> {}; |
| |
| template<class T1> struct B; |
| template struct A<B<char>>; |
| } // namespace type_pack2 |
| |
| namespace type_pack3 { |
| template<class T1, class T2 = float> struct A; |
| |
| template<class T3> struct B; |
| |
| template<template<class T4 > class TT1, class T5 > struct B<TT1<T5 >>; |
| |
| template<template<class T6, class ...T7s> class TT2, class T8, class ...T9s> struct B<TT2<T8, T9s...>> {}; |
| |
| template struct B<A<int>>; |
| } // namespace type_pack3 |
| |
| namespace gcc_issue { |
| template<class T1, class T2> struct A; |
| |
| template<template<class T1> class TT1, class T2> struct A<TT1<T2>, typename TT1<T2>::type>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template<template<class T3, class T4> class TT2, class T5, class T6> |
| struct A<TT2<T5, T6>, typename TT2<T5, T5>::type>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template <class T7, class T8 = T7> struct B { using type = int; }; |
| |
| template struct A<B<int>, int>; |
| // expected-error@-1 {{ambiguous partial specializations}} |
| } // namespace gcc_issue |
| |
| namespace ttp_defaults { |
| template <template <class T1> class TT1> struct A {}; |
| |
| template <template <class T2> class TT2> void f(A<TT2>); |
| // expected-note@-1 {{explicit instantiation candidate}} |
| |
| // FIXME: The default arguments on the TTP are not available during partial ordering. |
| template <template <class T3, class T4 = float> class TT3> void f(A<TT3>) {}; |
| // expected-note@-1 {{explicit instantiation candidate}} |
| |
| template <class T5, class T6 = int> struct B; |
| |
| template void f<B>(A<B>); |
| // expected-error@-1 {{partial ordering for explicit instantiation of 'f' is ambiguous}} |
| } // namespace ttp_defaults |
| |
| namespace ttp_only { |
| template <template <class... > class TT1> struct A { static constexpr int V = 0; }; |
| template <template <class > class TT2> struct A<TT2> { static constexpr int V = 1; }; |
| template <template <class, class> class TT3> struct A<TT3> { static constexpr int V = 2; }; |
| |
| template <class ... > struct B; |
| template <class > struct C; |
| template <class, class > struct D; |
| template <class, class, class> struct E; |
| |
| static_assert(A<B>::V == 0); |
| static_assert(A<C>::V == 1); |
| static_assert(A<D>::V == 2); |
| static_assert(A<E>::V == 0); |
| } // namespace ttp_only |
| |
| namespace consistency { |
| template<class T> struct nondeduced { using type = T; }; |
| template<class T8, class T9 = float> struct B; |
| |
| namespace t1 { |
| template<class T1, class T2, class T3> struct A; |
| |
| template<template<class, class> class TT1, |
| class T1, class T2, class T3, class T4> |
| struct A<TT1<T1, T2>, TT1<T3, T4>, typename nondeduced<TT1<T1, T2>>::type> {}; |
| |
| template<template<class> class UU1, |
| template<class> class UU2, |
| class U1, class U2> |
| struct A<UU1<U1>, UU2<U2>, typename nondeduced<UU1<U1>>::type>; |
| |
| template struct A<B<int>, B<int>, B<int>>; |
| } // namespace t1 |
| namespace t2 { |
| template<class T1, class T2, class T3> struct A; |
| |
| template<template<class, class> class TT1, |
| class T1, class T2, class T3, class T4> |
| struct A<TT1<T1, T2>, TT1<T3, T4>, typename nondeduced<TT1<T1, T4>>::type> {}; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template<template<class> class UU1, |
| template<class> class UU2, |
| class U1, class U2> |
| struct A<UU1<U1>, UU2<U2>, typename nondeduced<UU1<U1>>::type>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template struct A<B<int>, B<int>, B<int>>; |
| // expected-error@-1 {{ambiguous partial specializations}} |
| } // namespace t2 |
| namespace t3 { |
| template<class T1, class T2, class T3> struct A; |
| |
| template<template<class, class> class TT1, |
| class T1, class T2, class T3, class T4> |
| struct A<TT1<T1, T2>, TT1<T3, T4>, typename nondeduced<TT1<T1, T2>>::type> {}; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template<template<class> class UU1, |
| class U1, class U2> |
| struct A<UU1<U1>, UU1<U2>, typename nondeduced<UU1<U1>>::type>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template struct A<B<int>, B<int>, B<int>>; |
| // expected-error@-1 {{ambiguous partial specializations}} |
| } // namespace t3 |
| namespace t4 { |
| template<class T1, class T2, class T3> struct A; |
| |
| template<template<class, class> class TT1, |
| class T1, class T2, class T3, class T4> |
| struct A<TT1<T1, T2>, TT1<T3, T4>, typename nondeduced<TT1<T1, T4>>::type> {}; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template<template<class> class UU1, |
| class U1, class U2> |
| struct A<UU1<U1>, UU1<U2>, typename nondeduced<UU1<U1>>::type>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template struct A<B<int>, B<int>, B<int>>; |
| // expected-error@-1 {{ambiguous partial specializations}} |
| } // namespace t4 |
| namespace t5 { |
| template<class T1, class T2> struct A; |
| |
| template<template<class, class> class TT1, |
| class T1, class T2, class T3, class T4> |
| struct A<TT1<T1, T2>, TT1<T3, T4>> {}; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template<template<class> class UU1, |
| class U1, class U2> |
| struct A<UU1<U1>, UU1<U2>>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template struct A<B<int>, B<int>>; |
| // expected-error@-1 {{ambiguous partial specializations}} |
| } // namespace t5 |
| namespace t6 { |
| template<class T1, class T2> struct A; |
| |
| template<template<class, class> class TT1, |
| class T1, class T2, class T3> |
| struct A<TT1<T1, T2>, TT1<T1, T3>> {}; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template<template<class> class UU1, |
| class U1, class U2> |
| struct A<UU1<U1>, UU1<U2>>; |
| // expected-note@-1 {{partial specialization matches}} |
| |
| template struct A<B<int>, B<int>>; |
| // expected-error@-1 {{ambiguous partial specializations}} |
| } // namespace t6 |
| } // namespace consistency |
| |
| namespace classes { |
| namespace canon { |
| template<class T, class U> struct A {}; |
| |
| template<template<class> class TT> auto f(TT<int> a) { return a; } |
| // expected-note@-1 2{{substitution failure: too few template arguments}} |
| |
| A<int, float> v1; |
| A<int, double> v2; |
| |
| using X = decltype(f(v1)); |
| // expected-error@-1 {{no matching function for call}} |
| |
| using X = decltype(f(v2)); |
| // expected-error@-1 {{no matching function for call}} |
| } // namespace canon |
| namespace expr { |
| template <class T1, int E1> struct A { |
| static constexpr auto val = E1; |
| }; |
| template <template <class T3> class TT> void f(TT<int> v) { |
| // expected-note@-1 {{substitution failure: too few template arguments}} |
| static_assert(v.val == 3); |
| }; |
| void test() { |
| f(A<int, 3>()); |
| // expected-error@-1 {{no matching function for call}} |
| } |
| } // namespace expr |
| namespace packs { |
| template <class T1, class ...T2s> struct A { |
| static constexpr auto val = sizeof...(T2s); |
| }; |
| |
| template <template <class T3> class TT> void f(TT<int> v) { |
| // expected-note@-1 {{deduced type 'A<[...], (no argument), (no argument), (no argument)>' of 1st parameter does not match adjusted type 'A<[...], void, void, void>' of argument [with TT = A]}} |
| static_assert(v.val == 3); |
| }; |
| void test() { |
| f(A<int, void, void, void>()); |
| // expected-error@-1 {{no matching function for call}} |
| } |
| } // namespace packs |
| namespace nested { |
| template <class T1, int V1, int V2> struct A { |
| using type = T1; |
| static constexpr int v1 = V1, v2 = V2; |
| }; |
| |
| template <template <class T1> class TT1> auto f(TT1<int>) { |
| return TT1<float>(); |
| } |
| |
| template <template <class T2, int V3> class TT2> auto g(TT2<double, 1>) { |
| // expected-note@-1 {{too few template arguments for class template 'A'}} |
| return f(TT2<int, 2>()); |
| } |
| |
| using B = decltype(g(A<double, 1, 3>())); |
| // expected-error@-1 {{no matching function for call}} |
| |
| using X = B::type; // expected-error {{undeclared identifier 'B'}} |
| using X = float; |
| static_assert(B::v1 == 2); // expected-error {{undeclared identifier 'B'}} |
| static_assert(B::v2 == 3); // expected-error {{undeclared identifier 'B'}} |
| } |
| namespace defaulted { |
| template <class T1, class T2 = T1*> struct A { |
| using type = T2; |
| }; |
| |
| template <template <class> class TT> TT<float> f(TT<int>); |
| // expected-note@-1 {{deduced type 'A<[...], (default) int *>' of 1st parameter does not match adjusted type 'A<[...], double *>' of argument [with TT = A]}} |
| |
| using X = int*; // expected-note {{previous definition is here}} |
| using X = decltype(f(A<int>()))::type; |
| // expected-error@-1 {{different types ('decltype(f(A<int>()))::type' (aka 'float *') vs 'int *')}} |
| |
| using Y = double*; |
| using Y = decltype(f(A<int, double*>()))::type; |
| // expected-error@-1 {{no matching function for call}} |
| } // namespace defaulted |
| } // namespace classes |
| |
| namespace packs { |
| namespace t1 { |
| template<template<int, int...> class> struct A {}; |
| // expected-error@-1 {{non-type parameter of template template parameter cannot be narrowed from type 'int' to 'char'}} |
| // expected-note@-2 {{previous template template parameter is here}} |
| |
| template<char> struct B; |
| template struct A<B>; |
| // expected-note@-1 {{has different template parameters}} |
| } // namespace t1 |
| namespace t2 { |
| template<template<char, int...> class> struct A {}; |
| template<int> struct B; |
| template struct A<B>; |
| } // namespace t2 |
| namespace t3 { |
| template<template<int...> class> struct A {}; |
| // expected-error@-1 {{non-type parameter of template template parameter cannot be narrowed from type 'int' to 'char'}} |
| // expected-note@-2 {{previous template template parameter is here}} |
| |
| template<char> struct B; |
| template struct A<B>; |
| // expected-note@-1 {{has different template parameters}} |
| } // namespace t3 |
| namespace t4 { |
| template<template<char...> class> struct A {}; |
| template<int> struct B; |
| template struct A<B>; |
| } // namespace t4 |
| } // namespace packs |
| |
| namespace fun_tmpl_call { |
| namespace match_func { |
| template <template <class> class TT> void f(TT<int>) {}; |
| template <class...> struct A {}; |
| void test() { f(A<int>()); } |
| } // namespace match_func |
| namespace order_func_nonpack { |
| template <template <class> class TT> void f(TT<int>) {} |
| template <template <class...> class TT> void f(TT<int>) = delete; |
| |
| template <class> struct A {}; |
| void test() { f(A<int>()); } |
| } // namespace order_func_nonpack |
| namespace order_func_pack { |
| template <template <class> class TT> void f(TT<int>) = delete; |
| template <template <class...> class TT> void f(TT<int>) {} |
| template <class...> struct A {}; |
| void test() { f(A<int>()); } |
| } // namespace order_func_pack |
| namespace match_enum { |
| enum A {}; |
| template<template<A> class TT1> void f(TT1<{}>) {} |
| template<int> struct B {}; |
| template void f<B>(B<{}>); |
| } // namespace match_enum |
| namespace match_method { |
| struct A { |
| template <template <class> class TT> void f(TT<int>) {}; |
| }; |
| template <class...> struct B {}; |
| void test() { A().f(B<int>()); } |
| } // namespace match_method |
| namespace order_method_nonpack { |
| struct A { |
| template <template <class> class TT> void f(TT<int>) {} |
| template <template <class...> class TT> void f(TT<int>) = delete; |
| }; |
| template <class> struct B {}; |
| void test() { A().f(B<int>()); } |
| } // namespace order_method_nonpack |
| namespace order_method_pack { |
| struct A { |
| template <template <class> class TT> void f(TT<int>) = delete; |
| template <template <class...> class TT> void f(TT<int>) {} |
| }; |
| template <class...> struct B {}; |
| void test() { A().f(B<int>()); } |
| } // namespace order_method_pack |
| namespace match_conv { |
| struct A { |
| template <template <class> class TT> operator TT<int>() { return {}; } |
| }; |
| template <class...> struct B {}; |
| void test() { B<int> b = A(); } |
| } // namespace match_conv |
| namespace order_conv_nonpack { |
| struct A { |
| template <template <class> class TT> operator TT<int>() { return {}; }; |
| template <template <class...> class TT> operator TT<int>() = delete; |
| }; |
| template <class> struct B {}; |
| void test() { B<int> b = A(); } |
| } // namespace order_conv_nonpack |
| namespace order_conv_pack { |
| struct A { |
| template <template <class> class TT> operator TT<int>() = delete; |
| template <template <class...> class TT> operator TT<int>() { return {}; } |
| }; |
| template <class...> struct B {}; |
| void test() { B<int> b = A(); } |
| } // namespace order_conv_pack |
| namespace regression1 { |
| template <template <class, class...> class TT, class T1, class... T2s> |
| void f(TT<T1, T2s...>) {} |
| template <class> struct A {}; |
| void test() { f(A<int>()); } |
| } // namespace regression1 |
| } // namespace fun_tmpl_packs |
| |
| namespace partial { |
| namespace t1 { |
| template<template<class... T1s> class TT1> struct A {}; |
| |
| template<template<class T2> class TT2> struct A<TT2>; |
| |
| template<class... T3s> struct B; |
| template struct A<B>; |
| } // namespace t1 |
| namespace t2 { |
| template<template<class... T1s> class TT1> struct A; |
| |
| template<template<class T2> class TT2> struct A<TT2> {}; |
| |
| template<class T3> struct B; |
| template struct A<B>; |
| } // namespace t1 |
| |
| } // namespace partial |
| |
| namespace regression1 { |
| template <typename T, typename Y> struct map {}; |
| template <typename T> class foo {}; |
| |
| template <template <typename...> class MapType, typename Value> |
| Value bar(MapType<int, Value> map); |
| |
| template <template <typename...> class MapType, typename Value> |
| Value bar(MapType<int, foo<Value>> map); |
| |
| void aux() { |
| map<int, foo<int>> input; |
| bar(input); |
| } |
| } // namespace regression1 |
| |
| namespace constraints { |
| template <class T> concept C1 = true; |
| // expected-note@-1 {{similar constraint expression here}} |
| // expected-note@-2 2{{similar constraint expressions not considered equivalent}} |
| |
| template <class T> concept C2 = C1<T> && true; |
| // expected-note@-1 2{{similar constraint expression here}} |
| |
| template <class T> concept D1 = true; |
| // expected-note@-1 {{similar constraint expressions not considered equivalent}} |
| |
| namespace t1 { |
| template<template<C1, class... T1s> class TT1> // expected-note {{TT1' declared here}} |
| struct A {}; |
| template<D1, class T2> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t1 |
| namespace t2 { |
| template<template<C2, class... T1s> class TT1> struct A {}; |
| template<C1, class T2> struct B {}; |
| template struct A<B>; |
| } // namespace t2 |
| namespace t3 { |
| template<template<C1, class... T1s> class TT1> // expected-note {{'TT1' declared here}} |
| struct A {}; |
| template<C2, class T2> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t2 |
| namespace t4 { |
| // FIXME: This should be accepted. |
| template<template<C1... T1s> class TT1> // expected-note {{'TT1' declared here}} |
| struct A {}; |
| template<C1 T2> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t4 |
| namespace t5 { |
| // FIXME: This should be accepted |
| template<template<C2... T1s> class TT1> // expected-note {{'TT1' declared here}} |
| struct A {}; |
| template<C1 T2> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t5 |
| namespace t6 { |
| template<template<C1... T1s> class TT1> // expected-note {{'TT1' declared here}} |
| struct A {}; |
| template<C2 T2> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t6 |
| namespace t7 { |
| template<template<class... T1s> class TT1> |
| struct A {}; |
| template<C1 T2> struct B {}; |
| template struct A<B>; |
| } // namespace t7 |
| namespace t8 { |
| template<template<C1... T1s> class TT1> |
| struct A {}; |
| template<class T2> struct B {}; |
| template struct A<B>; |
| } // namespace t8 |
| namespace t9 { |
| template<template<C1... T1s> class TT1> // expected-note {{'TT1' declared here}} |
| struct A {}; |
| template<D1 T2> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t9 |
| namespace t10 { |
| template<template<class...> requires C1<int> class TT1> // expected-note {{'TT1' declared here}} |
| struct A {}; |
| |
| template<class> requires C2<int> struct B {}; // expected-note {{'B' declared here}} |
| template struct A<B>; |
| // expected-error@-1 {{'B' is more constrained than template template parameter 'TT1'}} |
| } // namespace t10 |
| namespace t11 { |
| template<template<class...> requires C2<int> class TT1> struct A {}; |
| template<class> requires C1<int> struct B {}; |
| template struct A<B>; |
| } // namespace t11 |
| } // namespace constraints |
| |
| namespace regression2 { |
| template <class> struct D {}; |
| |
| template <class ET, template <class> class VT> |
| struct D<VT<ET>>; |
| |
| template <typename, int> struct Matrix; |
| template struct D<Matrix<double, 3>>; |
| } // namespace regression2 |
| namespace regression3 { |
| struct None {}; |
| template<class T> struct Node { using type = T; }; |
| |
| template <template<class> class TT, class T> |
| struct A { |
| static_assert(!__is_same(T, None)); |
| using type2 = typename A<TT, typename T::type>::type2; |
| }; |
| |
| template <template<class> class TT> struct A<TT, None> { |
| using type2 = void; |
| }; |
| |
| template <class...> class B {}; |
| template struct A<B, Node<None>>; |
| } // namespace regression3 |
| namespace GH130362 { |
| template <template <template <class... T1> class TT1> class TT2> struct A {}; |
| template <template <class U1> class UU1> struct B {}; |
| template struct A<B>; |
| } // namespace GH130362 |
| |
| namespace nttp_auto { |
| namespace t1 { |
| template <template <auto... Va> class TT> struct A {}; |
| template <int Vi, short Vs> struct B; |
| template struct A<B>; |
| } // namespace t1 |
| namespace t2 { |
| template<template<auto... Va1, auto Va2> class> struct A {}; |
| // expected-error@-1 {{template parameter pack must be the last template parameter}} |
| template<int... Vi> struct B; |
| template struct A<B>; |
| } // namespace t2 |
| namespace t3 { |
| template<template<auto... Va1, auto... Va2> class> struct A {}; |
| // expected-error@-1 {{template parameter pack must be the last template parameter}} |
| template<int... Vi> struct B; |
| template struct A<B>; |
| } // namespace t3 |
| } // namespace nttp_auto |
| |
| namespace nttp_partial_order { |
| namespace t1 { |
| template<template<short> class TT1> void f(TT1<0>); |
| template<template<int> class TT2> void f(TT2<0>) {} |
| template<int> struct B {}; |
| template void f<B>(B<0>); |
| } // namespace t1 |
| namespace t2 { |
| struct A {} a; |
| template<template<A&> class TT1> void f(TT1<a>); |
| template<template<const A&> class TT2> void f(TT2<a>) {} |
| template<const A&> struct B {}; |
| template void f<B>(B<a>); |
| } // namespace t2 |
| namespace t3 { |
| enum A {}; |
| template<template<A> class TT1> void f(TT1<{}>); |
| template<template<int> class TT2> void f(TT2<{}>) {} |
| template<int> struct B {}; |
| template void f<B>(B<{}>); |
| } // namespace t3 |
| namespace t4 { |
| struct A {} a; |
| template<template<A*> class TT1> void f(TT1<&a>); |
| template<template<const A*> class TT2> void f(TT2<&a>) {} |
| template<const A*> struct B {}; |
| template void f<B>(B<&a>); |
| } // namespace t4 |
| namespace t5 { |
| struct A { int m; }; |
| template<template<int A::*> class TT1> void f(TT1<&A::m>); |
| template<template<const int A::*> class TT2> void f(TT2<&A::m>) {} |
| template<const int A::*> struct B {}; |
| template void f<B>(B<&A::m>); |
| } // namespace t5 |
| namespace t6 { |
| struct A {}; |
| using nullptr_t = decltype(nullptr); |
| template<template<nullptr_t> class TT2> void f(TT2<nullptr>); |
| template<template<A*> class TT1> void f(TT1<nullptr>) {} |
| template<A*> struct B {}; |
| template void f<B>(B<nullptr>); |
| } // namespace t6 |
| } // namespace nttp_partial_order |