2015-05-23 22:32:32 -06:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
2021-07-04 19:22:19 -06:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2015-05-23 22:32:32 -06:00
|
|
|
|
2014-03-01 20:44:56 -07:00
|
|
|
#include <gtest/gtest.h>
|
2024-01-11 22:42:51 -07:00
|
|
|
|
|
|
|
#include <memory>
|
2014-03-01 20:44:56 -07:00
|
|
|
#include <unordered_set>
|
|
|
|
|
|
|
|
#include "Common/CommonTypes.h"
|
2022-07-20 18:44:49 -06:00
|
|
|
#include "Core/HW/GPFifo.h"
|
2014-03-01 20:44:56 -07:00
|
|
|
#include "Core/HW/MMIO.h"
|
2024-01-11 22:42:51 -07:00
|
|
|
#include "Core/System.h"
|
2014-03-01 20:44:56 -07:00
|
|
|
|
|
|
|
// Tests that the UniqueID function returns a "unique enough" identifier
|
|
|
|
// number: that is, it is unique in the address ranges we care about.
|
|
|
|
TEST(UniqueID, UniqueEnough)
|
|
|
|
{
|
|
|
|
std::unordered_set<u32> ids;
|
2015-02-11 19:01:47 -07:00
|
|
|
for (u32 i = 0x0C000000; i < 0x0C010000; ++i)
|
2014-03-01 20:44:56 -07:00
|
|
|
{
|
|
|
|
u32 unique_id = MMIO::UniqueID(i);
|
|
|
|
EXPECT_EQ(ids.end(), ids.find(unique_id));
|
|
|
|
ids.insert(unique_id);
|
|
|
|
}
|
2015-02-11 19:01:47 -07:00
|
|
|
for (u32 i = 0x0D000000; i < 0x0D010000; ++i)
|
2014-03-01 20:44:56 -07:00
|
|
|
{
|
|
|
|
u32 unique_id = MMIO::UniqueID(i);
|
|
|
|
EXPECT_EQ(ids.end(), ids.find(unique_id));
|
|
|
|
ids.insert(unique_id);
|
|
|
|
}
|
|
|
|
}
|
2014-03-01 20:58:29 -07:00
|
|
|
|
2014-03-09 07:27:04 -06:00
|
|
|
TEST(IsMMIOAddress, SpecialAddresses)
|
|
|
|
{
|
2024-01-30 18:56:56 -07:00
|
|
|
constexpr bool is_wii = true;
|
2014-09-11 03:02:45 -06:00
|
|
|
|
2014-03-09 07:27:04 -06:00
|
|
|
// WG Pipe address, should not be handled by MMIO.
|
2024-01-30 18:56:56 -07:00
|
|
|
EXPECT_FALSE(MMIO::IsMMIOAddress(GPFifo::GATHER_PIPE_PHYSICAL_ADDRESS, is_wii));
|
2014-03-09 07:27:04 -06:00
|
|
|
|
2015-02-11 19:01:47 -07:00
|
|
|
// Locked L1 cache allocation.
|
2024-01-30 18:56:56 -07:00
|
|
|
EXPECT_FALSE(MMIO::IsMMIOAddress(0xE0000000, is_wii));
|
2014-09-11 03:02:45 -06:00
|
|
|
|
|
|
|
// Uncached mirror of MEM1, shouldn't be handled by MMIO
|
2024-01-30 18:56:56 -07:00
|
|
|
EXPECT_FALSE(MMIO::IsMMIOAddress(0xC0000000, is_wii));
|
2014-09-11 03:02:45 -06:00
|
|
|
|
2015-02-11 19:01:47 -07:00
|
|
|
// Effective address of an MMIO register; MMIO only deals with physical
|
|
|
|
// addresses.
|
2024-01-30 18:56:56 -07:00
|
|
|
EXPECT_FALSE(MMIO::IsMMIOAddress(0xCC0000E0, is_wii));
|
2015-02-11 19:01:47 -07:00
|
|
|
|
2022-07-20 18:44:49 -06:00
|
|
|
// And let's check some valid addresses too
|
2024-01-30 18:56:56 -07:00
|
|
|
EXPECT_TRUE(MMIO::IsMMIOAddress(0x0C0000E0, is_wii)); // GameCube MMIOs
|
|
|
|
EXPECT_TRUE(MMIO::IsMMIOAddress(0x0D00008C, is_wii)); // Wii MMIOs
|
|
|
|
EXPECT_TRUE(MMIO::IsMMIOAddress(0x0D800F10, is_wii)); // Mirror of Wii MMIOs
|
2014-03-09 07:27:04 -06:00
|
|
|
}
|
|
|
|
|
2014-03-01 20:58:29 -07:00
|
|
|
class MappingTest : public testing::Test
|
|
|
|
{
|
|
|
|
protected:
|
2024-01-11 22:42:51 -07:00
|
|
|
virtual void SetUp() override
|
|
|
|
{
|
|
|
|
m_system = &Core::System::GetInstance();
|
|
|
|
m_mapping = std::make_unique<MMIO::Mapping>();
|
|
|
|
}
|
|
|
|
virtual void TearDown() override
|
|
|
|
{
|
|
|
|
m_system = nullptr;
|
|
|
|
m_mapping.reset();
|
|
|
|
}
|
|
|
|
Core::System* m_system = nullptr;
|
|
|
|
std::unique_ptr<MMIO::Mapping> m_mapping;
|
2014-03-01 20:58:29 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(MappingTest, ReadConstant)
|
|
|
|
{
|
2015-02-11 19:01:47 -07:00
|
|
|
m_mapping->Register(0x0C001234, MMIO::Constant<u8>(0x42), MMIO::Nop<u8>());
|
|
|
|
m_mapping->Register(0x0C001234, MMIO::Constant<u16>(0x1234), MMIO::Nop<u16>());
|
|
|
|
m_mapping->Register(0x0C001234, MMIO::Constant<u32>(0xdeadbeef), MMIO::Nop<u32>());
|
2014-03-01 20:58:29 -07:00
|
|
|
|
2024-01-11 22:42:51 -07:00
|
|
|
u8 val8 = m_mapping->Read<u8>(*m_system, 0x0C001234);
|
|
|
|
u16 val16 = m_mapping->Read<u16>(*m_system, 0x0C001234);
|
|
|
|
u32 val32 = m_mapping->Read<u32>(*m_system, 0x0C001234);
|
2014-03-01 20:58:29 -07:00
|
|
|
|
|
|
|
EXPECT_EQ(0x42, val8);
|
|
|
|
EXPECT_EQ(0x1234, val16);
|
|
|
|
EXPECT_EQ(0xdeadbeef, val32);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(MappingTest, ReadWriteDirect)
|
|
|
|
{
|
|
|
|
u8 target_8 = 0;
|
|
|
|
u16 target_16 = 0;
|
|
|
|
u32 target_32 = 0;
|
2016-06-24 02:43:46 -06:00
|
|
|
|
2015-02-11 19:01:47 -07:00
|
|
|
m_mapping->Register(0x0C001234, MMIO::DirectRead<u8>(&target_8),
|
|
|
|
MMIO::DirectWrite<u8>(&target_8));
|
|
|
|
m_mapping->Register(0x0C001234, MMIO::DirectRead<u16>(&target_16),
|
|
|
|
MMIO::DirectWrite<u16>(&target_16));
|
|
|
|
m_mapping->Register(0x0C001234, MMIO::DirectRead<u32>(&target_32),
|
|
|
|
MMIO::DirectWrite<u32>(&target_32));
|
2016-06-24 02:43:46 -06:00
|
|
|
|
2014-03-07 23:36:44 -07:00
|
|
|
for (u32 i = 0; i < 100; ++i)
|
2014-03-01 20:58:29 -07:00
|
|
|
{
|
2024-01-11 22:42:51 -07:00
|
|
|
u8 val8 = m_mapping->Read<u8>(*m_system, 0x0C001234);
|
2015-02-11 19:01:47 -07:00
|
|
|
EXPECT_EQ(i, val8);
|
2024-01-11 22:42:51 -07:00
|
|
|
u16 val16 = m_mapping->Read<u16>(*m_system, 0x0C001234);
|
2015-02-11 19:01:47 -07:00
|
|
|
EXPECT_EQ(i, val16);
|
2024-01-11 22:42:51 -07:00
|
|
|
u32 val32 = m_mapping->Read<u32>(*m_system, 0x0C001234);
|
2015-02-11 19:01:47 -07:00
|
|
|
EXPECT_EQ(i, val32);
|
2016-06-24 02:43:46 -06:00
|
|
|
|
2015-02-11 19:01:47 -07:00
|
|
|
val8 += 1;
|
2024-01-11 22:42:51 -07:00
|
|
|
m_mapping->Write(*m_system, 0x0C001234, val8);
|
2015-02-11 19:01:47 -07:00
|
|
|
val16 += 1;
|
2024-01-11 22:42:51 -07:00
|
|
|
m_mapping->Write(*m_system, 0x0C001234, val16);
|
2015-02-11 19:01:47 -07:00
|
|
|
val32 += 1;
|
2024-01-11 22:42:51 -07:00
|
|
|
m_mapping->Write(*m_system, 0x0C001234, val32);
|
2014-03-01 20:58:29 -07:00
|
|
|
}
|
|
|
|
}
|
2014-03-09 07:27:04 -06:00
|
|
|
|
|
|
|
TEST_F(MappingTest, ReadWriteComplex)
|
|
|
|
{
|
|
|
|
bool read_called = false, write_called = false;
|
2016-06-24 02:43:46 -06:00
|
|
|
|
2022-10-13 18:17:09 -06:00
|
|
|
m_mapping->Register(0x0C001234, MMIO::ComplexRead<u8>([&read_called](Core::System&, u32 addr) {
|
2015-03-16 03:56:16 -06:00
|
|
|
EXPECT_EQ(0x0C001234u, addr);
|
2014-03-09 07:27:04 -06:00
|
|
|
read_called = true;
|
|
|
|
return 0x12;
|
|
|
|
}),
|
2022-10-13 18:17:09 -06:00
|
|
|
MMIO::ComplexWrite<u8>([&write_called](Core::System&, u32 addr, u8 val) {
|
2015-03-16 03:56:16 -06:00
|
|
|
EXPECT_EQ(0x0C001234u, addr);
|
2014-03-09 07:27:04 -06:00
|
|
|
EXPECT_EQ(0x34, val);
|
|
|
|
write_called = true;
|
|
|
|
}));
|
2016-06-24 02:43:46 -06:00
|
|
|
|
2024-01-11 22:42:51 -07:00
|
|
|
u8 val = m_mapping->Read<u8>(*m_system, 0x0C001234);
|
2015-02-11 19:01:47 -07:00
|
|
|
EXPECT_EQ(0x12, val);
|
2024-01-11 22:42:51 -07:00
|
|
|
m_mapping->Write(*m_system, 0x0C001234, (u8)0x34);
|
2016-06-24 02:43:46 -06:00
|
|
|
|
2014-03-09 07:27:04 -06:00
|
|
|
EXPECT_TRUE(read_called);
|
|
|
|
EXPECT_TRUE(write_called);
|
|
|
|
}
|