C++程序  |  158行  |  4.83 KB

#ifndef ANDROID_PDX_RPC_PAYLOAD_H_
#define ANDROID_PDX_RPC_PAYLOAD_H_

#include <iterator>

#include <pdx/client.h>
#include <pdx/rpc/message_buffer.h>
#include <pdx/service.h>

namespace android {
namespace pdx {
namespace rpc {

// Implements the payload interface, required by Serialize/Deserialize, on top
// of a thread-local MessageBuffer.
template <typename Slot>
class MessagePayload {
 public:
  using BufferType = typename MessageBuffer<Slot>::BufferType;
  using ValueType = typename MessageBuffer<Slot>::ValueType;

  // Constructs a MessagePayload with an empty TLS buffer.
  MessagePayload()
      : buffer_(MessageBuffer<Slot>::GetEmptyBuffer()),
        cursor_(buffer_.begin()),
        const_cursor_(buffer_.cbegin()) {}

  // Returns a reference to the cursor iterator to be used during serialization
  // into the underlying MessageBuffer.
  typename BufferType::iterator& Cursor() { return cursor_; }

  // Returns a reference to the const cursor iterator at the beginning of the
  // underlying MessageBuffer.
  typename BufferType::const_iterator& ConstCursor() { return const_cursor_; }

  // Returns a const iterator marking the end of the underlying MessageBuffer.
  typename BufferType::const_iterator ConstEnd() { return buffer_.cend(); }

  // Resizes the underlying MessageBuffer and sets the cursor to the beginning.
  void Resize(std::size_t size) {
    buffer_.resize(size);
    cursor_ = buffer_.begin();
    const_cursor_ = buffer_.cbegin();
  }

  // Resets the read cursor so that data can be read from the buffer again.
  void Rewind() { const_cursor_ = buffer_.cbegin(); }

  // Adds |size| bytes to the size of the underlying MessageBuffer and positions
  // the cursor at the beginning of the extended region.
  void Extend(std::size_t size) {
    const std::size_t offset = buffer_.size();
    buffer_.resize(offset + size);
    cursor_ = buffer_.begin() + offset;
    const_cursor_ = buffer_.cbegin() + offset;
  }

  // Clears the underlying MessageBuffer and sets the cursor to the beginning.
  void Clear() {
    buffer_.clear();
    cursor_ = buffer_.begin();
    const_cursor_ = buffer_.cbegin();
  }

  ValueType* Data() { return buffer_.data(); }
  const ValueType* Data() const { return buffer_.data(); }
  std::size_t Size() const { return buffer_.size(); }
  std::size_t Capacity() const { return buffer_.capacity(); }

 private:
  BufferType& buffer_;
  typename BufferType::iterator cursor_;
  typename BufferType::const_iterator const_cursor_;

  MessagePayload(const MessagePayload<Slot>&) = delete;
  void operator=(const MessagePayload<Slot>&) = delete;
};

// Implements the payload interface for service-side RPCs. Handles translating
// between remote and local handle spaces automatically.
template <typename Slot>
class ServicePayload : public MessagePayload<Slot>,
                       public MessageWriter,
                       public MessageReader {
 public:
  ServicePayload(Message& message) : message_(message) {}

  // MessageWriter
  void* GetNextWriteBufferSection(size_t size) override {
    this->Extend(size);
    return &*this->Cursor();
  }

  OutputResourceMapper* GetOutputResourceMapper() override { return &message_; }

  // MessageReader
  BufferSection GetNextReadBufferSection() override {
    return {&*this->ConstCursor(), &*this->ConstEnd()};
  }

  void ConsumeReadBufferSectionData(const void* new_start) override {
    std::advance(this->ConstCursor(),
                 PointerDistance(new_start, &*this->ConstCursor()));
  }

  InputResourceMapper* GetInputResourceMapper() override { return &message_; }

 private:
  Message& message_;
};

// Implements the payload interface for client-side RPCs. Handles gathering file
// handles to be sent over IPC automatically.
template <typename Slot>
class ClientPayload : public MessagePayload<Slot>,
                      public MessageWriter,
                      public MessageReader {
 public:
  using ContainerType =
      MessageBuffer<ThreadLocalTypeSlot<ClientPayload<Slot>>, 1024u, int>;
  using BufferType = typename ContainerType::BufferType;

  ClientPayload(Transaction& transaction) : transaction_{transaction} {}

  // MessageWriter
  void* GetNextWriteBufferSection(size_t size) override {
    this->Extend(size);
    return &*this->Cursor();
  }

  OutputResourceMapper* GetOutputResourceMapper() override {
    return &transaction_;
  }

  // MessageReader
  BufferSection GetNextReadBufferSection() override {
    return {&*this->ConstCursor(), &*this->ConstEnd()};
  }

  void ConsumeReadBufferSectionData(const void* new_start) override {
    std::advance(this->ConstCursor(),
                 PointerDistance(new_start, &*this->ConstCursor()));
  }

  InputResourceMapper* GetInputResourceMapper() override {
    return &transaction_;
  }

 private:
  Transaction& transaction_;
};

}  // namespace rpc
}  // namespace pdx
}  // namespace android

#endif  // ANDROID_PDX_RPC_PAYLOAD_H_