| // Copyright 2020 The Pigweed Authors |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); you may not |
| // use this file except in compliance with the License. You may obtain a copy of |
| // the License at |
| // |
| // https://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| // License for the specific language governing permissions and limitations under |
| // the License. |
| |
| #include "pw_rpc/internal/base_client_call.h" |
| |
| #include "pw_rpc/client.h" |
| |
| namespace pw::rpc::internal { |
| |
| BaseClientCall& BaseClientCall::operator=(BaseClientCall&& other) { |
| // If the current client call is active, it must be unregistered from the |
| // client as it will no longer be alive after assignment. |
| Unregister(); |
| |
| active_ = other.active_; |
| client_ = other.client_; |
| channel_id_ = other.channel_id_; |
| service_id_ = other.service_id_; |
| method_id_ = other.method_id_; |
| request_ = std::move(other.request_); |
| handler_ = other.handler_; |
| |
| if (other.active()) { |
| // If the call being assigned is active, replace it in the client's list |
| // with a reference to the current object. |
| other.Unregister(); |
| other.client_->RegisterCall(*this); |
| } |
| |
| return *this; |
| } |
| |
| void BaseClientCall::Cancel() { |
| if (!active()) { |
| return; |
| } |
| |
| rpc::Channel* channel = client_->GetChannel(channel_id_); |
| if (channel != nullptr) { |
| static_cast<Channel*>(channel) |
| ->Send(NewPacket(PacketType::CANCEL)) |
| .IgnoreError(); |
| } |
| } |
| |
| std::span<std::byte> BaseClientCall::AcquirePayloadBuffer() { |
| if (!active()) { |
| return {}; |
| } |
| |
| rpc::Channel* channel = client_->GetChannel(channel_id_); |
| if (!channel) { |
| return {}; |
| } |
| |
| request_ = static_cast<Channel*>(channel)->AcquireBuffer(); |
| return request_.payload(NewPacket(PacketType::REQUEST)); |
| } |
| |
| Status BaseClientCall::ReleasePayloadBuffer( |
| std::span<const std::byte> payload) { |
| if (!active()) { |
| return Status::FailedPrecondition(); |
| } |
| |
| rpc::Channel* channel = client_->GetChannel(channel_id_); |
| if (!channel) { |
| return Status::NotFound(); |
| } |
| |
| return static_cast<Channel*>(channel)->Send( |
| request_, NewPacket(PacketType::REQUEST, payload)); |
| } |
| |
| Packet BaseClientCall::NewPacket(PacketType type, |
| std::span<const std::byte> payload) const { |
| return Packet(type, channel_id_, service_id_, method_id_, payload); |
| } |
| |
| void BaseClientCall::Register() { client_->RegisterCall(*this); } |
| |
| void BaseClientCall::Unregister() { |
| if (active()) { |
| client_->RemoveCall(*this); |
| active_ = false; |
| } |
| } |
| |
| } // namespace pw::rpc::internal |