blob: 78070b318f4391e338f3623ccf99a31b6e630bb3 [file] [log] [blame]
// Copyright 2022 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef SRC_TESTS_BENCHMARKS_FIDL_DRIVER_LLCPP_ECHO_CALL_BENCHMARK_UTIL_H_
#define SRC_TESTS_BENCHMARKS_FIDL_DRIVER_LLCPP_ECHO_CALL_BENCHMARK_UTIL_H_
#include <lib/async/cpp/task.h>
#include <lib/fdf/cpp/dispatcher.h>
#include <lib/fdf/dispatcher.h>
#include <lib/fdf/env.h>
#include <lib/fidl/cpp/wire/arena.h>
#include <lib/fidl_driver/cpp/wire_messaging.h>
#include <lib/fit/defer.h>
#include <lib/sync/cpp/completion.h>
#include <zircon/status.h>
#include <zircon/types.h>
#include <perftest/perftest.h>
namespace driver_benchmarks {
template <typename ProtocolType, typename FidlType>
class EchoServerImpl : public fdf::WireServer<ProtocolType> {
void Echo(typename fdf::WireServer<ProtocolType>::EchoRequestView request, fdf::Arena& arena,
typename fdf::WireServer<ProtocolType>::EchoCompleter::Sync& completer) override {
completer.buffer(arena).Reply(std::move(request->val));
}
};
template <typename ProtocolType, typename BuilderFunc>
bool EchoCallBenchmark(perftest::RepeatState* state, BuilderFunc builder) {
using FidlType = std::invoke_result_t<BuilderFunc, fidl::AnyArena&>;
static_assert(fidl::IsFidlType<FidlType>::value, "FIDL type required");
state->DeclareStep("Setup/WallTime");
state->DeclareStep("EchoCall/WallTime");
state->DeclareStep("Teardown/WallTime");
void* driver = reinterpret_cast<void*>(uintptr_t(1));
fdf_env_register_driver_entry(driver);
auto deferred = fit::defer([]() { fdf_env_register_driver_exit(); });
libsync::Completion dispatcher_shutdown;
auto dispatcher = fdf::SynchronizedDispatcher::Create(
{}, "", [&](fdf_dispatcher_t* dispatcher) { dispatcher_shutdown.Signal(); });
ZX_ASSERT(ZX_OK == dispatcher.status_value());
auto channels = fdf::ChannelPair::Create(0);
ZX_ASSERT(ZX_OK == channels.status_value());
fdf::ServerEnd<ProtocolType> server_end(std::move(channels->end0));
fdf::ClientEnd<ProtocolType> client_end(std::move(channels->end1));
EchoServerImpl<ProtocolType, FidlType> server;
fdf::BindServer(dispatcher->get(), std::move(server_end), &server);
fdf::WireClient<ProtocolType> client;
libsync::Completion bound;
async::PostTask(dispatcher->async_dispatcher(), [&]() {
client.Bind(std::move(client_end), dispatcher->get());
bound.Signal();
});
bound.Wait();
while (state->KeepRunning()) {
libsync::Completion completion;
async::PostTask(dispatcher->async_dispatcher(), [&]() {
fidl::Arena<65536> fidl_arena;
FidlType aligned_value = builder(fidl_arena);
fdf::Arena arena('BNCH');
state->NextStep(); // End: Setup. Begin: EchoCall.
client.buffer(arena)
->Echo(std::move(aligned_value))
.ThenExactlyOnce(
[&state, &completion](fdf::WireUnownedResult<typename ProtocolType::Echo>& result) {
state->NextStep(); // End: EchoCall. Begin: Teardown
ZX_ASSERT(result.ok());
completion.Signal();
});
});
completion.Wait();
}
libsync::Completion destroyed;
async::PostTask(dispatcher->async_dispatcher(),
[client = std::move(client), &destroyed]() { destroyed.Signal(); });
destroyed.Wait();
dispatcher->ShutdownAsync();
dispatcher_shutdown.Wait();
return true;
}
} // namespace driver_benchmarks
#endif // SRC_TESTS_BENCHMARKS_FIDL_DRIVER_LLCPP_ECHO_CALL_BENCHMARK_UTIL_H_