#ifndef ANDROID_PDX_FILE_HANDLE_H_
#define ANDROID_PDX_FILE_HANDLE_H_

#include <fcntl.h>
#include <unistd.h>

#include <string>

namespace android {
namespace pdx {

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

// Manages ownership, sharing, and lifetime of file descriptors.
template <FileHandleMode Mode>
class FileHandle {
 public:
  static constexpr int kEmptyFileHandle = -1;

  // Constructs an empty FileHandle object.
  FileHandle() : fd_(kEmptyFileHandle) {}

  // Constructs a FileHandle from an integer file descriptor and takes
  // ownership.
  explicit FileHandle(int fd) : fd_(fd) {}

  // Constructs a FileHandle by opening |path|. The arguments follow the
  // semantics of open().
  FileHandle(const std::string& path, int flags, mode_t mode = 0) {
    fd_ = open(path.c_str(), flags, mode);
  }

  // Constructs a FileHandle by opening |path| relative to |dir_fd|, following
  // the semantics of openat().
  FileHandle(const int directory_fd, const std::string& path, int flags,
             mode_t mode = 0) {
    fd_ = openat(directory_fd, path.c_str(), flags, mode);
  }

  // Move constructor that assumes ownership of the file descriptor, leaving the
  // other FileHandle object empty.
  FileHandle(FileHandle&& other) {
    fd_ = other.fd_;
    other.fd_ = kEmptyFileHandle;
  }

  // Returns a FileHandle as a duplicate handle of |fd|. Borrowed handles and
  // handles in remote handle space are not duplicated.
  static FileHandle AsDuplicate(const int fd) {
    if (Mode == FileHandleMode::Local)
      return FileHandle(dup(fd));
    else
      return FileHandle(fd);
  }

  // Destructor closes the file descriptor when non-empty.
  ~FileHandle() { Close(); }

  // Move assignment operator that assumes ownership of the underlying file
  // descriptor, leaving the other FileHandle object empty.
  FileHandle& operator=(FileHandle&& other) {
    if (this != &other) {
      Reset(other.fd_);
      other.fd_ = kEmptyFileHandle;
    }
    return *this;
  }

  // Resets the underlying file handle to |fd|.
  void Reset(int fd) {
    Close();
    fd_ = fd;
  }

  // Closes the underlying file descriptor when non-empty.
  void Close() {
    if (IsValid() && Mode == FileHandleMode::Local)
      close(fd_);
    fd_ = kEmptyFileHandle;
  }

  // Return the internal fd, passing ownership to the caller.
  int Release() {
    int release_fd = fd_;
    fd_ = kEmptyFileHandle;
    return release_fd;
  }

  // Duplicates the underlying file descriptor and returns a FileHandle that
  // owns the new file descriptor. File descriptors are not duplicated when Mode
  // is Remote or Borrowed.
  FileHandle Duplicate() const {
    return FileHandle(Mode == FileHandleMode::Local ? dup(fd_) : fd_);
  }

  FileHandle<FileHandleMode::Borrowed> Borrow() const {
    return FileHandle<FileHandleMode::Borrowed>(Get());
  }

  // Gets the underlying file descriptor value.
  int Get() const { return fd_; }
  bool IsValid() const { return fd_ >= 0; }
  explicit operator bool() const { return IsValid(); }

 private:
  int fd_;

  FileHandle(const FileHandle&) = delete;
  void operator=(const FileHandle&) = delete;
};

// Alias for a file handle in the local process' handle space.
using LocalHandle = FileHandle<FileHandleMode::Local>;

// Alias for a file handle in another process' handle space. Handles returned
// from pushing a file object or channel must be stored in this type of handle
// class, which doesn't close the underlying file descriptor. The underlying
// file descriptor in this wrapper should not be passed to close() because doing
// so would close an unrelated file descriptor in the local handle space.
using RemoteHandle = FileHandle<FileHandleMode::Remote>;

// Alias for borrowed handles in the local process' handle space. A borrowed
// file handle is not close() because this wrapper does not own the underlying
// file descriptor. Care must be take to ensure that a borrowed file handle
// remains valid during use.
using BorrowedHandle = FileHandle<FileHandleMode::Borrowed>;

// FileReference is a 16 bit integer used in IPC serialization to be
// transferred across processes. You can convert this value to a local file
// handle by calling Transaction.GetFileHandle() on client side and
// Message.GetFileHandle() on service side.
using FileReference = int16_t;

}  // namespace pdx
}  // namespace android

#endif  // ANDROID_PDX_FILE_HANDLE_H_