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

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

#include <memory>

namespace test_transitivedependenciescompose {
[[maybe_unused]] constexpr uint64_t kTop_GetFoo_Ordinal = 2618685789258237543lu;

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

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

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

#ifdef __Fuchsia__
}  // namespace test_transitivedependenciescompose
::fidl::WireUnownedResult<::test_transitivedependenciescompose::Top::GetFoo>::
    WireUnownedResult(
        ::fidl::UnownedClientEnd<::test_transitivedependenciescompose::Top>
            client_end,
        ::fidl::internal::AnyBufferAllocator& allocator,
        ::fidl::internal::TransactionalRequest<
            ::test_transitivedependenciescompose::Top::GetFoo>* request) {
  constexpr uint32_t buffer_size = ::fidl::SyncClientMethodBufferSizeInChannel<
      ::test_transitivedependenciescompose::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_transitivedependenciescompose::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_transitivedependenciescompose::Top::GetFoo>>
      request_message(request_bytes, request_byte_capacity, request);
  auto& outgoing = request_message.GetOutgoingMessage();
  outgoing.Call<::fidl::internal::TransactionalResponse<
      ::test_transitivedependenciescompose::Top::GetFoo>>(
      client_end.handle(), response_bytes, response_byte_capacity);
  bytes_ = response_bytes;
  ::fidl::Status::operator=(outgoing);
}
namespace test_transitivedependenciescompose {
#endif  // __Fuchsia__

}  // namespace test_transitivedependenciescompose
#ifdef __Fuchsia__

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

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

#ifdef __Fuchsia__

::fidl::Status
fidl::WireSyncEventHandler<::test_transitivedependenciescompose::Top>::
    HandleOneEvent(
        ::fidl::UnownedClientEnd<::test_transitivedependenciescompose::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_transitivedependenciescompose::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_transitivedependenciescompose::Top>::entries_[] = {
    {
        ::test_transitivedependenciescompose::kTop_GetFoo_Ordinal,
        [](void* interface, ::fidl::IncomingMessage&& msg,
           internal::IncomingTransportContext transport_context,
           ::fidl::Transaction* txn) {
          ::fidl::WireRequest<::test_transitivedependenciescompose::Top::GetFoo>
              empty_request;
          auto* primary = &empty_request;
          ::fidl::internal::WireCompleter<
              ::test_transitivedependenciescompose::Top::GetFoo>::Sync
              completer(txn);
          reinterpret_cast<
              ::fidl::WireServer<::test_transitivedependenciescompose::Top>*>(
              interface)
              ->GetFoo(primary, completer);
          return ZX_OK;
        },
    },
};

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

::fidl::DispatchResult fidl::internal::WireServerDispatcher<
    ::test_transitivedependenciescompose::Top>::
    TryDispatch(
        ::fidl::WireServer<::test_transitivedependenciescompose::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_transitivedependenciescompose::Top>::Dispatch(
        ::fidl::WireServer<::test_transitivedependenciescompose::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_transitivedependenciescompose::Top>::
    dispatch_message(
        ::fidl::IncomingMessage&& msg, ::fidl::Transaction* txn,
        ::fidl::internal::IncomingTransportContext transport_context) {
  ::fidl::internal::
      WireServerDispatcher<::test_transitivedependenciescompose::Top>::Dispatch(
          this, std::move(msg), std::move(transport_context), txn);
}
#endif  // __Fuchsia__

#ifdef __Fuchsia__
void fidl::internal::
    WireCompleterImpl<::test_transitivedependenciescompose::Top::GetFoo>::Reply(
        const ::test_bottom::wire::Foo& foo) {
  ::fidl::internal::TransactionalResponse<
      ::test_transitivedependenciescompose::Top::GetFoo>
      _response{foo};
  FIDL_INTERNAL_DISABLE_AUTO_VAR_INIT
  ::fidl::unstable::OwnedEncodedMessage<
      ::fidl::internal::TransactionalResponse<
          ::test_transitivedependenciescompose::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_transitivedependenciescompose::Top::GetFoo>::
    Reply(const ::test_bottom::wire::Foo& foo) {
  ::fidl::internal::TransactionalResponse<
      ::test_transitivedependenciescompose::Top::GetFoo>
      _response{foo};
  constexpr uint32_t _buffer_size = ::fidl::ServerReplyBufferSizeInChannel<
      ::test_transitivedependenciescompose::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_transitivedependenciescompose::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_transitivedependenciescompose::Top::GetFoo>::_InitHeader() {
  ::fidl::InitTxnHeader(
      &header, 0, ::test_transitivedependenciescompose::kTop_GetFoo_Ordinal,
      ::test_transitivedependenciescompose::kTop_GetFoo_DynamicFlags);
}

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

#ifdef __Fuchsia__

#endif  // __Fuchsia__
