annotate Plugins/Input/console/Nes_Apu.h @ 334:0daaddb10914 trunk

[svn] Implement GYM playback.
author chainsaw
date Sun, 25 Dec 2005 13:31:46 -0800
parents 252843aac42f
children 7c5e886205ef
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
90
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
1
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
2 // NES 2A03 APU sound chip emulator
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
3
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
4 // Nes_Snd_Emu 0.1.6. Copyright (C) 2003-2005 Shay Green. GNU LGPL license.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
5
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
6 #ifndef NES_APU_H
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
7 #define NES_APU_H
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
8
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
9 typedef long nes_time_t; // CPU clock cycle count
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
10 typedef unsigned nes_addr_t; // 16-bit memory address
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
11
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
12 #include "Nes_Oscs.h"
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
13
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
14 class Tagged_Data;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
15
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
16 class Nes_Apu {
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
17 public:
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
18 Nes_Apu();
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
19 ~Nes_Apu();
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
20
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
21 // Initialization
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
22
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
23 // Reset internal frame counter, registers, and all oscillators.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
24 // Use PAL timing if pal_timing is true, otherwise use NTSC timing.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
25 // Set the DMC oscillator's initial DAC value to initial_dmc_dac without
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
26 // any audible click.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
27 void reset( bool pal_timing = false, int initial_dmc_dac = 0 );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
28
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
29 // Set memory reader callback used by DMC oscillator to fetch samples.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
30 // When callback is invoked, 'user_data' is passed unchanged as the
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
31 // first parameter.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
32 void dmc_reader( int (*callback)( void* user_data, nes_addr_t ), void* user_data = NULL );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
33
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
34 // Set IRQ time callback that is invoked when the time of earliest IRQ
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
35 // may have changed, or NULL to disable. When callback is invoked,
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
36 // 'user_data' is passed unchanged as the first parameter.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
37 void irq_notifier( void (*callback)( void* user_data ), void* user_data = NULL );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
38
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
39 // Reflect complete state between tagged data container.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
40 void reflect_state( Tagged_Data& );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
41
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
42 // Sound output
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
43
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
44 // Set overall volume (default is 1.0). Set nonlinear to true only when
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
45 // using special Nes_Nonlinearizer on output.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
46 void volume( double, bool nonlinear = false );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
47
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
48 // Set treble equalization (see notes.txt).
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
49 void treble_eq( const blip_eq_t& );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
50
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
51 // Set sound output of all oscillators to buffer. If buffer is NULL, no
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
52 // sound is generated and emulation accuracy is reduced.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
53 void output( Blip_Buffer* );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
54
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
55 // Set sound output of specific oscillator to buffer. If buffer is NULL,
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
56 // the specified oscillator is muted and emulation accuracy is reduced.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
57 // The oscillators are indexed as follows: 0) Square 1, 1) Square 2,
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
58 // 2) Triangle, 3) Noise, 4) DMC.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
59 enum { osc_count = 5 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
60 void osc_output( int index, Blip_Buffer* buffer );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
61
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
62 // Emulation
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
63
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
64 // All time values are the number of CPU clock cycles relative to the
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
65 // beginning of the current time frame.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
66
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
67 // Emulate memory write at specified time. Address must be in the range
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
68 // start_addr to end_addr.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
69 enum { start_addr = 0x4000 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
70 enum { end_addr = 0x4017 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
71 void write_register( nes_time_t, nes_addr_t, int data );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
72
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
73 // Emulate memory read of the status register at specified time.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
74 enum { status_addr = 0x4015 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
75 int read_status( nes_time_t );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
76
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
77 // Get time that APU-generated IRQ will occur if no further register reads
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
78 // or writes occur. If IRQ is already pending, returns irq_waiting. If no
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
79 // IRQ will occur, returns no_irq.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
80 enum { no_irq = LONG_MAX / 2 + 1 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
81 enum { irq_waiting = 0 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
82 nes_time_t earliest_irq() const;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
83
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
84 // Run APU until specified time, so that any DMC memory reads can be
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
85 // accounted for (i.e. inserting CPU wait states).
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
86 void run_until( nes_time_t );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
87
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
88 // Count number of DMC reads that would occur if 'run_until( t )' were executed.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
89 // If last_read is not NULL, set *last_read to the earliest time that
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
90 // 'count_dmc_reads( time )' would result in the same result.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
91 int count_dmc_reads( nes_time_t t, nes_time_t* last_read = NULL ) const;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
92
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
93 // Run all oscillators up to specified time, end current time frame, then
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
94 // start a new time frame at time 0. Time frames have no effect on emulation
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
95 // and each can be whatever length is convenient.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
96 void end_frame( nes_time_t );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
97
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
98 // End of public interface.
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
99
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
100 private:
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
101 // noncopyable
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
102 Nes_Apu( const Nes_Apu& );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
103 Nes_Apu& operator = ( const Nes_Apu& );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
104
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
105 Nes_Osc* oscs [osc_count];
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
106 Nes_Square square1;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
107 Nes_Square square2;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
108 Nes_Noise noise;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
109 Nes_Triangle triangle;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
110 Nes_Dmc dmc;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
111
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
112 nes_time_t last_time; // has been run until this time in current frame
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
113 nes_time_t earliest_irq_;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
114 nes_time_t next_irq;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
115 int frame_period;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
116 int frame_delay; // cycles until frame counter runs next
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
117 int frame; // current frame (0-3)
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
118 int osc_enables;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
119 int frame_mode;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
120 bool irq_flag;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
121 void (*irq_notifier_)( void* user_data );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
122 void* irq_data;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
123 Nes_Square::Synth square_synth; // shared by squares
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
124
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
125 void irq_changed();
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
126 void state_restored();
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
127
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
128 friend struct Nes_Dmc;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
129 };
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
130
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
131 inline void Nes_Apu::osc_output( int osc, Blip_Buffer* buf ) {
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
132 assert(( "Nes_Apu::osc_output(): Index out of range", 0 <= osc && osc < osc_count ));
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
133 oscs [osc]->output = buf;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
134 }
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
135
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
136 inline nes_time_t Nes_Apu::earliest_irq() const {
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
137 return earliest_irq_;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
138 }
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
139
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
140 inline void Nes_Apu::dmc_reader( int (*func)( void*, nes_addr_t ), void* user_data ) {
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
141 dmc.rom_reader_data = user_data;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
142 dmc.rom_reader = func;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
143 }
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
144
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
145 inline void Nes_Apu::irq_notifier( void (*func)( void* user_data ), void* user_data ) {
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
146 irq_notifier_ = func;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
147 irq_data = user_data;
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
148 }
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
149
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
150 inline int Nes_Apu::count_dmc_reads( nes_time_t time, nes_time_t* last_read ) const {
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
151 return dmc.count_reads( time, last_read );
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
152 }
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
153
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
154 #endif
252843aac42f [svn] Import the initial sources for console music support.
nenolod
parents:
diff changeset
155