While it's not necessary to qualify calls to `declval` it makes error messages very crypric if the declaration isn't reachable anymore
For example:
```
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:53:66: error: no type named 'type' in 'std::common_type<long, long>'
typedef chrono::duration<typename common_type<_Rep1, _Rep2>::type,
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~^~~~
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__type_traits/common_type.h:107:14: note: in instantiation of template class 'std::common_type<std::chrono::duration<long, std::ratio<3600, 1>>, std::chrono::duration<long, std::ratio<3600, 1>>>' requested here
: public common_type<_Tp, _Tp> {};
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:279:58: note: in instantiation of template class 'std::common_type<std::chrono::duration<long, std::ratio<3600, 1>>>' requested here
_LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR typename common_type<duration>::type operator+() const {return typename common_type<duration>::type(*this);}
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:308:54: note: in instantiation of template class 'std::chrono::duration<long, std::ratio<3600, 1>>' requested here
typedef duration< int, ratio_multiply<ratio<24>, hours::period>> days;
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:280:81: error: no type named 'type' in 'std::common_type<std::chrono::duration<long, std::ratio<3600, 1>>>'
_LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR typename common_type<duration>::type operator-() const {return typename common_type<duration>::type(-__rep_);}
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~^~~~
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:308:54: note: in instantiation of template class 'std::chrono::duration<long, std::ratio<3600, 1>>' requested here
typedef duration< int, ratio_multiply<ratio<24>, hours::period>> days;
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:53:66: error: no type named 'type' in 'std::common_type<int, int>'
typedef chrono::duration<typename common_type<_Rep1, _Rep2>::type,
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~^~~~
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__type_traits/common_type.h:107:14: note: in instantiation of template class 'std::common_type<std::chrono::duration<int, std::ratio<86400, 1>>, std::chrono::duration<int, std::ratio<86400, 1>>>' requested here
: public common_type<_Tp, _Tp> {};
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:279:58: note: in instantiation of template class 'std::common_type<std::chrono::duration<int, std::ratio<86400, 1>>>' requested here
_LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR typename common_type<duration>::type operator+() const {return typename common_type<duration>::type(*this);}
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__chrono/duration.h:309:55: note: in instantiation of template class 'std::chrono::duration<int, std::ratio<86400, 1>>' requested here
typedef duration< int, ratio_multiply<ratio<7>, days::period>> weeks;
^
19 similar errors omitted
```
changes with qualification added to:
```
While building module 'std' imported from /home/nikolask/llvm-projects/libcxx/libcxx/test/std/utilities/meta/meta.trans/meta.trans.other/common_type.pass.cpp:13:
In file included from <module-includes>:17:
In file included from /home/nikolask/llvm-projects/libcxx/build/include/c++/v1/math.h:309:
In file included from /home/nikolask/llvm-projects/libcxx/build/include/c++/v1/limits:107:
In file included from /home/nikolask/llvm-projects/libcxx/build/include/c++/v1/type_traits:432:
In file included from /home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__type_traits/common_reference.h:13:
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__type_traits/common_type.h:28:43: error: declaration of 'declval' must be imported from module 'std.utility.__utility.declval' before it is required
using __cond_type = decltype(false ? std::declval<_Tp>() : std::declval<_Up>());
^
/home/nikolask/llvm-projects/libcxx/build/include/c++/v1/__utility/declval.h:30:34: note: declaration here is not visible
decltype(std::__declval<_Tp>(0)) declval() _NOEXCEPT;
^
/home/nikolask/llvm-projects/libcxx/libcxx/test/std/utilities/meta/meta.trans/meta.trans.other/common_type.pass.cpp:13:10: fatal error: could not build module 'std'
#include <functional>
~~~~~~~~^
2 errors generated.
```
Reviewed By: ldionne, Mordante, #libc
Spies: libcxx-commits
Differential Revision: https://reviews.llvm.org/D130854
282 lines
11 KiB
C++
282 lines
11 KiB
C++
// -*- C++ -*-
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef _LIBCPP___ITERATOR_COMMON_ITERATOR_H
|
|
#define _LIBCPP___ITERATOR_COMMON_ITERATOR_H
|
|
|
|
#include <__assert>
|
|
#include <__concepts/assignable.h>
|
|
#include <__concepts/constructible.h>
|
|
#include <__concepts/convertible_to.h>
|
|
#include <__concepts/copyable.h>
|
|
#include <__concepts/derived_from.h>
|
|
#include <__concepts/equality_comparable.h>
|
|
#include <__concepts/same_as.h>
|
|
#include <__config>
|
|
#include <__iterator/concepts.h>
|
|
#include <__iterator/incrementable_traits.h>
|
|
#include <__iterator/iter_move.h>
|
|
#include <__iterator/iter_swap.h>
|
|
#include <__iterator/iterator_traits.h>
|
|
#include <__iterator/readable_traits.h>
|
|
#include <__type_traits/is_pointer.h>
|
|
#include <__utility/declval.h>
|
|
#include <variant>
|
|
|
|
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
|
# pragma GCC system_header
|
|
#endif
|
|
|
|
_LIBCPP_BEGIN_NAMESPACE_STD
|
|
|
|
#if _LIBCPP_STD_VER > 17
|
|
|
|
template<class _Iter>
|
|
concept __can_use_postfix_proxy =
|
|
constructible_from<iter_value_t<_Iter>, iter_reference_t<_Iter>> &&
|
|
move_constructible<iter_value_t<_Iter>>;
|
|
|
|
template<input_or_output_iterator _Iter, sentinel_for<_Iter> _Sent>
|
|
requires (!same_as<_Iter, _Sent> && copyable<_Iter>)
|
|
class common_iterator {
|
|
struct __proxy {
|
|
constexpr const iter_value_t<_Iter>* operator->() const noexcept {
|
|
return _VSTD::addressof(__value_);
|
|
}
|
|
iter_value_t<_Iter> __value_;
|
|
};
|
|
|
|
struct __postfix_proxy {
|
|
constexpr const iter_value_t<_Iter>& operator*() const noexcept {
|
|
return __value_;
|
|
}
|
|
iter_value_t<_Iter> __value_;
|
|
};
|
|
|
|
public:
|
|
variant<_Iter, _Sent> __hold_;
|
|
|
|
common_iterator() requires default_initializable<_Iter> = default;
|
|
|
|
constexpr common_iterator(_Iter __i) : __hold_(in_place_type<_Iter>, _VSTD::move(__i)) {}
|
|
constexpr common_iterator(_Sent __s) : __hold_(in_place_type<_Sent>, _VSTD::move(__s)) {}
|
|
|
|
template<class _I2, class _S2>
|
|
requires convertible_to<const _I2&, _Iter> && convertible_to<const _S2&, _Sent>
|
|
constexpr common_iterator(const common_iterator<_I2, _S2>& __other)
|
|
: __hold_([&]() -> variant<_Iter, _Sent> {
|
|
_LIBCPP_ASSERT(!__other.__hold_.valueless_by_exception(), "Attempted to construct from a valueless common_iterator");
|
|
if (__other.__hold_.index() == 0)
|
|
return variant<_Iter, _Sent>{in_place_index<0>, _VSTD::__unchecked_get<0>(__other.__hold_)};
|
|
return variant<_Iter, _Sent>{in_place_index<1>, _VSTD::__unchecked_get<1>(__other.__hold_)};
|
|
}()) {}
|
|
|
|
template<class _I2, class _S2>
|
|
requires convertible_to<const _I2&, _Iter> && convertible_to<const _S2&, _Sent> &&
|
|
assignable_from<_Iter&, const _I2&> && assignable_from<_Sent&, const _S2&>
|
|
common_iterator& operator=(const common_iterator<_I2, _S2>& __other) {
|
|
_LIBCPP_ASSERT(!__other.__hold_.valueless_by_exception(), "Attempted to assign from a valueless common_iterator");
|
|
|
|
auto __idx = __hold_.index();
|
|
auto __other_idx = __other.__hold_.index();
|
|
|
|
// If they're the same index, just assign.
|
|
if (__idx == 0 && __other_idx == 0)
|
|
_VSTD::__unchecked_get<0>(__hold_) = _VSTD::__unchecked_get<0>(__other.__hold_);
|
|
else if (__idx == 1 && __other_idx == 1)
|
|
_VSTD::__unchecked_get<1>(__hold_) = _VSTD::__unchecked_get<1>(__other.__hold_);
|
|
|
|
// Otherwise replace with the oposite element.
|
|
else if (__other_idx == 1)
|
|
__hold_.template emplace<1>(_VSTD::__unchecked_get<1>(__other.__hold_));
|
|
else if (__other_idx == 0)
|
|
__hold_.template emplace<0>(_VSTD::__unchecked_get<0>(__other.__hold_));
|
|
|
|
return *this;
|
|
}
|
|
|
|
constexpr decltype(auto) operator*()
|
|
{
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator");
|
|
return *_VSTD::__unchecked_get<_Iter>(__hold_);
|
|
}
|
|
|
|
constexpr decltype(auto) operator*() const
|
|
requires __dereferenceable<const _Iter>
|
|
{
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator");
|
|
return *_VSTD::__unchecked_get<_Iter>(__hold_);
|
|
}
|
|
|
|
template<class _I2 = _Iter>
|
|
decltype(auto) operator->() const
|
|
requires indirectly_readable<const _I2> &&
|
|
(requires(const _I2& __i) { __i.operator->(); } ||
|
|
is_reference_v<iter_reference_t<_I2>> ||
|
|
constructible_from<iter_value_t<_I2>, iter_reference_t<_I2>>)
|
|
{
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to dereference a non-dereferenceable common_iterator");
|
|
if constexpr (is_pointer_v<_Iter> || requires(const _Iter& __i) { __i.operator->(); }) {
|
|
return _VSTD::__unchecked_get<_Iter>(__hold_);
|
|
} else if constexpr (is_reference_v<iter_reference_t<_Iter>>) {
|
|
auto&& __tmp = *_VSTD::__unchecked_get<_Iter>(__hold_);
|
|
return _VSTD::addressof(__tmp);
|
|
} else {
|
|
return __proxy{*_VSTD::__unchecked_get<_Iter>(__hold_)};
|
|
}
|
|
}
|
|
|
|
common_iterator& operator++() {
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to increment a non-dereferenceable common_iterator");
|
|
++_VSTD::__unchecked_get<_Iter>(__hold_); return *this;
|
|
}
|
|
|
|
decltype(auto) operator++(int) {
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__hold_), "Attempted to increment a non-dereferenceable common_iterator");
|
|
if constexpr (forward_iterator<_Iter>) {
|
|
auto __tmp = *this;
|
|
++*this;
|
|
return __tmp;
|
|
} else if constexpr (requires (_Iter& __i) { { *__i++ } -> __can_reference; } ||
|
|
!__can_use_postfix_proxy<_Iter>) {
|
|
return _VSTD::__unchecked_get<_Iter>(__hold_)++;
|
|
} else {
|
|
auto __p = __postfix_proxy{**this};
|
|
++*this;
|
|
return __p;
|
|
}
|
|
}
|
|
|
|
template<class _I2, sentinel_for<_Iter> _S2>
|
|
requires sentinel_for<_Sent, _I2>
|
|
_LIBCPP_HIDE_FROM_ABI
|
|
friend constexpr bool operator==(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) {
|
|
_LIBCPP_ASSERT(!__x.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
|
|
_LIBCPP_ASSERT(!__y.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
|
|
|
|
auto __x_index = __x.__hold_.index();
|
|
auto __y_index = __y.__hold_.index();
|
|
|
|
if (__x_index == __y_index)
|
|
return true;
|
|
|
|
if (__x_index == 0)
|
|
return _VSTD::__unchecked_get<_Iter>(__x.__hold_) == _VSTD::__unchecked_get<_S2>(__y.__hold_);
|
|
|
|
return _VSTD::__unchecked_get<_Sent>(__x.__hold_) == _VSTD::__unchecked_get<_I2>(__y.__hold_);
|
|
}
|
|
|
|
template<class _I2, sentinel_for<_Iter> _S2>
|
|
requires sentinel_for<_Sent, _I2> && equality_comparable_with<_Iter, _I2>
|
|
_LIBCPP_HIDE_FROM_ABI
|
|
friend constexpr bool operator==(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) {
|
|
_LIBCPP_ASSERT(!__x.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
|
|
_LIBCPP_ASSERT(!__y.__hold_.valueless_by_exception(), "Attempted to compare a valueless common_iterator");
|
|
|
|
auto __x_index = __x.__hold_.index();
|
|
auto __y_index = __y.__hold_.index();
|
|
|
|
if (__x_index == 1 && __y_index == 1)
|
|
return true;
|
|
|
|
if (__x_index == 0 && __y_index == 0)
|
|
return _VSTD::__unchecked_get<_Iter>(__x.__hold_) == _VSTD::__unchecked_get<_I2>(__y.__hold_);
|
|
|
|
if (__x_index == 0)
|
|
return _VSTD::__unchecked_get<_Iter>(__x.__hold_) == _VSTD::__unchecked_get<_S2>(__y.__hold_);
|
|
|
|
return _VSTD::__unchecked_get<_Sent>(__x.__hold_) == _VSTD::__unchecked_get<_I2>(__y.__hold_);
|
|
}
|
|
|
|
template<sized_sentinel_for<_Iter> _I2, sized_sentinel_for<_Iter> _S2>
|
|
requires sized_sentinel_for<_Sent, _I2>
|
|
_LIBCPP_HIDE_FROM_ABI
|
|
friend constexpr iter_difference_t<_I2> operator-(const common_iterator& __x, const common_iterator<_I2, _S2>& __y) {
|
|
_LIBCPP_ASSERT(!__x.__hold_.valueless_by_exception(), "Attempted to subtract from a valueless common_iterator");
|
|
_LIBCPP_ASSERT(!__y.__hold_.valueless_by_exception(), "Attempted to subtract a valueless common_iterator");
|
|
|
|
auto __x_index = __x.__hold_.index();
|
|
auto __y_index = __y.__hold_.index();
|
|
|
|
if (__x_index == 1 && __y_index == 1)
|
|
return 0;
|
|
|
|
if (__x_index == 0 && __y_index == 0)
|
|
return _VSTD::__unchecked_get<_Iter>(__x.__hold_) - _VSTD::__unchecked_get<_I2>(__y.__hold_);
|
|
|
|
if (__x_index == 0)
|
|
return _VSTD::__unchecked_get<_Iter>(__x.__hold_) - _VSTD::__unchecked_get<_S2>(__y.__hold_);
|
|
|
|
return _VSTD::__unchecked_get<_Sent>(__x.__hold_) - _VSTD::__unchecked_get<_I2>(__y.__hold_);
|
|
}
|
|
|
|
_LIBCPP_HIDE_FROM_ABI friend constexpr iter_rvalue_reference_t<_Iter> iter_move(const common_iterator& __i)
|
|
noexcept(noexcept(ranges::iter_move(std::declval<const _Iter&>())))
|
|
requires input_iterator<_Iter>
|
|
{
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__i.__hold_), "Attempted to iter_move a non-dereferenceable common_iterator");
|
|
return ranges::iter_move( _VSTD::__unchecked_get<_Iter>(__i.__hold_));
|
|
}
|
|
|
|
template<indirectly_swappable<_Iter> _I2, class _S2>
|
|
_LIBCPP_HIDE_FROM_ABI friend constexpr void iter_swap(const common_iterator& __x, const common_iterator<_I2, _S2>& __y)
|
|
noexcept(noexcept(ranges::iter_swap(std::declval<const _Iter&>(), std::declval<const _I2&>())))
|
|
{
|
|
_LIBCPP_ASSERT(std::holds_alternative<_Iter>(__x.__hold_), "Attempted to iter_swap a non-dereferenceable common_iterator");
|
|
_LIBCPP_ASSERT(std::holds_alternative<_I2>(__y.__hold_), "Attempted to iter_swap a non-dereferenceable common_iterator");
|
|
return ranges::iter_swap(_VSTD::__unchecked_get<_Iter>(__x.__hold_), _VSTD::__unchecked_get<_I2>(__y.__hold_));
|
|
}
|
|
};
|
|
|
|
template<class _Iter, class _Sent>
|
|
struct incrementable_traits<common_iterator<_Iter, _Sent>> {
|
|
using difference_type = iter_difference_t<_Iter>;
|
|
};
|
|
|
|
template<class _Iter>
|
|
concept __denotes_forward_iter =
|
|
requires { typename iterator_traits<_Iter>::iterator_category; } &&
|
|
derived_from<typename iterator_traits<_Iter>::iterator_category, forward_iterator_tag>;
|
|
|
|
template<class _Iter, class _Sent>
|
|
concept __common_iter_has_ptr_op = requires(const common_iterator<_Iter, _Sent>& __a) {
|
|
__a.operator->();
|
|
};
|
|
|
|
template<class, class>
|
|
struct __arrow_type_or_void {
|
|
using type = void;
|
|
};
|
|
|
|
template<class _Iter, class _Sent>
|
|
requires __common_iter_has_ptr_op<_Iter, _Sent>
|
|
struct __arrow_type_or_void<_Iter, _Sent> {
|
|
using type = decltype(std::declval<const common_iterator<_Iter, _Sent>&>().operator->());
|
|
};
|
|
|
|
template<input_iterator _Iter, class _Sent>
|
|
struct iterator_traits<common_iterator<_Iter, _Sent>> {
|
|
using iterator_concept = _If<forward_iterator<_Iter>,
|
|
forward_iterator_tag,
|
|
input_iterator_tag>;
|
|
using iterator_category = _If<__denotes_forward_iter<_Iter>,
|
|
forward_iterator_tag,
|
|
input_iterator_tag>;
|
|
using pointer = typename __arrow_type_or_void<_Iter, _Sent>::type;
|
|
using value_type = iter_value_t<_Iter>;
|
|
using difference_type = iter_difference_t<_Iter>;
|
|
using reference = iter_reference_t<_Iter>;
|
|
};
|
|
|
|
#endif // _LIBCPP_STD_VER > 17
|
|
|
|
_LIBCPP_END_NAMESPACE_STD
|
|
|
|
#endif // _LIBCPP___ITERATOR_COMMON_ITERATOR_H
|