mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-26 00:26:05 +00:00

As reported in https://reviews.llvm.org/D151953#4472195, the std::move algorithm (and various other functions that relied on it) stopped working after starting to use `__constexpr_memmove` in its implementation. This patch fixes the underlying issue in `__constexpr_memmove` and adds tests for various related algorithms and functions that were not exercising trivial move-only types. Differential Revision: https://reviews.llvm.org/D154613
129 lines
4.4 KiB
C++
129 lines
4.4 KiB
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 MOVEONLY_H
|
|
#define MOVEONLY_H
|
|
|
|
#include "test_macros.h"
|
|
|
|
#include <cstddef>
|
|
#include <functional>
|
|
|
|
class MoveOnly
|
|
{
|
|
int data_;
|
|
public:
|
|
TEST_CONSTEXPR MoveOnly(int data = 1) : data_(data) {}
|
|
|
|
MoveOnly(const MoveOnly&) = delete;
|
|
MoveOnly& operator=(const MoveOnly&) = delete;
|
|
|
|
TEST_CONSTEXPR_CXX14 MoveOnly(MoveOnly&& x) TEST_NOEXCEPT
|
|
: data_(x.data_) {x.data_ = 0;}
|
|
TEST_CONSTEXPR_CXX14 MoveOnly& operator=(MoveOnly&& x)
|
|
{data_ = x.data_; x.data_ = 0; return *this;}
|
|
|
|
TEST_CONSTEXPR int get() const {return data_;}
|
|
|
|
friend TEST_CONSTEXPR bool operator==(const MoveOnly& x, const MoveOnly& y)
|
|
{ return x.data_ == y.data_; }
|
|
friend TEST_CONSTEXPR bool operator!=(const MoveOnly& x, const MoveOnly& y)
|
|
{ return x.data_ != y.data_; }
|
|
friend TEST_CONSTEXPR bool operator< (const MoveOnly& x, const MoveOnly& y)
|
|
{ return x.data_ < y.data_; }
|
|
friend TEST_CONSTEXPR bool operator<=(const MoveOnly& x, const MoveOnly& y)
|
|
{ return x.data_ <= y.data_; }
|
|
friend TEST_CONSTEXPR bool operator> (const MoveOnly& x, const MoveOnly& y)
|
|
{ return x.data_ > y.data_; }
|
|
friend TEST_CONSTEXPR bool operator>=(const MoveOnly& x, const MoveOnly& y)
|
|
{ return x.data_ >= y.data_; }
|
|
|
|
#if TEST_STD_VER > 17
|
|
friend constexpr auto operator<=>(const MoveOnly&, const MoveOnly&) = default;
|
|
#endif // TEST_STD_VER > 17
|
|
|
|
TEST_CONSTEXPR_CXX14 MoveOnly operator+(const MoveOnly& x) const
|
|
{ return MoveOnly(data_ + x.data_); }
|
|
TEST_CONSTEXPR_CXX14 MoveOnly operator*(const MoveOnly& x) const
|
|
{ return MoveOnly(data_ * x.data_); }
|
|
|
|
template<class T>
|
|
friend void operator,(MoveOnly const&, T) = delete;
|
|
|
|
template<class T>
|
|
friend void operator,(T, MoveOnly const&) = delete;
|
|
};
|
|
|
|
template <>
|
|
struct std::hash<MoveOnly>
|
|
{
|
|
typedef MoveOnly argument_type;
|
|
typedef std::size_t result_type;
|
|
TEST_CONSTEXPR std::size_t operator()(const MoveOnly& x) const {return static_cast<size_t>(x.get());}
|
|
};
|
|
|
|
class TrivialMoveOnly {
|
|
int data_;
|
|
|
|
public:
|
|
TEST_CONSTEXPR TrivialMoveOnly(int data = 1) : data_(data) {}
|
|
|
|
TrivialMoveOnly(const TrivialMoveOnly&) = delete;
|
|
TrivialMoveOnly& operator=(const TrivialMoveOnly&) = delete;
|
|
|
|
TrivialMoveOnly(TrivialMoveOnly&&) = default;
|
|
TrivialMoveOnly& operator=(TrivialMoveOnly&&) = default;
|
|
|
|
TEST_CONSTEXPR int get() const { return data_; }
|
|
|
|
friend TEST_CONSTEXPR bool operator==(const TrivialMoveOnly& x, const TrivialMoveOnly& y) {
|
|
return x.data_ == y.data_;
|
|
}
|
|
friend TEST_CONSTEXPR bool operator!=(const TrivialMoveOnly& x, const TrivialMoveOnly& y) {
|
|
return x.data_ != y.data_;
|
|
}
|
|
friend TEST_CONSTEXPR bool operator<(const TrivialMoveOnly& x, const TrivialMoveOnly& y) {
|
|
return x.data_ < y.data_;
|
|
}
|
|
friend TEST_CONSTEXPR bool operator<=(const TrivialMoveOnly& x, const TrivialMoveOnly& y) {
|
|
return x.data_ <= y.data_;
|
|
}
|
|
friend TEST_CONSTEXPR bool operator>(const TrivialMoveOnly& x, const TrivialMoveOnly& y) {
|
|
return x.data_ > y.data_;
|
|
}
|
|
friend TEST_CONSTEXPR bool operator>=(const TrivialMoveOnly& x, const TrivialMoveOnly& y) {
|
|
return x.data_ >= y.data_;
|
|
}
|
|
|
|
#if TEST_STD_VER > 17
|
|
friend constexpr auto operator<=>(const TrivialMoveOnly&, const TrivialMoveOnly&) = default;
|
|
#endif // TEST_STD_VER > 17
|
|
|
|
TEST_CONSTEXPR_CXX14 TrivialMoveOnly operator+(const TrivialMoveOnly& x) const {
|
|
return TrivialMoveOnly(data_ + x.data_);
|
|
}
|
|
TEST_CONSTEXPR_CXX14 TrivialMoveOnly operator*(const TrivialMoveOnly& x) const {
|
|
return TrivialMoveOnly(data_ * x.data_);
|
|
}
|
|
|
|
template<class T>
|
|
friend void operator,(TrivialMoveOnly const&, T) = delete;
|
|
|
|
template<class T>
|
|
friend void operator,(T, TrivialMoveOnly const&) = delete;
|
|
};
|
|
|
|
template <>
|
|
struct std::hash<TrivialMoveOnly> {
|
|
typedef TrivialMoveOnly argument_type;
|
|
typedef std::size_t result_type;
|
|
TEST_CONSTEXPR std::size_t operator()(const TrivialMoveOnly& x) const { return static_cast<size_t>(x.get()); }
|
|
};
|
|
|
|
#endif // MOVEONLY_H
|