mirror of
https://github.com/shadps4-emu/shadPS4.git
synced 2025-01-25 16:01:41 +00:00
89 lines
2.6 KiB
C++
89 lines
2.6 KiB
C++
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#pragma once
|
|
|
|
#include "common/assert.h"
|
|
#include "common/logging/log.h"
|
|
#include "common/singleton.h"
|
|
#include "common/types.h"
|
|
#include "magic_enum.hpp"
|
|
|
|
#include <functional>
|
|
#include <mutex>
|
|
#include <optional>
|
|
#include <queue>
|
|
|
|
namespace Platform {
|
|
|
|
enum class InterruptId : u32 {
|
|
Compute0RelMem = 0u,
|
|
Compute1RelMem = 1u,
|
|
Compute2RelMem = 2u,
|
|
Compute3RelMem = 3u,
|
|
Compute4RelMem = 4u,
|
|
Compute5RelMem = 5u,
|
|
Compute6RelMem = 6u,
|
|
GfxEop = 7u,
|
|
GfxFlip = 8u
|
|
};
|
|
|
|
using IrqHandler = std::function<void(InterruptId)>;
|
|
|
|
struct IrqController {
|
|
void RegisterOnce(InterruptId irq, IrqHandler handler) {
|
|
ASSERT_MSG(static_cast<u32>(irq) < irq_contexts.size(), "Invalid IRQ number");
|
|
auto& ctx = irq_contexts[static_cast<u32>(irq)];
|
|
std::unique_lock lock{ctx.m_lock};
|
|
ctx.one_time_subscribers.emplace(handler);
|
|
}
|
|
|
|
void Register(InterruptId irq, IrqHandler handler) {
|
|
ASSERT_MSG(static_cast<u32>(irq) < irq_contexts.size(), "Invalid IRQ number");
|
|
auto& ctx = irq_contexts[static_cast<u32>(irq)];
|
|
ASSERT_MSG(!ctx.persistent_handler.has_value(),
|
|
"Too many persistent handlers"); // Add a slot map if so
|
|
|
|
std::unique_lock lock{ctx.m_lock};
|
|
ctx.persistent_handler.emplace(handler);
|
|
}
|
|
|
|
void Unregister(InterruptId irq) {
|
|
ASSERT_MSG(static_cast<u32>(irq) < irq_contexts.size(), "Invalid IRQ number");
|
|
auto& ctx = irq_contexts[static_cast<u32>(irq)];
|
|
std::unique_lock lock{ctx.m_lock};
|
|
ctx.persistent_handler.reset();
|
|
}
|
|
|
|
void Signal(InterruptId irq) {
|
|
ASSERT_MSG(static_cast<u32>(irq) < irq_contexts.size(), "Unexpected IRQ signaled");
|
|
auto& ctx = irq_contexts[static_cast<u32>(irq)];
|
|
std::unique_lock lock{ctx.m_lock};
|
|
|
|
LOG_TRACE(Core, "IRQ signaled: {}", magic_enum::enum_name(irq));
|
|
|
|
if (ctx.persistent_handler) {
|
|
ctx.persistent_handler.value()(irq);
|
|
}
|
|
|
|
while (!ctx.one_time_subscribers.empty()) {
|
|
const auto& h = ctx.one_time_subscribers.front();
|
|
h(irq);
|
|
|
|
ctx.one_time_subscribers.pop();
|
|
}
|
|
}
|
|
|
|
private:
|
|
struct IrqContext {
|
|
std::optional<IrqHandler> persistent_handler{};
|
|
std::queue<IrqHandler> one_time_subscribers{};
|
|
std::mutex m_lock{};
|
|
};
|
|
std::array<IrqContext, magic_enum::enum_count<InterruptId>()> irq_contexts{};
|
|
};
|
|
|
|
using IrqC = Common::Singleton<IrqController>;
|
|
|
|
} // namespace Platform
|