summaryrefslogtreecommitdiff
path: root/plugins/gme/game-music-emu-0.6pre/gme/Nsf_Cpu.cpp
blob: 296c2f1cf4f883ac685e1f21cfee7e9e2dbf80ca (plain)
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
// Normal CPU for NSF emulator

// Game_Music_Emu 0.6-pre. http://www.slack.net/~ant/

#include "Nsf_Impl.h"

#include "blargg_endian.h"

#ifdef BLARGG_DEBUG_H
	//#define CPU_LOG_START 1000000
	//#include "nes_cpu_log.h"
	#undef LOG_MEM
#endif

/* Copyright (C) 2003-2008 Shay Green. This module is free software; you
can redistribute it and/or modify it under the terms of the GNU Lesser
General Public License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version. This
module is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
details. You should have received a copy of the GNU Lesser General Public
License along with this module; if not, write to the Free Software Foundation,
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */

#include "blargg_source.h"

#ifndef LOG_MEM
	#define LOG_MEM( addr, str, data ) data
#endif

int Nsf_Impl::read_mem( addr_t addr )
{
	int result = low_ram [addr & (low_ram_size-1)]; // also handles wrap-around
	if ( addr & 0xE000 )
	{
		result = *cpu.get_code( addr );
		if ( addr < sram_addr )
		{
			if ( addr == apu.status_addr )
				result = apu.read_status( time() );
			else
				result = cpu_read( addr );
		}
	}
	return LOG_MEM( addr, ">", result );
}

void Nsf_Impl::write_mem( addr_t addr, int data )
{
	(void) LOG_MEM( addr, "<", data );
	
	int offset = addr - sram_addr;
	if ( (unsigned) offset < sram_size )
	{
		sram() [offset] = data;
	}
	else
	{
		// after sram because CPU handles most low_ram accesses internally already
		int temp = addr & (low_ram_size-1); // also handles wrap-around
		if ( !(addr & 0xE000) )
		{
			low_ram [temp] = data;
		}
		else
		{
			int bank = addr - banks_addr;
			if ( (unsigned) bank < bank_count )
			{
				write_bank( bank, data );
			}
			else if ( (unsigned) (addr - apu.io_addr) < apu.io_size )
			{
				apu.write_register( time(), addr, data );
			}
			else
			{
			#if !NSF_EMU_APU_ONLY
				// 0x8000-0xDFFF is writable
				int i = addr - 0x8000;
				if ( (unsigned) i < fdsram_size && fds_enabled() )
					fdsram() [i] = data;
				else
			#endif
				cpu_write( addr, data );
			}
		}
	}
}

#define READ_LOW(  addr       ) (LOG_MEM( addr, ">", low_ram [addr] ))
#define WRITE_LOW( addr, data ) (LOG_MEM( addr, "<", low_ram [addr] = data ))

#define CAN_WRITE_FAST( addr )  (addr < low_ram_size)
#define WRITE_FAST              WRITE_LOW

// addr < 0x2000 || addr >= 0x8000
#define CAN_READ_FAST( addr )   ((addr ^ 0x8000) < 0xA000)
#define READ_FAST( addr, out  ) (LOG_MEM( addr, ">", out = READ_CODE( addr ) ))

#define READ_MEM(  addr       ) read_mem(  addr )
#define WRITE_MEM( addr, data ) write_mem( addr, data )

#define CPU cpu

#define CPU_BEGIN \
bool Nsf_Impl::run_cpu_until( time_t end )\
{\
	cpu.set_end_time( end );\
	if ( *cpu.get_code( cpu.r.pc ) != cpu.halt_opcode )\
	{
		#include "Nes_Cpu_run.h"
	}
	return cpu.time_past_end() < 0;
}