// RUN: rm -rf %t
// RUN: mkdir %t
// RUN: split-file %s %t
//
// RUN: %clang_cc1 -std=c++20 %t/A.cppm -emit-module-interface -o %t/A.pcm
// RUN: %clang_cc1 -std=c++20 -fprebuilt-module-path=%t -I%t -DDIFFERENT %t/B.cppm -verify
// RUN: %clang_cc1 -std=c++20 -fprebuilt-module-path=%t -I%t %t/B.cppm -verify
//
// Testing the behavior of `-fskip-odr-check-in-gmf`
// RUN: %clang_cc1 -std=c++20 -fskip-odr-check-in-gmf %t/A.cppm -emit-module-interface -o %t/A.pcm
// RUN: %clang_cc1 -std=c++20 -fskip-odr-check-in-gmf -fprebuilt-module-path=%t -I%t \
// RUN: -DDIFFERENT -DSKIP_ODR_CHECK_IN_GMF %t/B.cppm -verify
//--- foo.h
#ifndef FOO_H
#define FOO_H
template <class T>
concept Range = requires(T &t) { t.begin(); };
template<class _Tp>
concept __integer_like = true;
template <class _Tp>
concept __member_size = requires(_Tp &&t) { t.size(); };
template <class First, class Second>
concept C = requires(First x, Second y) { x + y; };
struct A {
public:
template <Range T>
using range_type = T;
};
struct __fn {
template <__member_size _Tp>
constexpr __integer_like auto operator()(_Tp&& __t) const {
return __t.size();
}
template <__integer_like _Tp, C<_Tp> Sentinel>
constexpr _Tp operator()(_Tp &&__t, Sentinel &&last) const {
return __t;
}
template <template <class> class H, class S, C<H<S>> Sentinel>
constexpr H<S> operator()(H<S> &&__s, Sentinel &&last) const {
return __s;
}
// Tests that we could find different concept definition indeed.
#ifndef DIFFERENT
template <__integer_like _Tp, __integer_like _Up, C<_Tp> Sentinel>
constexpr _Tp operator()(_Tp &&__t, _Up _u, Sentinel &&last) const {
return __t;
}
#else
template <__integer_like _Tp, __integer_like _Up, C<_Up> Sentinel>
constexpr _Tp operator()(_Tp &&__t, _Up _u, Sentinel &&last) const {
return __t;
}
#endif
};
#endif
//--- A.cppm
module;
#include "foo.h"
export module A;
//--- B.cppm
module;
#include "foo.h"
export module B;
import A;
#ifdef SKIP_ODR_CHECK_IN_GMF
// [email protected]:* {{call to object of type '__fn' is ambiguous}}
// expected-note@* 1+{{candidate function}}
#elif defined(DIFFERENT)
// [email protected]:41 {{'__fn::operator()' from module 'A.<global>' is not present in definition of '__fn' provided earlier}}
// expected-note@* 1+{{declaration of 'operator()' does not match}}
#else
// expected-no-diagnostics
#endif
template <class T>
struct U {
auto operator+(U) { return 0; }
};
void foo() {
A a;
struct S {
int size() { return 0; }
auto operator+(S s) { return 0; }
};
__fn{}(S());
__fn{}(S(), S());
__fn{}(S(), S(), S());
__fn{}(U<int>(), U<int>());
}