#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_