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

base / test / move_only_int.h [blame]

// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef BASE_TEST_MOVE_ONLY_INT_H_
#define BASE_TEST_MOVE_ONLY_INT_H_

#include <utility>

#include "base/functional/callback.h"
#include "third_party/abseil-cpp/absl/cleanup/cleanup.h"

namespace base {

// A move-only class that holds an integer. This is designed for testing
// containers. See also CopyOnlyInt.
class MoveOnlyInt {
 public:
  explicit MoveOnlyInt(int data = 1) : data_(data) {}
  MoveOnlyInt(MoveOnlyInt&& other) : data_(other.data_) { other.data_ = 0; }

  MoveOnlyInt(const MoveOnlyInt&) = delete;
  MoveOnlyInt& operator=(const MoveOnlyInt&) = delete;

  ~MoveOnlyInt();

  MoveOnlyInt& operator=(MoveOnlyInt&& other) {
    data_ = other.data_;
    other.data_ = 0;
    return *this;
  }

  friend bool operator==(const MoveOnlyInt& lhs,
                         const MoveOnlyInt& rhs) = default;
  friend auto operator<=>(const MoveOnlyInt& lhs,
                          const MoveOnlyInt& rhs) = default;

  friend bool operator==(const MoveOnlyInt& lhs, int rhs) {
    return lhs.data_ == rhs;
  }
  friend bool operator==(int lhs, const MoveOnlyInt& rhs) {
    return lhs == rhs.data_;
  }
  friend auto operator<=>(const MoveOnlyInt& lhs, int rhs) {
    return lhs.data_ <=> rhs;
  }
  friend auto operator<=>(int lhs, const MoveOnlyInt& rhs) {
    return lhs <=> rhs.data_;
  }

  int data() const { return data_; }

  // Called with the value of `data()` when an instance of `MoveOnlyInt` is
  // destroyed. Returns an `absl::Cleanup` scoper that automatically
  // unregisters the callback when the scoper is destroyed.
  static auto SetScopedDestructionCallback(
      RepeatingCallback<void(int)> callback) {
    GetDestructionCallbackStorage() = std::move(callback);
    return absl::Cleanup([] { GetDestructionCallbackStorage().Reset(); });
  }

 private:
  static RepeatingCallback<void(int)>& GetDestructionCallbackStorage();

  volatile int data_;
};

}  // namespace base

#endif  // BASE_TEST_MOVE_ONLY_INT_H_