|  | // 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/check.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 |