| 
									
										
										
										
											2015-05-24 06:32:32 +02:00
										 |  |  | // Copyright 2014 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2+
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-02 04:44:56 +01:00
										 |  |  | #include <gtest/gtest.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | #include <unordered_set>
 | 
					
						
							| 
									
										
										
										
											2014-03-02 04:44:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   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); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-03-02 04:44:56 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-02 04:58:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-09 14:27:04 +01:00
										 |  |  | TEST(IsMMIOAddress, SpecialAddresses) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   SConfig::Init(); | 
					
						
							|  |  |  |   SConfig::GetInstance().bWii = true; | 
					
						
							| 
									
										
										
										
											2014-09-11 21:02:45 +12:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // WG Pipe address, should not be handled by MMIO.
 | 
					
						
							|  |  |  |   EXPECT_FALSE(MMIO::IsMMIOAddress(0x0C008000)); | 
					
						
							| 
									
										
										
										
											2014-03-09 14:27:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Locked L1 cache allocation.
 | 
					
						
							|  |  |  |   EXPECT_FALSE(MMIO::IsMMIOAddress(0xE0000000)); | 
					
						
							| 
									
										
										
										
											2014-09-11 21:02:45 +12:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Uncached mirror of MEM1, shouldn't be handled by MMIO
 | 
					
						
							|  |  |  |   EXPECT_FALSE(MMIO::IsMMIOAddress(0xC0000000)); | 
					
						
							| 
									
										
										
										
											2014-09-11 21:02:45 +12:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Effective address of an MMIO register; MMIO only deals with physical
 | 
					
						
							|  |  |  |   // addresses.
 | 
					
						
							|  |  |  |   EXPECT_FALSE(MMIO::IsMMIOAddress(0xCC0000E0)); | 
					
						
							| 
									
										
										
										
											2015-02-11 18:01:47 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // 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
 | 
					
						
							| 
									
										
										
										
											2014-09-11 21:02:45 +12:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   SConfig::Shutdown(); | 
					
						
							| 
									
										
										
										
											2014-03-09 14:27:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-02 04:58:29 +01:00
										 |  |  | class MappingTest : public testing::Test | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | protected: | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   virtual void SetUp() override { m_mapping = new MMIO::Mapping(); } | 
					
						
							|  |  |  |   virtual void TearDown() override { delete m_mapping; } | 
					
						
							|  |  |  |   MMIO::Mapping* m_mapping; | 
					
						
							| 
									
										
										
										
											2014-03-02 04:58:29 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST_F(MappingTest, ReadConstant) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02: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-02 04:58:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   u8 val8 = m_mapping->Read<u8>(0x0C001234); | 
					
						
							|  |  |  |   u16 val16 = m_mapping->Read<u16>(0x0C001234); | 
					
						
							|  |  |  |   u32 val32 = m_mapping->Read<u32>(0x0C001234); | 
					
						
							| 
									
										
										
										
											2014-03-02 04:58:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   EXPECT_EQ(0x42, val8); | 
					
						
							|  |  |  |   EXPECT_EQ(0x1234, val16); | 
					
						
							|  |  |  |   EXPECT_EQ(0xdeadbeef, val32); | 
					
						
							| 
									
										
										
										
											2014-03-02 04:58:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST_F(MappingTest, ReadWriteDirect) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   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); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-03-02 04:58:29 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-09 14:27:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | TEST_F(MappingTest, ReadWriteComplex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   bool read_called = false, write_called = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   m_mapping->Register(0x0C001234, MMIO::ComplexRead<u8>([&read_called](u32 addr) { | 
					
						
							|  |  |  |                         EXPECT_EQ(0x0C001234u, addr); | 
					
						
							|  |  |  |                         read_called = true; | 
					
						
							|  |  |  |                         return 0x12; | 
					
						
							|  |  |  |                       }), | 
					
						
							|  |  |  |                       MMIO::ComplexWrite<u8>([&write_called](u32 addr, u8 val) { | 
					
						
							|  |  |  |                         EXPECT_EQ(0x0C001234u, 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); | 
					
						
							| 
									
										
										
										
											2014-03-09 14:27:04 +01:00
										 |  |  | } |