// RUN: %clang_cc1 -std=c++2a -verify %s template constexpr static bool is_same_v = false; template constexpr static bool is_same_v = true; namespace templates { template concept AtLeast1 = sizeof(T) >= 1; template int foo(T t) requires (sizeof(T) == 4) { // expected-note {{candidate function}} return 0; } template char foo(T t) requires AtLeast1 { // expected-note {{candidate function}} return 'a'; } template double foo(T t) requires (AtLeast1 && sizeof(T) <= 2) { return 'a'; } static_assert(is_same_v); // expected-error {{call to 'foo' is ambiguous}} static_assert(is_same_v); template void bar() requires (sizeof(T) == 1) { } // expected-note@-1{{similar constraint expressions not considered equivalent}} // expected-note@-2{{candidate function [with T = char]}} template void bar() requires (sizeof(T) == 1 && sizeof(T) >= 0) { } // expected-note@-1{{candidate function [with T = char]}} // expected-note@-2{{similar constraint expression here}} static_assert(is_same_v()), void>); // expected-error@-1{{call to 'bar' is ambiguous}} template constexpr int baz() requires AtLeast1 { // expected-note {{candidate function}} return 1; } template requires AtLeast1 constexpr int baz() { // expected-note {{candidate function [with T = int]}} return 2; } static_assert(baz() == 1); // expected-error {{call to 'baz' is ambiguous}} } namespace non_template { template concept AtLeast2 = sizeof(T) >= 2; template concept AtMost8 = sizeof(T) <= 8; template int foo() requires AtLeast2 && AtMost8 { return 0; } template double foo() requires AtLeast2 { return 0.0; } template double baz() requires AtLeast2 && AtMost8 { // expected-note {{candidate function}} return 0.0; } template int baz() requires AtMost8 && AtLeast2 { // expected-note {{candidate function}} return 0.0; } template void bar() requires (sizeof(char[8]) >= 8) { } // expected-note@-1 {{candidate function}} // expected-note@-2 {{similar constraint expressions not considered equivalent}} template void bar() requires (sizeof(char[8]) >= 8 && sizeof(int) <= 30) { } // expected-note@-1 {{candidate function}} // expected-note@-2 {{similar constraint expression here}} static_assert(is_same_v()), int>); static_assert(is_same_v()), int>); // expected-error {{call to 'baz' is ambiguous}} static_assert(is_same_v()), void>); // expected-error {{call to 'bar' is ambiguous}} template constexpr int goo(int a) requires AtLeast2 && true { // expected-note {{candidate function}} return 1; } template constexpr int goo(const int b) requires AtLeast2 { // expected-note {{candidate function}} return 2; } // [temp.func.order] p5 // Since, in a call context, such type deduction considers only parameters // for which there are explicit call arguments, some parameters are ignored // (namely, function parameter packs, parameters with default arguments, and // ellipsis parameters). template constexpr int doo(int a, ...) requires AtLeast2 && true { return 1; } template constexpr int doo(int b) requires AtLeast2 { return 2; } // By temp.func.order-6.2.2, this is ambiguous because parameter a and b have different types. static_assert(goo(1) == 1); // expected-error {{call to 'goo' is ambiguous}} static_assert(doo(2) == 1); }