-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathMemory.cpp
120 lines (100 loc) · 2.83 KB
/
Memory.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#pragma once
#include "Memory.h"
#include <assert.h>
#include "Cartridge.h"
#include "Video.h"
Memory::Memory( const Cartridge *cartridge, Video *video )
: cartridge( cartridge )
, video ( video )
{
map = new byte[ 64_KB ];
Reset();
}
Memory::~Memory()
{
delete[] map;
}
void Memory::Reset()
{
memset( map, 0x00, 64_KB );
MapCartridge();
map[ Video::PPUCTRL_REGISTER, 0x00 ];
map[ Video::PPUMASK_REGISTER, 0x00 ];
map[ Video::PPUSTATUS_REGISTER, 0b1010'0000 ];
map[ Video::OAMA_REGISTER, 0x00 ];
map[ Video::OAMADATA_REGISTER, 0x00 ];
map[ Video::PPUSCROLL_REGISTER, 0x00 ];
map[ Video::PPUDATA_ADDRESS, 0x00 ];
}
void Memory::MapCartridge()
{
assert( cartridge != nullptr );
/* For now only support NROM with PRG ROM of 16KB and no ram */
Cartridge::Header header = cartridge->GetHeader();
assert( header.mapper == 0x00 && header.prgRomSizeKB == 16 && !header.hasPRGRam);
const byte * const rom = cartridge->GetRom();
/* Map the PRG ROM to 0x8000 */
memcpy(&map[0x8000], &rom[0x0010], 16_KB );
/* Mirror the PRG ROM in 0xC000 */
memcpy(&map[0xC000], &rom[0x0010], 16_KB );
}
byte Memory::Read( word address )
{
if ( address >= 0x2000 && address <= 0x3FFF )
{
const word ppuRegister = ( address % 8 ) + 0x2000;
if ( ppuRegister == Video::PPUSTATUS_REGISTER )
{
ResetAddressLatch();
const byte ppuStatus = map[ Video::PPUSTATUS_REGISTER ];
byte newPpuStatus = ppuStatus ^ 0b1000'0000;
Write( Video::PPUSTATUS_REGISTER, newPpuStatus );
}
return map[ ppuRegister ];
}
else
{
return map[ address ];
}
}
void Memory::Write( word address, byte data )
{
if ( address >= 0x2000 && address <= 0x3FFF )
{
const word ppuRegister = ( address % 8 ) + 0x2000;
if ( ppuRegister == Video::PPUDATA_ADDRESS )
{
const word address = map[ Video::PPUADDR_REGISTER ];
if ( IsAddressLatchClear )
{
IsAddressLatchClear = false;
currentVRamAddress = ( data | currentVRamAddress ) << 8;
}
else
{
currentVRamAddress = data | currentVRamAddress;
}
map[ Video::PPUDATA_ADDRESS ] = data;
const byte ppuControlRegister = map[ Video::PPUCTRL_REGISTER ];
const byte incrementAmount = ( ppuControlRegister | 0b0000'0100 ) >> 2;
map[ Video::PPUADDR_REGISTER ] = incrementAmount;
}
else
{
map[ ppuRegister ] = data;
}
}
else
{
map[ address ] = data;
}
}
const byte *const Memory::GetMemoryMap() const
{
return map;
}
void Memory::ResetAddressLatch()
{
IsAddressLatchClear = true;
currentVRamAddress = 0x00;
}