| #include "pdx/service.h" |
| |
| #include <fcntl.h> |
| #include <log/log.h> |
| #include <utils/misc.h> |
| |
| #include <algorithm> |
| #include <cstdint> |
| |
| #include <pdx/trace.h> |
| |
| namespace android { |
| namespace pdx { |
| |
| std::shared_ptr<Channel> Channel::GetFromMessageInfo(const MessageInfo& info) { |
| return info.channel ? info.channel->shared_from_this() |
| : std::shared_ptr<Channel>(); |
| } |
| |
| Message::Message() : replied_(true) {} |
| |
| Message::Message(const MessageInfo& info) |
| : service_{Service::GetFromMessageInfo(info)}, |
| channel_{Channel::GetFromMessageInfo(info)}, |
| info_{info}, |
| replied_{IsImpulse()} { |
| auto svc = service_.lock(); |
| if (svc) |
| state_ = svc->endpoint()->AllocateMessageState(); |
| } |
| |
| // C++11 specifies the move semantics for shared_ptr but not weak_ptr. This |
| // means we have to manually implement the desired move semantics for Message. |
| Message::Message(Message&& other) noexcept { *this = std::move(other); } |
| |
| Message& Message::operator=(Message&& other) noexcept { |
| Destroy(); |
| auto base = reinterpret_cast<std::uint8_t*>(&info_); |
| std::fill(&base[0], &base[sizeof(info_)], 0); |
| replied_ = true; |
| std::swap(service_, other.service_); |
| std::swap(channel_, other.channel_); |
| std::swap(info_, other.info_); |
| std::swap(state_, other.state_); |
| std::swap(replied_, other.replied_); |
| return *this; |
| } |
| |
| Message::~Message() { Destroy(); } |
| |
| void Message::Destroy() { |
| auto svc = service_.lock(); |
| if (svc) { |
| if (!replied_) { |
| ALOGE( |
| "ERROR: Service \"%s\" failed to reply to message: op=%d pid=%d " |
| "cid=%d\n", |
| svc->name_.c_str(), info_.op, info_.pid, info_.cid); |
| svc->DefaultHandleMessage(*this); |
| } |
| svc->endpoint()->FreeMessageState(state_); |
| } |
| state_ = nullptr; |
| service_.reset(); |
| channel_.reset(); |
| } |
| |
| const std::uint8_t* Message::ImpulseBegin() const { |
| return reinterpret_cast<const std::uint8_t*>(info_.impulse); |
| } |
| |
| const std::uint8_t* Message::ImpulseEnd() const { |
| return ImpulseBegin() + (IsImpulse() ? GetSendLength() : 0); |
| } |
| |
| Status<size_t> Message::ReadVector(const struct iovec* vector, |
| size_t vector_length) { |
| PDX_TRACE_NAME("Message::ReadVector"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->ReadMessageData(this, vector, vector_length); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<void> Message::ReadVectorAll(const struct iovec* vector, |
| size_t vector_length) { |
| PDX_TRACE_NAME("Message::ReadVectorAll"); |
| if (auto svc = service_.lock()) { |
| const auto status = |
| svc->endpoint()->ReadMessageData(this, vector, vector_length); |
| if (!status) |
| return status.error_status(); |
| size_t size_to_read = 0; |
| for (size_t i = 0; i < vector_length; i++) |
| size_to_read += vector[i].iov_len; |
| if (status.get() < size_to_read) |
| return ErrorStatus{EIO}; |
| return {}; |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<size_t> Message::Read(void* buffer, size_t length) { |
| PDX_TRACE_NAME("Message::Read"); |
| if (auto svc = service_.lock()) { |
| const struct iovec vector = {buffer, length}; |
| return svc->endpoint()->ReadMessageData(this, &vector, 1); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<size_t> Message::WriteVector(const struct iovec* vector, |
| size_t vector_length) { |
| PDX_TRACE_NAME("Message::WriteVector"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->WriteMessageData(this, vector, vector_length); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<void> Message::WriteVectorAll(const struct iovec* vector, |
| size_t vector_length) { |
| PDX_TRACE_NAME("Message::WriteVector"); |
| if (auto svc = service_.lock()) { |
| const auto status = |
| svc->endpoint()->WriteMessageData(this, vector, vector_length); |
| if (!status) |
| return status.error_status(); |
| size_t size_to_write = 0; |
| for (size_t i = 0; i < vector_length; i++) |
| size_to_write += vector[i].iov_len; |
| if (status.get() < size_to_write) |
| return ErrorStatus{EIO}; |
| return {}; |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<size_t> Message::Write(const void* buffer, size_t length) { |
| PDX_TRACE_NAME("Message::Write"); |
| if (auto svc = service_.lock()) { |
| const struct iovec vector = {const_cast<void*>(buffer), length}; |
| return svc->endpoint()->WriteMessageData(this, &vector, 1); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<FileReference> Message::PushFileHandle(const LocalHandle& handle) { |
| PDX_TRACE_NAME("Message::PushFileHandle"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->PushFileHandle(this, handle); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<FileReference> Message::PushFileHandle(const BorrowedHandle& handle) { |
| PDX_TRACE_NAME("Message::PushFileHandle"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->PushFileHandle(this, handle); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<FileReference> Message::PushFileHandle(const RemoteHandle& handle) { |
| PDX_TRACE_NAME("Message::PushFileHandle"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->PushFileHandle(this, handle); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<ChannelReference> Message::PushChannelHandle( |
| const LocalChannelHandle& handle) { |
| PDX_TRACE_NAME("Message::PushChannelHandle"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->PushChannelHandle(this, handle); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<ChannelReference> Message::PushChannelHandle( |
| const BorrowedChannelHandle& handle) { |
| PDX_TRACE_NAME("Message::PushChannelHandle"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->PushChannelHandle(this, handle); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<ChannelReference> Message::PushChannelHandle( |
| const RemoteChannelHandle& handle) { |
| PDX_TRACE_NAME("Message::PushChannelHandle"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->PushChannelHandle(this, handle); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| bool Message::GetFileHandle(FileReference ref, LocalHandle* handle) { |
| PDX_TRACE_NAME("Message::GetFileHandle"); |
| auto svc = service_.lock(); |
| if (!svc) |
| return false; |
| |
| if (ref >= 0) { |
| *handle = svc->endpoint()->GetFileHandle(this, ref); |
| if (!handle->IsValid()) |
| return false; |
| } else { |
| *handle = LocalHandle{ref}; |
| } |
| return true; |
| } |
| |
| bool Message::GetChannelHandle(ChannelReference ref, |
| LocalChannelHandle* handle) { |
| PDX_TRACE_NAME("Message::GetChannelHandle"); |
| auto svc = service_.lock(); |
| if (!svc) |
| return false; |
| |
| if (ref >= 0) { |
| *handle = svc->endpoint()->GetChannelHandle(this, ref); |
| if (!handle->valid()) |
| return false; |
| } else { |
| *handle = LocalChannelHandle{nullptr, ref}; |
| } |
| return true; |
| } |
| |
| Status<void> Message::Reply(int return_code) { |
| PDX_TRACE_NAME("Message::Reply"); |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| const auto ret = svc->endpoint()->MessageReply(this, return_code); |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::ReplyFileDescriptor(unsigned int fd) { |
| PDX_TRACE_NAME("Message::ReplyFileDescriptor"); |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| const auto ret = svc->endpoint()->MessageReplyFd(this, fd); |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::ReplyError(unsigned int error) { |
| PDX_TRACE_NAME("Message::ReplyError"); |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| const auto ret = |
| svc->endpoint()->MessageReply(this, -static_cast<int>(error)); |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::Reply(const LocalHandle& handle) { |
| PDX_TRACE_NAME("Message::ReplyFileHandle"); |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| Status<void> ret; |
| |
| if (handle) |
| ret = svc->endpoint()->MessageReplyFd(this, handle.Get()); |
| else |
| ret = svc->endpoint()->MessageReply(this, handle.Get()); |
| |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::Reply(const BorrowedHandle& handle) { |
| PDX_TRACE_NAME("Message::ReplyFileHandle"); |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| Status<void> ret; |
| |
| if (handle) |
| ret = svc->endpoint()->MessageReplyFd(this, handle.Get()); |
| else |
| ret = svc->endpoint()->MessageReply(this, handle.Get()); |
| |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::Reply(const RemoteHandle& handle) { |
| PDX_TRACE_NAME("Message::ReplyFileHandle"); |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| Status<void> ret; |
| |
| if (handle) |
| ret = svc->endpoint()->MessageReply(this, handle.Get()); |
| else |
| ret = svc->endpoint()->MessageReply(this, handle.Get()); |
| |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::Reply(const LocalChannelHandle& handle) { |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| const auto ret = svc->endpoint()->MessageReplyChannelHandle(this, handle); |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::Reply(const BorrowedChannelHandle& handle) { |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| const auto ret = svc->endpoint()->MessageReplyChannelHandle(this, handle); |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::Reply(const RemoteChannelHandle& handle) { |
| auto svc = service_.lock(); |
| if (!replied_ && svc) { |
| const auto ret = svc->endpoint()->MessageReplyChannelHandle(this, handle); |
| replied_ = ret.ok(); |
| return ret; |
| } else { |
| return ErrorStatus{EINVAL}; |
| } |
| } |
| |
| Status<void> Message::ModifyChannelEvents(int clear_mask, int set_mask) { |
| PDX_TRACE_NAME("Message::ModifyChannelEvents"); |
| if (auto svc = service_.lock()) { |
| return svc->endpoint()->ModifyChannelEvents(info_.cid, clear_mask, |
| set_mask); |
| } else { |
| return ErrorStatus{ESHUTDOWN}; |
| } |
| } |
| |
| Status<RemoteChannelHandle> Message::PushChannel( |
| int flags, const std::shared_ptr<Channel>& channel, int* channel_id) { |
| PDX_TRACE_NAME("Message::PushChannel"); |
| if (auto svc = service_.lock()) { |
| return svc->PushChannel(this, flags, channel, channel_id); |
| } else { |
| return ErrorStatus(ESHUTDOWN); |
| } |
| } |
| |
| Status<RemoteChannelHandle> Message::PushChannel( |
| Service* service, int flags, const std::shared_ptr<Channel>& channel, |
| int* channel_id) { |
| PDX_TRACE_NAME("Message::PushChannel"); |
| return service->PushChannel(this, flags, channel, channel_id); |
| } |
| |
| Status<int> Message::CheckChannel(ChannelReference ref, |
| std::shared_ptr<Channel>* channel) const { |
| PDX_TRACE_NAME("Message::CheckChannel"); |
| if (auto svc = service_.lock()) { |
| return svc->CheckChannel(this, ref, channel); |
| } else { |
| return ErrorStatus(ESHUTDOWN); |
| } |
| } |
| |
| Status<int> Message::CheckChannel(const Service* service, ChannelReference ref, |
| std::shared_ptr<Channel>* channel) const { |
| PDX_TRACE_NAME("Message::CheckChannel"); |
| return service->CheckChannel(this, ref, channel); |
| } |
| |
| pid_t Message::GetProcessId() const { return info_.pid; } |
| |
| pid_t Message::GetThreadId() const { return info_.tid; } |
| |
| uid_t Message::GetEffectiveUserId() const { return info_.euid; } |
| |
| gid_t Message::GetEffectiveGroupId() const { return info_.egid; } |
| |
| int Message::GetChannelId() const { return info_.cid; } |
| |
| int Message::GetMessageId() const { return info_.mid; } |
| |
| int Message::GetOp() const { return info_.op; } |
| |
| int Message::GetFlags() const { return info_.flags; } |
| |
| size_t Message::GetSendLength() const { return info_.send_len; } |
| |
| size_t Message::GetReceiveLength() const { return info_.recv_len; } |
| |
| size_t Message::GetFileDescriptorCount() const { return info_.fd_count; } |
| |
| std::shared_ptr<Channel> Message::GetChannel() const { return channel_.lock(); } |
| |
| Status<void> Message::SetChannel(const std::shared_ptr<Channel>& chan) { |
| channel_ = chan; |
| Status<void> status; |
| if (auto svc = service_.lock()) |
| status = svc->SetChannel(info_.cid, chan); |
| return status; |
| } |
| |
| std::shared_ptr<Service> Message::GetService() const { return service_.lock(); } |
| |
| const MessageInfo& Message::GetInfo() const { return info_; } |
| |
| Service::Service(const std::string& name, std::unique_ptr<Endpoint> endpoint) |
| : name_(name), endpoint_{std::move(endpoint)} { |
| if (!endpoint_) |
| return; |
| |
| const auto status = endpoint_->SetService(this); |
| ALOGE_IF(!status, "Failed to set service context because: %s", |
| status.GetErrorMessage().c_str()); |
| } |
| |
| Service::~Service() { |
| if (endpoint_) { |
| const auto status = endpoint_->SetService(nullptr); |
| ALOGE_IF(!status, "Failed to clear service context because: %s", |
| status.GetErrorMessage().c_str()); |
| } |
| } |
| |
| std::shared_ptr<Service> Service::GetFromMessageInfo(const MessageInfo& info) { |
| return info.service ? info.service->shared_from_this() |
| : std::shared_ptr<Service>(); |
| } |
| |
| bool Service::IsInitialized() const { return endpoint_.get() != nullptr; } |
| |
| std::shared_ptr<Channel> Service::OnChannelOpen(Message& /*message*/) { |
| return nullptr; |
| } |
| |
| void Service::OnChannelClose(Message& /*message*/, |
| const std::shared_ptr<Channel>& /*channel*/) {} |
| |
| Status<void> Service::SetChannel(int channel_id, |
| const std::shared_ptr<Channel>& channel) { |
| PDX_TRACE_NAME("Service::SetChannel"); |
| std::lock_guard<std::mutex> autolock(channels_mutex_); |
| |
| const auto status = endpoint_->SetChannel(channel_id, channel.get()); |
| if (!status) { |
| ALOGE("%s::SetChannel: Failed to set channel context: %s\n", name_.c_str(), |
| status.GetErrorMessage().c_str()); |
| |
| // It's possible someone mucked with things behind our back by calling the C |
| // API directly. Since we know the channel id isn't valid, make sure we |
| // don't have it in the channels map. |
| if (status.error() == ENOENT) |
| channels_.erase(channel_id); |
| } else { |
| if (channel != nullptr) |
| channels_[channel_id] = channel; |
| else |
| channels_.erase(channel_id); |
| } |
| return status; |
| } |
| |
| std::shared_ptr<Channel> Service::GetChannel(int channel_id) const { |
| PDX_TRACE_NAME("Service::GetChannel"); |
| std::lock_guard<std::mutex> autolock(channels_mutex_); |
| |
| auto search = channels_.find(channel_id); |
| if (search != channels_.end()) |
| return search->second; |
| else |
| return nullptr; |
| } |
| |
| Status<void> Service::CloseChannel(int channel_id) { |
| PDX_TRACE_NAME("Service::CloseChannel"); |
| std::lock_guard<std::mutex> autolock(channels_mutex_); |
| |
| const auto status = endpoint_->CloseChannel(channel_id); |
| |
| // Always erase the map entry, in case someone mucked with things behind our |
| // back using the C API directly. |
| channels_.erase(channel_id); |
| |
| return status; |
| } |
| |
| Status<void> Service::ModifyChannelEvents(int channel_id, int clear_mask, |
| int set_mask) { |
| PDX_TRACE_NAME("Service::ModifyChannelEvents"); |
| return endpoint_->ModifyChannelEvents(channel_id, clear_mask, set_mask); |
| } |
| |
| Status<RemoteChannelHandle> Service::PushChannel( |
| Message* message, int flags, const std::shared_ptr<Channel>& channel, |
| int* channel_id) { |
| PDX_TRACE_NAME("Service::PushChannel"); |
| |
| std::lock_guard<std::mutex> autolock(channels_mutex_); |
| |
| int channel_id_temp = -1; |
| Status<RemoteChannelHandle> ret = |
| endpoint_->PushChannel(message, flags, channel.get(), &channel_id_temp); |
| ALOGE_IF(!ret.ok(), "%s::PushChannel: Failed to push channel: %s", |
| name_.c_str(), strerror(ret.error())); |
| |
| if (channel && channel_id_temp != -1) |
| channels_[channel_id_temp] = channel; |
| if (channel_id) |
| *channel_id = channel_id_temp; |
| |
| return ret; |
| } |
| |
| Status<int> Service::CheckChannel(const Message* message, ChannelReference ref, |
| std::shared_ptr<Channel>* channel) const { |
| PDX_TRACE_NAME("Service::CheckChannel"); |
| |
| // Synchronization to maintain consistency between the kernel's channel |
| // context pointer and the userspace channels_ map. Other threads may attempt |
| // to modify the map at the same time, which could cause the channel context |
| // pointer returned by the kernel to be invalid. |
| std::lock_guard<std::mutex> autolock(channels_mutex_); |
| |
| Channel* channel_context = nullptr; |
| Status<int> ret = endpoint_->CheckChannel( |
| message, ref, channel ? &channel_context : nullptr); |
| if (ret && channel) { |
| if (channel_context) |
| *channel = channel_context->shared_from_this(); |
| else |
| *channel = nullptr; |
| } |
| |
| return ret; |
| } |
| |
| std::string Service::DumpState(size_t /*max_length*/) { return ""; } |
| |
| Status<void> Service::HandleMessage(Message& message) { |
| return DefaultHandleMessage(message); |
| } |
| |
| void Service::HandleImpulse(Message& /*impulse*/) {} |
| |
| Status<void> Service::HandleSystemMessage(Message& message) { |
| const MessageInfo& info = message.GetInfo(); |
| |
| switch (info.op) { |
| case opcodes::CHANNEL_OPEN: { |
| ALOGD("%s::OnChannelOpen: pid=%d cid=%d\n", name_.c_str(), info.pid, |
| info.cid); |
| message.SetChannel(OnChannelOpen(message)); |
| return message.Reply(0); |
| } |
| |
| case opcodes::CHANNEL_CLOSE: { |
| ALOGD("%s::OnChannelClose: pid=%d cid=%d\n", name_.c_str(), info.pid, |
| info.cid); |
| OnChannelClose(message, Channel::GetFromMessageInfo(info)); |
| message.SetChannel(nullptr); |
| return message.Reply(0); |
| } |
| |
| case opcodes::REPORT_SYSPROP_CHANGE: |
| ALOGD("%s:REPORT_SYSPROP_CHANGE: pid=%d cid=%d\n", name_.c_str(), |
| info.pid, info.cid); |
| OnSysPropChange(); |
| android::report_sysprop_change(); |
| return message.Reply(0); |
| |
| case opcodes::DUMP_STATE: { |
| ALOGD("%s:DUMP_STATE: pid=%d cid=%d\n", name_.c_str(), info.pid, |
| info.cid); |
| auto response = DumpState(message.GetReceiveLength()); |
| const size_t response_size = response.size() < message.GetReceiveLength() |
| ? response.size() |
| : message.GetReceiveLength(); |
| const Status<size_t> status = |
| message.Write(response.data(), response_size); |
| if (status && status.get() < response_size) |
| return message.ReplyError(EIO); |
| else |
| return message.Reply(status); |
| } |
| |
| default: |
| return ErrorStatus{EOPNOTSUPP}; |
| } |
| } |
| |
| Status<void> Service::DefaultHandleMessage(Message& message) { |
| const MessageInfo& info = message.GetInfo(); |
| |
| ALOGD_IF(TRACE, "Service::DefaultHandleMessage: pid=%d cid=%d op=%d\n", |
| info.pid, info.cid, info.op); |
| |
| switch (info.op) { |
| case opcodes::CHANNEL_OPEN: |
| case opcodes::CHANNEL_CLOSE: |
| case opcodes::REPORT_SYSPROP_CHANGE: |
| case opcodes::DUMP_STATE: |
| return HandleSystemMessage(message); |
| |
| default: |
| return message.ReplyError(EOPNOTSUPP); |
| } |
| } |
| |
| void Service::OnSysPropChange() {} |
| |
| Status<void> Service::ReceiveAndDispatch() { |
| Message message; |
| const auto status = endpoint_->MessageReceive(&message); |
| if (!status) { |
| ALOGE("Failed to receive message: %s\n", status.GetErrorMessage().c_str()); |
| return status; |
| } |
| |
| std::shared_ptr<Service> service = message.GetService(); |
| |
| if (!service) { |
| ALOGE("Service::ReceiveAndDispatch: service context is NULL!!!\n"); |
| // Don't block the sender indefinitely in this error case. |
| endpoint_->MessageReply(&message, -EINVAL); |
| return ErrorStatus{EINVAL}; |
| } |
| |
| if (message.IsImpulse()) { |
| service->HandleImpulse(message); |
| return {}; |
| } else if (service->HandleSystemMessage(message)) { |
| return {}; |
| } else { |
| return service->HandleMessage(message); |
| } |
| } |
| |
| Status<void> Service::Cancel() { return endpoint_->Cancel(); } |
| |
| } // namespace pdx |
| } // namespace android |