| // 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_server_writer.h" |
| |
| #include "pw_assert/assert.h" |
| #include "pw_rpc/internal/method.h" |
| #include "pw_rpc/internal/packet.h" |
| #include "pw_rpc/internal/server.h" |
| |
| namespace pw::rpc::internal { |
| |
| BaseServerWriter::BaseServerWriter(ServerCall& call) |
| : call_(call), state_(kOpen) { |
| call_.server().RegisterWriter(*this); |
| } |
| |
| BaseServerWriter& BaseServerWriter::operator=(BaseServerWriter&& other) { |
| Finish(); |
| |
| state_ = other.state_; |
| |
| if (other.open()) { |
| other.call_.server().RemoveWriter(other); |
| other.state_ = kClosed; |
| |
| other.call_.server().RegisterWriter(*this); |
| } |
| |
| call_ = std::move(other.call_); |
| response_ = std::move(other.response_); |
| |
| return *this; |
| } |
| |
| uint32_t BaseServerWriter::method_id() const { return call_.method().id(); } |
| |
| Status BaseServerWriter::Finish(Status status) { |
| if (!open()) { |
| return Status::FailedPrecondition(); |
| } |
| |
| // If the ServerWriter implementer or user forgets to release an acquired |
| // buffer before finishing, release it here. |
| if (!response_.empty()) { |
| ReleasePayloadBuffer(); |
| } |
| |
| Close(); |
| |
| // Send a control packet indicating that the stream (and RPC) has terminated. |
| return call_.channel().Send(Packet(PacketType::SERVER_STREAM_END, |
| call_.channel().id(), |
| call_.service().id(), |
| method().id(), |
| {}, |
| status)); |
| } |
| |
| std::span<std::byte> BaseServerWriter::AcquirePayloadBuffer() { |
| PW_DCHECK(open()); |
| |
| // Only allow having one active buffer at a time. |
| if (response_.empty()) { |
| response_ = call_.channel().AcquireBuffer(); |
| } |
| |
| return response_.payload(ResponsePacket()); |
| } |
| |
| Status BaseServerWriter::ReleasePayloadBuffer( |
| std::span<const std::byte> payload) { |
| PW_DCHECK(open()); |
| return call_.channel().Send(response_, ResponsePacket(payload)); |
| } |
| |
| Status BaseServerWriter::ReleasePayloadBuffer() { |
| PW_DCHECK(open()); |
| call_.channel().Release(response_); |
| return OkStatus(); |
| } |
| |
| void BaseServerWriter::Close() { |
| if (!open()) { |
| return; |
| } |
| |
| call_.server().RemoveWriter(*this); |
| state_ = kClosed; |
| } |
| |
| Packet BaseServerWriter::ResponsePacket( |
| std::span<const std::byte> payload) const { |
| return Packet(PacketType::RESPONSE, |
| call_.channel().id(), |
| call_.service().id(), |
| method().id(), |
| payload); |
| } |
| |
| } // namespace pw::rpc::internal |