template <typename _Tp, typename _Up>
constexpr _Tp&&
get(pair<_Tp, _Up>&& __p) noexcept
- { return std::move(__p.first); }
+ { return std::forward<_Tp>(__p.first); }
template <typename _Tp, typename _Up>
constexpr const _Tp&&
get(const pair<_Tp, _Up>&& __p) noexcept
- { return std::move(__p.first); }
+ { return std::forward<const _Tp>(__p.first); }
template <typename _Tp, typename _Up>
constexpr _Tp&
template <typename _Tp, typename _Up>
constexpr _Tp&&
get(pair<_Up, _Tp>&& __p) noexcept
- { return std::move(__p.second); }
+ { return std::forward<_Tp>(__p.second); }
template <typename _Tp, typename _Up>
constexpr const _Tp&&
get(const pair<_Up, _Tp>&& __p) noexcept
- { return std::move(__p.second); }
+ { return std::forward<const _Tp>(__p.second); }
#if __cplusplus > 202002L
template<typename _T1, typename _T2, typename _U1, typename _U2,
const int&& cpsecond __attribute__((unused)) =
std::get<int>(std::move(cp));
}
+
+// PR libstdc++/121745 return of get(pair<_Up, _Tp>&& __p) may be ill-formed
+void
+test_pr121745(std::pair<float&, int&> p)
+{
+ float& pfirst = std::get<float&>(std::move(p));
+ int& psecond = std::get<int&>(std::move(p));
+
+ const auto& p2 = p;
+ float& p2first = std::get<float&>(std::move(p2));
+ int& p2second = std::get<int&>(std::move(p2));
+}
+
+template<typename T, typename Pair>
+using get_t = decltype(std::get<T>(std::declval<Pair>()));
+
+// Check that get<T>(Pair) returns Ret
+template<typename T, typename Pair, typename Ret>
+constexpr bool verify = std::is_same<get_t<T, Pair>, Ret>::value;
+
+template<typename T1, typename T2>
+void
+check()
+{
+ // Overloads for accessing first member
+ static_assert( verify<T1, std::pair<T1, T2>&, T1&>,
+ "T1& get(pair<T1, T2>&)" );
+ static_assert( verify<T1, const std::pair<T1, T2>&, const T1&>,
+ "const T1& get(const pair<T1, T2>&)" );
+ static_assert( verify<T1, std::pair<T1, T2>&&, T1&&>,
+ "T1&& get(pair<T1, T2>&&)" );
+ static_assert( verify<T1, const std::pair<T1, T2>&&, const T1&&>,
+ "const T1&& get(const pair<T1, T2>&&)" );
+
+ // Overloads for accessing second member
+ static_assert( verify<T2, std::pair<T1, T2>&, T2&>,
+ "T2& get(pair<T1, T2>&)" );
+ static_assert( verify<T2, const std::pair<T1, T2>&, const T2&>,
+ "const T2& get(const pair<T1, T2>&)" );
+ static_assert( verify<T2, std::pair<T1, T2>&&, T2&&>,
+ "T2&& get(pair<T1, T2>&&)" );
+ static_assert( verify<T2, const std::pair<T1, T2>&&, const T2&&>,
+ "const T2&& get(const pair<T1, T2>&&)" );
+}
+
+void
+test_all()
+{
+ check<float, int>();
+ check<float&, int&>();
+ check<float&&, int&&>();
+}