summaryrefslogtreecommitdiff
path: root/libstdc++-v3/testsuite/std/concepts/concepts.lang/concept.commonref/1.cc
blob: 118ee8c4b586569323e6503ffc894739c1a2e778 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
// Copyright (C) 2019-2023 Free Software Foundation, Inc.
//
// This file is part of the GNU ISO C++ Library.  This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.

// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.

// You should have received a copy of the GNU General Public License along
// with this library; see the file COPYING3.  If not see
// <http://www.gnu.org/licenses/>.

// { dg-options "-std=gnu++2a" }
// { dg-do compile { target c++2a } }

#include <concepts>

static_assert( std::common_reference_with<int, int> );
static_assert( std::common_reference_with<int, const int> );
static_assert( std::common_reference_with<int&&, const int&> );
static_assert( std::common_reference_with<int&, const int&&> );
static_assert( ! std::common_reference_with<int, void> );
static_assert( ! std::common_reference_with<int, int*> );
static_assert( ! std::common_reference_with<int, int()> );

struct A { A(int) { } };
static_assert( std::common_reference_with<A, int> );

struct B { };
static_assert( ! std::common_reference_with<A, B> );

struct C { C(A&) { } };
static_assert( std::common_reference_with<A&, C> );
static_assert( std::common_reference_with<A&, C&&> );
static_assert( std::common_reference_with<A&, const C&> );
static_assert( std::common_reference_with<A&, C&> );
static_assert( ! std::common_reference_with<const A&, C> );
static_assert( ! std::common_reference_with<const A&, const C&> );

struct D;
struct E { E(D&) { } };
struct D { D(E&) { } };
static_assert( ! std::common_reference_with<D&, E&> ); // ambiguous conversion

struct F;
struct G { G(const F&) { } };
struct F { F(const G&) { } };
namespace std
{
  template<template<typename> class Qual1, template<typename> class Qual2>
    struct basic_common_reference<F, G, Qual1, Qual2>
    { using type = Qual1<Qual2<F>>; };
  template<template<typename> class Qual1, template<typename> class Qual2>
    struct basic_common_reference<G, F, Qual1, Qual2>
    { using type = Qual1<Qual2<F>>; };
}

static_assert( ! std::common_reference_with<F&, G&> );
static_assert( std::common_reference_with<F, G> );
static_assert( std::common_reference_with<F, const G> );
static_assert( std::common_reference_with<const F, const G> );

struct Base { };
struct Derived : Base { };
static_assert( std::common_reference_with<Derived&, Base&> );
static_assert( std::common_with<Derived*, Base*> );