forked from dolphin-emu/dolphin
		
	MMIO registers are located at 0x0C000000 and 0x0D000000, not 0xCC000000. The 0xCC000000 addresses are just an artifact of address translation.
		
			
				
	
	
		
			128 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include <unordered_set>
 | 
						|
#include <gtest/gtest.h>
 | 
						|
 | 
						|
#include "Common/CommonTypes.h"
 | 
						|
#include "Core/HW/MMIO.h"
 | 
						|
 | 
						|
// 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;
 | 
						|
	for (u32 i = 0x0C000000; i < 0x0C010000; ++i)
 | 
						|
	{
 | 
						|
		u32 unique_id = MMIO::UniqueID(i);
 | 
						|
		EXPECT_EQ(ids.end(), ids.find(unique_id));
 | 
						|
		ids.insert(unique_id);
 | 
						|
	}
 | 
						|
	for (u32 i = 0x0D000000; i < 0x0D010000; ++i)
 | 
						|
	{
 | 
						|
		u32 unique_id = MMIO::UniqueID(i);
 | 
						|
		EXPECT_EQ(ids.end(), ids.find(unique_id));
 | 
						|
		ids.insert(unique_id);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
TEST(IsMMIOAddress, SpecialAddresses)
 | 
						|
{
 | 
						|
	SConfig::Init();
 | 
						|
	SConfig::GetInstance().m_LocalCoreStartupParameter.bWii = true;
 | 
						|
 | 
						|
	// WG Pipe address, should not be handled by MMIO.
 | 
						|
	EXPECT_FALSE(MMIO::IsMMIOAddress(0x0C008000));
 | 
						|
 | 
						|
	// Locked L1 cache allocation.
 | 
						|
	EXPECT_FALSE(MMIO::IsMMIOAddress(0xE0000000));
 | 
						|
 | 
						|
	// Uncached mirror of MEM1, shouldn't be handled by MMIO
 | 
						|
	EXPECT_FALSE(MMIO::IsMMIOAddress(0xC0000000));
 | 
						|
 | 
						|
	// Effective address of an MMIO register; MMIO only deals with physical
 | 
						|
	// addresses.
 | 
						|
	EXPECT_FALSE(MMIO::IsMMIOAddress(0xCC0000E0));
 | 
						|
 | 
						|
	// And lets check some valid addresses too
 | 
						|
	EXPECT_TRUE(MMIO::IsMMIOAddress(0x0C0000E0)); // Gamecube MMIOs
 | 
						|
	EXPECT_TRUE(MMIO::IsMMIOAddress(0x0D00008C)); // Wii MMIOs
 | 
						|
	EXPECT_TRUE(MMIO::IsMMIOAddress(0x0D800F10)); // Mirror of Wii MMIOs
 | 
						|
 | 
						|
	SConfig::Shutdown();
 | 
						|
}
 | 
						|
 | 
						|
class MappingTest : public testing::Test
 | 
						|
{
 | 
						|
protected:
 | 
						|
	virtual void SetUp() override
 | 
						|
	{
 | 
						|
		m_mapping = new MMIO::Mapping();
 | 
						|
	}
 | 
						|
 | 
						|
	virtual void TearDown() override
 | 
						|
	{
 | 
						|
		delete m_mapping;
 | 
						|
	}
 | 
						|
 | 
						|
	MMIO::Mapping* m_mapping;
 | 
						|
};
 | 
						|
 | 
						|
TEST_F(MappingTest, ReadConstant)
 | 
						|
{
 | 
						|
	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>());
 | 
						|
 | 
						|
	u8 val8 = m_mapping->Read<u8>(0x0C001234);
 | 
						|
	u16 val16 = m_mapping->Read<u16>(0x0C001234);
 | 
						|
	u32 val32 = m_mapping->Read<u32>(0x0C001234);
 | 
						|
 | 
						|
	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;
 | 
						|
 | 
						|
	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));
 | 
						|
 | 
						|
	for (u32 i = 0; i < 100; ++i)
 | 
						|
	{
 | 
						|
		u8 val8 = m_mapping->Read<u8>(0x0C001234);  EXPECT_EQ(i, val8);
 | 
						|
		u16 val16 = m_mapping->Read<u16>(0x0C001234); EXPECT_EQ(i, val16);
 | 
						|
		u32 val32 = m_mapping->Read<u32>(0x0C001234); EXPECT_EQ(i, val32);
 | 
						|
 | 
						|
		val8 += 1; m_mapping->Write(0x0C001234, val8);
 | 
						|
		val16 += 1; m_mapping->Write(0x0C001234, val16);
 | 
						|
		val32 += 1; m_mapping->Write(0x0C001234, val32);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(MappingTest, ReadWriteComplex)
 | 
						|
{
 | 
						|
	bool read_called = false, write_called = false;
 | 
						|
 | 
						|
	m_mapping->Register(0x0C001234,
 | 
						|
		MMIO::ComplexRead<u8>([&read_called](u32 addr) {
 | 
						|
			EXPECT_EQ(0x0C001234, addr);
 | 
						|
			read_called = true;
 | 
						|
			return 0x12;
 | 
						|
		}),
 | 
						|
		MMIO::ComplexWrite<u8>([&write_called](u32 addr, u8 val) {
 | 
						|
			EXPECT_EQ(0x0C001234, addr);
 | 
						|
			EXPECT_EQ(0x34, val);
 | 
						|
			write_called = true;
 | 
						|
		})
 | 
						|
	);
 | 
						|
 | 
						|
	u8 val = m_mapping->Read<u8>(0x0C001234); EXPECT_EQ(0x12, val);
 | 
						|
	m_mapping->Write(0x0C001234, (u8)0x34);
 | 
						|
 | 
						|
	EXPECT_TRUE(read_called);
 | 
						|
	EXPECT_TRUE(write_called);
 | 
						|
}
 |