// WARNING: This file is machine generated by fidlgen.

#include <fidl/test.foreigntypeinresponseusedthroughcompose/cpp/wire_messaging.h>

#include <memory>

namespace test_foreigntypeinresponseusedthroughcompose {
[[maybe_unused]] constexpr uint64_t kTop_GetFoo_Ordinal = 4619180266423108403lu;

[[maybe_unused]] constexpr ::fidl::MessageDynamicFlags
    kTop_GetFoo_DynamicFlags = ::fidl::MessageDynamicFlags::kStrictMethod;

extern "C" const fidl_type_t test_middle_MiddleGetFooTopResponseTable;
#ifdef __Fuchsia__
}  // namespace test_foreigntypeinresponseusedthroughcompose
::fidl::WireResult<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    WireResult(::fidl::UnownedClientEnd<
                   ::test_foreigntypeinresponseusedthroughcompose::Top>
                   client,
               ::fidl::internal::TransactionalRequest<
                   ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>*
                   request) {
  FIDL_INTERNAL_DISABLE_AUTO_VAR_INIT
  ::fidl::unstable::OwnedEncodedMessage<
      ::fidl::internal::TransactionalRequest<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>,
      ::fidl::internal::ChannelTransport>
      request_message(::fidl::internal::AllowUnownedInputRef{}, request);
  auto& outgoing = request_message.GetOutgoingMessage();
  outgoing.Call<::fidl::internal::TransactionalResponse<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>>(
      client.handle(), bytes_.data(), static_cast<uint32_t>(bytes_.size()));
  SetStatus(outgoing);
}

::fidl::WireResult<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    WireResult(::fidl::UnownedClientEnd<
                   ::test_foreigntypeinresponseusedthroughcompose::Top>
                   client,
               ::fidl::internal::TransactionalRequest<
                   ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>*
                   request,
               zx_time_t deadline) {
  FIDL_INTERNAL_DISABLE_AUTO_VAR_INIT
  ::fidl::unstable::OwnedEncodedMessage<
      ::fidl::internal::TransactionalRequest<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>,
      ::fidl::internal::ChannelTransport>
      request_message(::fidl::internal::AllowUnownedInputRef{}, request);
  auto& outgoing = request_message.GetOutgoingMessage();
  outgoing.Call<::fidl::internal::TransactionalResponse<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>>(
      client.handle(), bytes_.data(), static_cast<uint32_t>(bytes_.size()),
      fidl::CallOptions{.deadline = deadline});
  SetStatus(outgoing);
}
namespace test_foreigntypeinresponseusedthroughcompose {
#endif  // __Fuchsia__

#ifdef __Fuchsia__
}  // namespace test_foreigntypeinresponseusedthroughcompose
::fidl::WireUnownedResult<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    WireUnownedResult(
        ::fidl::UnownedClientEnd<
            ::test_foreigntypeinresponseusedthroughcompose::Top>
            client_end,
        ::fidl::internal::AnyBufferAllocator& allocator,
        ::fidl::internal::TransactionalRequest<
            ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>*
            request) {
  constexpr uint32_t buffer_size = ::fidl::SyncClientMethodBufferSizeInChannel<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>();
  ::fitx::result<::fidl::Error, ::fidl::BufferSpan> allocation =
      allocator.TryAllocate(buffer_size);
  if (!allocation.is_ok()) {
    ::fidl::Status::operator=(allocation.error_value());
    return;
  }
  uint8_t* buffer = allocation->data;
  constexpr uint32_t request_byte_capacity = ::fidl::MaxSizeInChannel<
      ::fidl::internal::TransactionalRequest<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>,
      ::fidl::MessageDirection::kSending>();
  uint8_t* request_bytes = buffer;
  static_assert(buffer_size > request_byte_capacity);
  uint32_t response_byte_capacity = buffer_size - request_byte_capacity;
  uint8_t* response_bytes = &buffer[request_byte_capacity];

  ::fidl::unstable::UnownedEncodedMessage<
      ::fidl::internal::TransactionalRequest<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>>
      request_message(request_bytes, request_byte_capacity, request);
  auto& outgoing = request_message.GetOutgoingMessage();
  outgoing.Call<::fidl::internal::TransactionalResponse<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>>(
      client_end.handle(), response_bytes, response_byte_capacity);
  bytes_ = response_bytes;
  ::fidl::Status::operator=(outgoing);
}
namespace test_foreigntypeinresponseusedthroughcompose {
#endif  // __Fuchsia__

}  // namespace test_foreigntypeinresponseusedthroughcompose
#ifdef __Fuchsia__

::fidl::internal::WireThenable<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
fidl::internal::WireWeakAsyncClientImpl<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::GetFoo() {
  ::fidl::internal::TransactionalRequest<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
      _request{};
  return ::fidl::internal::WireThenable<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>{
      _client_base(), ::fidl::WriteOptions{},
      ::fidl::internal::AllowUnownedInputRef{}, &_request};
}
::fidl::internal::WireBufferThenable<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
fidl::internal::WireWeakAsyncBufferClientImpl<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::GetFoo() {
  constexpr uint32_t _buffer_size =
      ::fidl::AsyncClientMethodBufferSizeInChannel<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>();
  ::fidl::internal::TransactionalRequest<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
      _request{};
  return ::fidl::internal::WireBufferThenable<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>{
      _client_base(), ::fidl::WriteOptions{}, _allocator(), _buffer_size,
      &_request};
}

::fidl::WireResult<::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
fidl::internal::WireWeakSyncClientImpl<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::GetFoo() {
  return _client_base()->MakeSyncCallWith(
      [&](std::shared_ptr<::fidl::internal::AnyTransport> _transport) {
        ::fidl::internal::TransactionalRequest<
            ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
            _request{};
        return ::fidl::WireResult<
            ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>(
            ::fidl::UnownedClientEnd<
                ::test_foreigntypeinresponseusedthroughcompose::Top>(
                _transport->get<::fidl::internal::ChannelTransport>()),
            &_request);
      });
}
#endif  // __Fuchsia__

#ifdef __Fuchsia__

::fidl::Status fidl::WireSyncEventHandler<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::
    HandleOneEvent(::fidl::UnownedClientEnd<
                   ::test_foreigntypeinresponseusedthroughcompose::Top>
                       client_end) {
  zx_status_t status = client_end.channel()->wait_one(
      ZX_CHANNEL_READABLE | ZX_CHANNEL_PEER_CLOSED, ::zx::time::infinite(),
      nullptr);
  if (status != ZX_OK) {
    return ::fidl::Status::TransportError(
        status, ::fidl::internal::kErrorWaitOneFailed);
  }
  constexpr uint32_t kHandleAllocSize = ([]() constexpr {
    uint32_t x = 0;
    if (x > ZX_CHANNEL_MAX_MSG_HANDLES) {
      x = ZX_CHANNEL_MAX_MSG_HANDLES;
    }
    return x;
  })();
  static_assert(kHandleAllocSize <= ZX_CHANNEL_MAX_MSG_HANDLES);
  ::fidl::internal::InlineMessageBuffer<24> read_storage;
  std::array<zx_handle_t, kHandleAllocSize> read_handles;
  // TODO(fxbug.dev/85734) Remove this channel-specific allocation.
  std::array<fidl_channel_handle_metadata_t, kHandleAllocSize>
      read_handle_metadata;
  ::fidl::IncomingMessage msg = ::fidl::MessageRead(
      zx::unowned_channel(client_end.handle()), read_storage.view(),
      read_handles.data(), read_handle_metadata.data(), kHandleAllocSize,
      ReadOptions{.discardable = true});
  if (msg.status() == ZX_ERR_BUFFER_TOO_SMALL) {
    // Message size is unexpectedly larger than calculated.
    // This can only be due to a newer version of the protocol defining a new
    // event, whose size exceeds the maximum of known events in the current
    // protocol.
    return ::fidl::Status::UnexpectedMessage(
        ZX_ERR_BUFFER_TOO_SMALL,
        ::fidl::internal::kErrorSyncEventBufferTooSmall);
  }
  if (!msg.ok()) {
    return msg;
  }
  fidl_message_header_t* hdr = msg.header();
  switch (hdr->ordinal) {
    default: {
      return ::fidl::Status::UnknownOrdinal();
    }
  }
}

std::optional<::fidl::UnbindInfo> fidl::internal::WireEventDispatcher<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::
    DispatchEvent(
        ::fidl::IncomingMessage& msg,
        ::fidl::internal::IncomingTransportContext transport_context) {
  switch (msg.header()->ordinal) {
    default:
      break;
  }
  return ::fidl::UnbindInfo::UnknownOrdinal();
}
#endif  // __Fuchsia__

#ifdef __Fuchsia__

constexpr ::fidl::internal::MethodEntry fidl::internal::WireServerDispatcher<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::entries_[] = {
    {
        ::test_foreigntypeinresponseusedthroughcompose::kTop_GetFoo_Ordinal,
        [](void* interface, ::fidl::IncomingMessage&& msg,
           internal::IncomingTransportContext transport_context,
           ::fidl::Transaction* txn) {
          ::fidl::WireRequest<
              ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
              empty_request;
          auto* primary = &empty_request;
          ::fidl::internal::WireCompleter<
              ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::Sync
              completer(txn);
          reinterpret_cast<::fidl::WireServer<
              ::test_foreigntypeinresponseusedthroughcompose::Top>*>(interface)
              ->GetFoo(primary, completer);
          return ZX_OK;
        },
    },
};

const ::fidl::internal::MethodEntry* fidl::internal::WireServerDispatcher<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::entries_end_ =
    &entries_[1];

::fidl::DispatchResult fidl::internal::WireServerDispatcher<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::
    TryDispatch(::fidl::WireServer<
                    ::test_foreigntypeinresponseusedthroughcompose::Top>* impl,
                ::fidl::IncomingMessage& msg,
                internal::IncomingTransportContext transport_context,
                ::fidl::Transaction* txn) {
  return ::fidl::internal::TryDispatch(impl, msg, std::move(transport_context),
                                       txn, entries_, entries_end_);
}

void fidl::internal::WireServerDispatcher<
    ::test_foreigntypeinresponseusedthroughcompose::Top>::
    Dispatch(::fidl::WireServer<
                 ::test_foreigntypeinresponseusedthroughcompose::Top>* impl,
             ::fidl::IncomingMessage&& msg,
             internal::IncomingTransportContext transport_context,
             ::fidl::Transaction* txn) {
  ::fidl::internal::Dispatch(impl, msg, std::move(transport_context), txn,
                             entries_, entries_end_);
}
void fidl::WireServer<::test_foreigntypeinresponseusedthroughcompose::Top>::
    dispatch_message(
        ::fidl::IncomingMessage&& msg, ::fidl::Transaction* txn,
        ::fidl::internal::IncomingTransportContext transport_context) {
  ::fidl::internal::WireServerDispatcher<
      ::test_foreigntypeinresponseusedthroughcompose::Top>::
      Dispatch(this, std::move(msg), std::move(transport_context), txn);
}
#endif  // __Fuchsia__

#ifdef __Fuchsia__
void fidl::internal::WireCompleterImpl<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    Reply(const ::test_bottom::wire::Foo& foo) {
  ::fidl::internal::TransactionalResponse<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
      _response{foo};
  FIDL_INTERNAL_DISABLE_AUTO_VAR_INIT
  ::fidl::unstable::OwnedEncodedMessage<
      ::fidl::internal::TransactionalResponse<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>,
      ::fidl::internal::ChannelTransport>
      _response_message{::fidl::internal::AllowUnownedInputRef{}, &_response};
  return _core()->SendReply(&_response_message.GetOutgoingMessage(),
                            ::fidl::internal::OutgoingTransportContext());
}

void fidl::internal::WireBufferCompleterImpl<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    Reply(const ::test_bottom::wire::Foo& foo) {
  ::fidl::internal::TransactionalResponse<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>
      _response{foo};
  constexpr uint32_t _buffer_size = ::fidl::ServerReplyBufferSizeInChannel<
      ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>();
  ::fitx::result<::fidl::Error, ::fidl::BufferSpan> _allocation =
      _allocator().TryAllocate(_buffer_size);
  if (!_allocation.is_ok()) {
    ::fidl::OutgoingMessage _failure{_allocation.error_value()};
    return _core()->SendReply(&_failure,
                              ::fidl::internal::OutgoingTransportContext());
  }
  ::fidl::unstable::UnownedEncodedMessage<
      ::fidl::internal::TransactionalResponse<
          ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>,
      ::fidl::internal::ChannelTransport>
      _response_message(_allocation->data, _buffer_size, &_response);
  return _core()->SendReply(&_response_message.GetOutgoingMessage(),
                            ::fidl::internal::OutgoingTransportContext());
}
#endif  // __Fuchsia__

void ::fidl::internal::TransactionalRequest<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    _InitHeader() {
  ::fidl::InitTxnHeader(
      &header, 0,
      ::test_foreigntypeinresponseusedthroughcompose::kTop_GetFoo_Ordinal,
      ::test_foreigntypeinresponseusedthroughcompose::kTop_GetFoo_DynamicFlags);
}

void ::fidl::internal::TransactionalResponse<
    ::test_foreigntypeinresponseusedthroughcompose::Top::GetFoo>::
    _InitHeader() {
  ::fidl::InitTxnHeader(
      &header, 0,
      ::test_foreigntypeinresponseusedthroughcompose::kTop_GetFoo_Ordinal,
      ::test_foreigntypeinresponseusedthroughcompose::kTop_GetFoo_DynamicFlags);
}

#ifdef __Fuchsia__

#endif  // __Fuchsia__
