// 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/raw/internal/method.h"

#include <cstddef>
#include <cstring>

#include "pw_rpc/internal/packet.h"

namespace pw::rpc::internal {

void RawMethod::SynchronousUnaryInvoker(const CallContext& context,
                                        const Packet& request) {
  RawUnaryResponder responder(context.ClaimLocked());
  rpc_lock().unlock();
  // TODO(hepler): Remove support for raw synchronous unary methods. Unlike
  //     synchronous Nanopb methods, they provide little value compared to
  //     asynchronous unary methods. For now, just provide a fixed buffer on the
  //     stack.
  std::byte payload_buffer[64] = {};

  StatusWithSize sws =
      static_cast<const RawMethod&>(context.method())
          .function_.synchronous_unary(
              context.service(), request.payload(), span(payload_buffer));

  responder.Finish(span(payload_buffer, sws.size()), sws.status())
      .IgnoreError();
}

void RawMethod::AsynchronousUnaryInvoker(const CallContext& context,
                                         const Packet& request) {
  RawUnaryResponder responder(context.ClaimLocked());
  rpc_lock().unlock();
  static_cast<const RawMethod&>(context.method())
      .function_.asynchronous_unary(
          context.service(), request.payload(), responder);
}

void RawMethod::ServerStreamingInvoker(const CallContext& context,
                                       const Packet& request) {
  RawServerWriter server_writer(context.ClaimLocked());
  rpc_lock().unlock();
  static_cast<const RawMethod&>(context.method())
      .function_.server_streaming(
          context.service(), request.payload(), server_writer);
}

void RawMethod::ClientStreamingInvoker(const CallContext& context,
                                       const Packet&) {
  RawServerReader reader(context.ClaimLocked());
  rpc_lock().unlock();
  static_cast<const RawMethod&>(context.method())
      .function_.stream_request(context.service(), reader);
}

void RawMethod::BidirectionalStreamingInvoker(const CallContext& context,
                                              const Packet&) {
  RawServerReaderWriter reader_writer(context.ClaimLocked());
  rpc_lock().unlock();
  static_cast<const RawMethod&>(context.method())
      .function_.stream_request(context.service(), reader_writer);
}

}  // namespace pw::rpc::internal
