2022-04-28 16:24:11 +00:00
|
|
|
// SPDX-FileCopyrightText: 2016 Dolphin Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2017-11-25 13:56:57 +00:00
|
|
|
|
2018-08-07 23:24:39 +00:00
|
|
|
#include <catch2/catch.hpp>
|
2017-11-25 13:56:57 +00:00
|
|
|
|
|
|
|
#include <array>
|
|
|
|
#include <bitset>
|
2020-07-15 23:14:21 +00:00
|
|
|
#include <chrono>
|
2019-10-08 21:18:06 +00:00
|
|
|
#include <cstdlib>
|
2019-11-27 02:48:56 +00:00
|
|
|
#include <memory>
|
2017-11-25 13:56:57 +00:00
|
|
|
#include <string>
|
2019-11-27 02:48:56 +00:00
|
|
|
|
2017-11-25 13:56:57 +00:00
|
|
|
#include "core/core.h"
|
|
|
|
#include "core/core_timing.h"
|
|
|
|
|
2020-04-17 02:43:33 +00:00
|
|
|
namespace {
|
2017-11-25 13:56:57 +00:00
|
|
|
// Numbers are chosen randomly to make sure the correct one is given.
|
2020-04-23 16:58:41 +00:00
|
|
|
constexpr std::array<u64, 5> CB_IDS{{42, 144, 93, 1026, UINT64_C(0xFFFF7FFFF7FFFF)}};
|
|
|
|
constexpr std::array<u64, 5> calls_order{{2, 0, 1, 4, 3}};
|
|
|
|
std::array<s64, 5> delays{};
|
2017-11-25 13:56:57 +00:00
|
|
|
|
2020-04-17 02:43:33 +00:00
|
|
|
std::bitset<CB_IDS.size()> callbacks_ran_flags;
|
|
|
|
u64 expected_callback = 0;
|
2017-11-25 13:56:57 +00:00
|
|
|
|
|
|
|
template <unsigned int IDX>
|
2020-07-27 23:00:41 +00:00
|
|
|
void HostCallbackTemplate(std::uintptr_t user_data, std::chrono::nanoseconds ns_late) {
|
2017-11-25 13:56:57 +00:00
|
|
|
static_assert(IDX < CB_IDS.size(), "IDX out of range");
|
|
|
|
callbacks_ran_flags.set(IDX);
|
2020-07-27 23:00:41 +00:00
|
|
|
REQUIRE(CB_IDS[IDX] == user_data);
|
2020-07-15 23:14:21 +00:00
|
|
|
delays[IDX] = ns_late.count();
|
2020-02-25 02:04:12 +00:00
|
|
|
++expected_callback;
|
2017-11-25 13:56:57 +00:00
|
|
|
}
|
|
|
|
|
2019-02-14 17:42:58 +00:00
|
|
|
struct ScopeInit final {
|
2017-11-25 13:56:57 +00:00
|
|
|
ScopeInit() {
|
2020-04-01 21:28:49 +00:00
|
|
|
core_timing.SetMulticore(true);
|
2020-02-25 02:04:12 +00:00
|
|
|
core_timing.Initialize([]() {});
|
2017-11-25 13:56:57 +00:00
|
|
|
}
|
|
|
|
~ScopeInit() {
|
2019-02-14 17:42:58 +00:00
|
|
|
core_timing.Shutdown();
|
2017-11-25 13:56:57 +00:00
|
|
|
}
|
2019-02-14 17:42:58 +00:00
|
|
|
|
|
|
|
Core::Timing::CoreTiming core_timing;
|
2017-11-25 13:56:57 +00:00
|
|
|
};
|
|
|
|
|
2020-05-14 20:17:44 +00:00
|
|
|
u64 TestTimerSpeed(Core::Timing::CoreTiming& core_timing) {
|
2020-08-03 15:12:52 +00:00
|
|
|
const u64 start = core_timing.GetGlobalTimeNs().count();
|
|
|
|
volatile u64 placebo = 0;
|
2020-05-14 20:17:44 +00:00
|
|
|
for (std::size_t i = 0; i < 1000; i++) {
|
2020-08-03 15:12:52 +00:00
|
|
|
placebo = placebo + core_timing.GetGlobalTimeNs().count();
|
2020-05-14 20:17:44 +00:00
|
|
|
}
|
2020-08-03 15:12:52 +00:00
|
|
|
const u64 end = core_timing.GetGlobalTimeNs().count();
|
|
|
|
return end - start;
|
2020-05-14 20:17:44 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 16:58:41 +00:00
|
|
|
} // Anonymous namespace
|
|
|
|
|
2019-10-08 21:18:06 +00:00
|
|
|
TEST_CASE("CoreTiming[BasicOrder]", "[core]") {
|
|
|
|
ScopeInit guard;
|
|
|
|
auto& core_timing = guard.core_timing;
|
2020-02-25 02:04:12 +00:00
|
|
|
std::vector<std::shared_ptr<Core::Timing::EventType>> events{
|
|
|
|
Core::Timing::CreateEvent("callbackA", HostCallbackTemplate<0>),
|
|
|
|
Core::Timing::CreateEvent("callbackB", HostCallbackTemplate<1>),
|
|
|
|
Core::Timing::CreateEvent("callbackC", HostCallbackTemplate<2>),
|
|
|
|
Core::Timing::CreateEvent("callbackD", HostCallbackTemplate<3>),
|
|
|
|
Core::Timing::CreateEvent("callbackE", HostCallbackTemplate<4>),
|
|
|
|
};
|
|
|
|
|
|
|
|
expected_callback = 0;
|
|
|
|
|
|
|
|
core_timing.SyncPause(true);
|
|
|
|
|
2020-07-15 23:14:21 +00:00
|
|
|
const u64 one_micro = 1000U;
|
2020-02-25 02:04:12 +00:00
|
|
|
for (std::size_t i = 0; i < events.size(); i++) {
|
2020-07-15 23:14:21 +00:00
|
|
|
const u64 order = calls_order[i];
|
|
|
|
const auto future_ns = std::chrono::nanoseconds{static_cast<s64>(i * one_micro + 100)};
|
|
|
|
|
|
|
|
core_timing.ScheduleEvent(future_ns, events[order], CB_IDS[order]);
|
2020-02-25 02:04:12 +00:00
|
|
|
}
|
|
|
|
/// test pause
|
|
|
|
REQUIRE(callbacks_ran_flags.none());
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
core_timing.Pause(false); // No need to sync
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
while (core_timing.HasPendingEvents())
|
|
|
|
;
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
REQUIRE(callbacks_ran_flags.all());
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
for (std::size_t i = 0; i < delays.size(); i++) {
|
|
|
|
const double delay = static_cast<double>(delays[i]);
|
|
|
|
const double micro = delay / 1000.0f;
|
|
|
|
const double mili = micro / 1000.0f;
|
|
|
|
printf("HostTimer Pausing Delay[%zu]: %.3f %.6f\n", i, micro, mili);
|
2019-10-08 21:18:06 +00:00
|
|
|
}
|
2020-02-25 02:04:12 +00:00
|
|
|
}
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
TEST_CASE("CoreTiming[BasicOrderNoPausing]", "[core]") {
|
2019-10-08 21:18:06 +00:00
|
|
|
ScopeInit guard;
|
|
|
|
auto& core_timing = guard.core_timing;
|
2020-02-25 02:04:12 +00:00
|
|
|
std::vector<std::shared_ptr<Core::Timing::EventType>> events{
|
|
|
|
Core::Timing::CreateEvent("callbackA", HostCallbackTemplate<0>),
|
|
|
|
Core::Timing::CreateEvent("callbackB", HostCallbackTemplate<1>),
|
|
|
|
Core::Timing::CreateEvent("callbackC", HostCallbackTemplate<2>),
|
|
|
|
Core::Timing::CreateEvent("callbackD", HostCallbackTemplate<3>),
|
|
|
|
Core::Timing::CreateEvent("callbackE", HostCallbackTemplate<4>),
|
|
|
|
};
|
|
|
|
|
|
|
|
core_timing.SyncPause(true);
|
|
|
|
core_timing.SyncPause(false);
|
|
|
|
|
|
|
|
expected_callback = 0;
|
|
|
|
|
2020-07-15 22:30:06 +00:00
|
|
|
const u64 start = core_timing.GetGlobalTimeNs().count();
|
|
|
|
const u64 one_micro = 1000U;
|
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
for (std::size_t i = 0; i < events.size(); i++) {
|
2020-07-15 22:30:06 +00:00
|
|
|
const u64 order = calls_order[i];
|
|
|
|
const auto future_ns = std::chrono::nanoseconds{static_cast<s64>(i * one_micro + 100)};
|
|
|
|
core_timing.ScheduleEvent(future_ns, events[order], CB_IDS[order]);
|
2020-02-25 02:04:12 +00:00
|
|
|
}
|
2020-07-15 22:30:06 +00:00
|
|
|
|
|
|
|
const u64 end = core_timing.GetGlobalTimeNs().count();
|
2020-02-25 02:04:12 +00:00
|
|
|
const double scheduling_time = static_cast<double>(end - start);
|
|
|
|
const double timer_time = static_cast<double>(TestTimerSpeed(core_timing));
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
while (core_timing.HasPendingEvents())
|
|
|
|
;
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
REQUIRE(callbacks_ran_flags.all());
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
for (std::size_t i = 0; i < delays.size(); i++) {
|
|
|
|
const double delay = static_cast<double>(delays[i]);
|
|
|
|
const double micro = delay / 1000.0f;
|
|
|
|
const double mili = micro / 1000.0f;
|
|
|
|
printf("HostTimer No Pausing Delay[%zu]: %.3f %.6f\n", i, micro, mili);
|
|
|
|
}
|
2019-10-08 21:18:06 +00:00
|
|
|
|
2020-02-25 02:04:12 +00:00
|
|
|
const double micro = scheduling_time / 1000.0f;
|
|
|
|
const double mili = micro / 1000.0f;
|
|
|
|
printf("HostTimer No Pausing Scheduling Time: %.3f %.6f\n", micro, mili);
|
|
|
|
printf("HostTimer No Pausing Timer Time: %.3f %.6f\n", timer_time / 1000.f,
|
|
|
|
timer_time / 1000000.f);
|
2019-10-08 21:18:06 +00:00
|
|
|
}
|