Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / Modules / concept.cppm
blob0e85a46411a544872f9485bc0ef03eee68bf979c
1 // RUN: rm -rf %t
2 // RUN: mkdir %t
3 // RUN: split-file %s %t
4 //
5 // RUN: %clang_cc1 -std=c++20 %t/A.cppm -emit-module-interface -o %t/A.pcm
6 // RUN: %clang_cc1 -std=c++20 -fprebuilt-module-path=%t -I%t -DDIFFERENT %t/B.cppm -verify
7 // RUN: %clang_cc1 -std=c++20 -fprebuilt-module-path=%t -I%t %t/B.cppm -verify
9 //--- foo.h
10 #ifndef FOO_H
11 #define FOO_H
13 template <class T>
14 concept Range = requires(T &t) { t.begin(); };
16 template<class _Tp>
17 concept __integer_like = true;
19 template <class _Tp>
20 concept __member_size = requires(_Tp &&t) { t.size(); };
22 template <class First, class Second>
23 concept C = requires(First x, Second y) { x + y; };
25 struct A {
26 public:
27   template <Range T>
28   using range_type = T;
31 struct __fn {
32   template <__member_size _Tp>
33   constexpr __integer_like auto operator()(_Tp&& __t) const {
34     return __t.size();
35   }
37   template <__integer_like _Tp, C<_Tp> Sentinel>
38   constexpr _Tp operator()(_Tp &&__t, Sentinel &&last) const {
39     return __t;
40   }
42   template <template <class> class H, class S, C<H<S>> Sentinel>
43   constexpr H<S> operator()(H<S> &&__s, Sentinel &&last) const {
44     return __s;
45   }
47 // Tests that we could find different concept definition indeed.
48 #ifndef DIFFERENT
49   template <__integer_like _Tp, __integer_like _Up, C<_Tp> Sentinel>
50   constexpr _Tp operator()(_Tp &&__t, _Up _u, Sentinel &&last) const {
51     return __t;
52   }
53 #else
54   template <__integer_like _Tp, __integer_like _Up, C<_Up> Sentinel>
55   constexpr _Tp operator()(_Tp &&__t, _Up _u, Sentinel &&last) const {
56     return __t;
57   }
58 #endif
60 #endif
62 //--- A.cppm
63 module;
64 #include "foo.h"
65 export module A;
67 //--- B.cppm
68 module;
69 #include "foo.h"
70 export module B;
71 import A;
73 #ifdef DIFFERENT
74 // expected-error@foo.h:41 {{'__fn::operator()' from module 'A.<global>' is not present in definition of '__fn' provided earlier}}
75 // expected-note@* 1+{{declaration of 'operator()' does not match}}
76 #else
77 // expected-no-diagnostics
78 #endif
80 template <class T>
81 struct U {
82   auto operator+(U) { return 0; }
85 void foo() {
86     A a;
87     struct S {
88         int size() { return 0; }
89         auto operator+(S s) { return 0; }
90     };
91     __fn{}(S());
92     __fn{}(S(), S());
93     __fn{}(S(), S(), S());
95     __fn{}(U<int>(), U<int>());