#ifndef ANDROID_PDX_CHANNEL_HANDLE_H_
#define ANDROID_PDX_CHANNEL_HANDLE_H_

#include <cstdint>
#include <type_traits>

namespace android {
namespace pdx {

enum class ChannelHandleMode {
  Local,
  Borrowed,
  Remote,
};

class ChannelManagerInterface {
 public:
  virtual void CloseHandle(std::int32_t handle) = 0;

 protected:
  // Nobody should be allowed to delete the instance of channel manager
  // through this interface.
  virtual ~ChannelManagerInterface() = default;
};

class ChannelHandleBase {
 public:
  ChannelHandleBase() = default;
  ChannelHandleBase(const int32_t& value) : value_{value} {}

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

  std::int32_t value() const { return value_; }
  bool valid() const { return value_ >= 0; }
  explicit operator bool() const { return valid(); }

  void Close() { value_ = kEmptyHandle; }

 protected:
  // Must not be used by itself. Must be derived from.
  ~ChannelHandleBase() = default;
  enum : std::int32_t { kEmptyHandle = -1 };

  std::int32_t value_{kEmptyHandle};
};

template <ChannelHandleMode Mode>
class ChannelHandle : public ChannelHandleBase {
 public:
  ChannelHandle() = default;
  using ChannelHandleBase::ChannelHandleBase;
  ChannelHandle(ChannelHandle&& other) : ChannelHandleBase{other.value_} {
    other.value_ = kEmptyHandle;
  }
  ~ChannelHandle() = default;

  ChannelHandle Duplicate() const { return ChannelHandle{value_}; }

  ChannelHandle& operator=(ChannelHandle&& other) {
    value_ = other.value_;
    other.value_ = kEmptyHandle;
    return *this;
  }
};

template <>
class ChannelHandle<ChannelHandleMode::Local> : public ChannelHandleBase {
 public:
  ChannelHandle() = default;
  ChannelHandle(ChannelManagerInterface* manager, int32_t value)
      : ChannelHandleBase{value}, manager_{manager} {}

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

  ChannelHandle(ChannelHandle&& other)
      : ChannelHandleBase{other.value_}, manager_{other.manager_} {
    other.manager_ = nullptr;
    other.value_ = kEmptyHandle;
  }

  ChannelHandle& operator=(ChannelHandle&& other) {
    value_ = other.value_;
    manager_ = other.manager_;
    other.value_ = kEmptyHandle;
    other.manager_ = nullptr;
    return *this;
  }

  ~ChannelHandle() {
    if (manager_)
      manager_->CloseHandle(value_);
  }

  ChannelHandle<ChannelHandleMode::Borrowed> Borrow() const {
    return ChannelHandle<ChannelHandleMode::Borrowed>{value_};
  }

  void Close() {
    if (manager_)
      manager_->CloseHandle(value_);
    manager_ = nullptr;
    value_ = kEmptyHandle;
  }

 private:
  ChannelManagerInterface* manager_{nullptr};
};

using LocalChannelHandle = ChannelHandle<ChannelHandleMode::Local>;
using BorrowedChannelHandle = ChannelHandle<ChannelHandleMode::Borrowed>;
using RemoteChannelHandle = ChannelHandle<ChannelHandleMode::Remote>;

// ChannelReference is a 32 bit integer used in IPC serialization to be
// transferred across processes. You can convert this value to a local channel
// handle by calling Transaction.GetChannelHandle().
using ChannelReference = int32_t;

}  // namespace pdx
}  // namespace android

#endif  // ANDROID_PDX_CHANNEL_HANDLE_H_