blob: 1c20e8c6b2ab4a13b60f202a49cad669eed3a0a0 [file] [log] [blame]
Wyatt Heplercb9d9572020-06-01 11:25:58 -07001// Copyright 2020 The Pigweed Authors
2//
3// Licensed under the Apache License, Version 2.0 (the "License"); you may not
4// use this file except in compliance with the License. You may obtain a copy of
5// the License at
6//
7// https://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12// License for the specific language governing permissions and limitations under
13// the License.
14
Wyatt Hepler74e7ab62021-06-15 19:44:32 -070015#include "pw_rpc/internal/responder.h"
Wyatt Heplercb9d9572020-06-01 11:25:58 -070016
Wyatt Heplerf298de42021-03-19 15:06:36 -070017#include "pw_assert/check.h"
Wyatt Heplercb9d9572020-06-01 11:25:58 -070018#include "pw_rpc/internal/method.h"
19#include "pw_rpc/internal/packet.h"
Wyatt Hepler60fd70f2020-06-17 09:04:25 -070020#include "pw_rpc/internal/server.h"
Wyatt Heplercb9d9572020-06-01 11:25:58 -070021
22namespace pw::rpc::internal {
23
Wyatt Hepler5180fe22021-06-15 19:32:08 -070024Responder::Responder(ServerCall& call) : call_(call), state_(kOpen) {
25 call_.server().RegisterResponder(*this);
Wyatt Hepler60fd70f2020-06-17 09:04:25 -070026}
27
Wyatt Hepler5180fe22021-06-15 19:32:08 -070028Responder& Responder::operator=(Responder&& other) {
Wyatt Heplerb7643d72020-07-10 10:33:14 -070029 Finish();
30
31 state_ = other.state_;
32
33 if (other.open()) {
Wyatt Hepler5180fe22021-06-15 19:32:08 -070034 other.call_.server().RemoveResponder(other);
Wyatt Heplerb7643d72020-07-10 10:33:14 -070035 other.state_ = kClosed;
36
Wyatt Hepler5180fe22021-06-15 19:32:08 -070037 other.call_.server().RegisterResponder(*this);
Wyatt Heplerb7643d72020-07-10 10:33:14 -070038 }
39
Wyatt Hepler671946e2020-06-09 14:39:33 -070040 call_ = std::move(other.call_);
Wyatt Heplercb9d9572020-06-01 11:25:58 -070041 response_ = std::move(other.response_);
Wyatt Hepler671946e2020-06-09 14:39:33 -070042
Wyatt Heplercb9d9572020-06-01 11:25:58 -070043 return *this;
44}
45
Wyatt Hepler5180fe22021-06-15 19:32:08 -070046uint32_t Responder::method_id() const { return call_.method().id(); }
Wyatt Hepleraf835682020-06-17 11:42:53 -070047
Wyatt Hepler5180fe22021-06-15 19:32:08 -070048Status Responder::Finish(Status status) {
Wyatt Hepler671946e2020-06-09 14:39:33 -070049 if (!open()) {
Wyatt Heplerd08e5822021-02-18 17:53:38 -080050 return Status::FailedPrecondition();
Wyatt Heplercb9d9572020-06-01 11:25:58 -070051 }
Wyatt Hepler671946e2020-06-09 14:39:33 -070052
Wyatt Hepler5180fe22021-06-15 19:32:08 -070053 // If the Responder implementer or user forgets to release an acquired
Alexei Frolovb3701c22020-11-04 11:30:47 -080054 // buffer before finishing, release it here.
55 if (!response_.empty()) {
56 ReleasePayloadBuffer();
57 }
58
Alexei Frolov062ed182020-09-28 16:23:06 -070059 Close();
Wyatt Hepleraf835682020-06-17 11:42:53 -070060
Wyatt Hepler712d3672020-07-13 15:52:11 -070061 // Send a control packet indicating that the stream (and RPC) has terminated.
Wyatt Heplerd08e5822021-02-18 17:53:38 -080062 return call_.channel().Send(Packet(PacketType::SERVER_STREAM_END,
63 call_.channel().id(),
64 call_.service().id(),
65 method().id(),
66 {},
67 status));
Wyatt Heplercb9d9572020-06-01 11:25:58 -070068}
69
Wyatt Hepler5180fe22021-06-15 19:32:08 -070070std::span<std::byte> Responder::AcquirePayloadBuffer() {
Wyatt Heplerd08e5822021-02-18 17:53:38 -080071 PW_DCHECK(open());
Wyatt Heplercb9d9572020-06-01 11:25:58 -070072
Alexei Frolovb3701c22020-11-04 11:30:47 -080073 // Only allow having one active buffer at a time.
74 if (response_.empty()) {
75 response_ = call_.channel().AcquireBuffer();
76 }
77
Wyatt Hepler0f262352020-07-29 09:51:27 -070078 return response_.payload(ResponsePacket());
Wyatt Heplercb9d9572020-06-01 11:25:58 -070079}
80
Wyatt Hepler5180fe22021-06-15 19:32:08 -070081Status Responder::ReleasePayloadBuffer(std::span<const std::byte> payload) {
Wyatt Heplerd08e5822021-02-18 17:53:38 -080082 PW_DCHECK(open());
Wyatt Hepler0f262352020-07-29 09:51:27 -070083 return call_.channel().Send(response_, ResponsePacket(payload));
Wyatt Heplercb9d9572020-06-01 11:25:58 -070084}
85
Wyatt Hepler5180fe22021-06-15 19:32:08 -070086Status Responder::ReleasePayloadBuffer() {
Wyatt Heplerd08e5822021-02-18 17:53:38 -080087 PW_DCHECK(open());
Alexei Frolov909bdfa2020-11-03 16:21:47 -080088 call_.channel().Release(response_);
Wyatt Hepler1b3da3a2021-01-07 13:26:57 -080089 return OkStatus();
Alexei Frolov909bdfa2020-11-03 16:21:47 -080090}
91
Wyatt Hepler5180fe22021-06-15 19:32:08 -070092void Responder::Close() {
Alexei Frolov062ed182020-09-28 16:23:06 -070093 if (!open()) {
94 return;
95 }
96
Wyatt Hepler5180fe22021-06-15 19:32:08 -070097 call_.server().RemoveResponder(*this);
Alexei Frolov062ed182020-09-28 16:23:06 -070098 state_ = kClosed;
99}
100
Wyatt Hepler5180fe22021-06-15 19:32:08 -0700101Packet Responder::ResponsePacket(std::span<const std::byte> payload) const {
Wyatt Hepler0f262352020-07-29 09:51:27 -0700102 return Packet(PacketType::RESPONSE,
Wyatt Hepler671946e2020-06-09 14:39:33 -0700103 call_.channel().id(),
104 call_.service().id(),
105 method().id(),
106 payload);
Wyatt Heplercb9d9572020-06-01 11:25:58 -0700107}
108
109} // namespace pw::rpc::internal