blob: 478c67ad22c71a68cf846a5d407ae1f90cf53302 [file] [log] [blame]
// 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_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(); }
void BaseServerWriter::Finish(Status status) {
if (!open()) {
return;
}
// 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.
call_.channel().Send(Packet(PacketType::SERVER_STREAM_END,
call_.channel().id(),
call_.service().id(),
method().id(),
{},
status));
}
std::span<std::byte> BaseServerWriter::AcquirePayloadBuffer() {
if (!open()) {
return {};
}
// 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) {
if (!open()) {
return Status::FailedPrecondition();
}
return call_.channel().Send(response_, ResponsePacket(payload));
}
Status BaseServerWriter::ReleasePayloadBuffer() {
if (!open()) {
return Status::FailedPrecondition();
}
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