mirror of
https://github.com/RPCSX/rpcsx.git
synced 2026-01-31 21:04:40 +01:00
87 lines
1.8 KiB
C++
87 lines
1.8 KiB
C++
#pragma once
|
|
|
|
#include <util/types.hpp>
|
|
#include "Utilities/mutex.h"
|
|
#include "Emu/CPU/CPUThread.h"
|
|
|
|
namespace rsx
|
|
{
|
|
struct rsx_iomap_table
|
|
{
|
|
static constexpr u32 c_lock_stride = 8192;
|
|
|
|
std::array<atomic_t<u32>, 4096> ea;
|
|
std::array<atomic_t<u32>, 4096> io;
|
|
std::array<shared_mutex, 0x1'0000'0000 / c_lock_stride> rs;
|
|
|
|
rsx_iomap_table() noexcept;
|
|
|
|
// Try to get the real address given a mapped address
|
|
// Returns -1 on failure
|
|
u32 get_addr(u32 offs) const noexcept
|
|
{
|
|
return this->ea[offs >> 20] | (offs & 0xFFFFF);
|
|
}
|
|
|
|
template <bool IsFullLock, uint Stride>
|
|
bool lock(u32 addr, u32 len, cpu_thread* self = nullptr) noexcept
|
|
{
|
|
if (len <= 1) return false;
|
|
const u32 end = addr + len - 1;
|
|
|
|
bool added_wait = false;
|
|
|
|
for (u32 block = addr / c_lock_stride; block <= (end / c_lock_stride); block += Stride)
|
|
{
|
|
auto& mutex_ = rs[block];
|
|
|
|
if (IsFullLock ? !mutex_.try_lock() : !mutex_.try_lock_shared()) [[ unlikely ]]
|
|
{
|
|
if (self)
|
|
{
|
|
added_wait |= !self->state.test_and_set(cpu_flag::wait);
|
|
}
|
|
|
|
if (!self || self->id_type() != 0x55u)
|
|
{
|
|
IsFullLock ? mutex_.lock() : mutex_.lock_shared();
|
|
}
|
|
else
|
|
{
|
|
while (IsFullLock ? !mutex_.try_lock() : !mutex_.try_lock_shared())
|
|
{
|
|
self->cpu_wait({});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (added_wait)
|
|
{
|
|
self->check_state();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template <bool IsFullLock, uint Stride>
|
|
void unlock(u32 addr, u32 len) noexcept
|
|
{
|
|
ensure(len >= 1);
|
|
const u32 end = addr + len - 1;
|
|
|
|
for (u32 block = (addr / 8192); block <= (end / 8192); block += Stride)
|
|
{
|
|
if constexpr (IsFullLock)
|
|
{
|
|
rs[block].unlock();
|
|
}
|
|
else
|
|
{
|
|
rs[block].unlock_shared();
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|