summaryrefslogtreecommitdiffstats
path: root/gst/nsf/nes_apu.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/nsf/nes_apu.c')
-rw-r--r--gst/nsf/nes_apu.c134
1 files changed, 89 insertions, 45 deletions
diff --git a/gst/nsf/nes_apu.c b/gst/nsf/nes_apu.c
index 0e2c9520..47e1d72b 100644
--- a/gst/nsf/nes_apu.c
+++ b/gst/nsf/nes_apu.c
@@ -57,6 +57,11 @@ static int8 noise_long_lut[APU_NOISE_32K];
static int8 noise_short_lut[APU_NOISE_93];
#endif /* !REALTIME_NOISE */
+/* $$$ ben : last error */
+#define SET_APU_ERROR(APU,X) \
+if (APU) (APU)->errstr = "apu: " X; else
+
+#define APU_MIX_ENABLE(BIT) (apu->mix_enable&(1<<(BIT)))
/* vblank length table used for rectangles, triangle, noise */
static const uint8 vbl_length[32] = {
@@ -103,15 +108,16 @@ void
apu_setcontext (apu_t * src_apu)
{
apu = src_apu;
+ /* $$$ ben reset eoor string here. */
+ SET_APU_ERROR (apu, "no error");
}
-
/*
** Simple queue routines
*/
#define APU_QEMPTY() (apu->q_head == apu->q_tail)
-static void
+static int
apu_enqueue (apudata_t * d)
{
ASSERT (apu);
@@ -119,8 +125,12 @@ apu_enqueue (apudata_t * d)
apu->q_head = (apu->q_head + 1) & APUQUEUE_MASK;
- if (APU_QEMPTY ())
+ if (APU_QEMPTY ()) {
log_printf ("apu: queue overflow\n");
+ SET_APU_ERROR (apu, "queue overflow");
+ return -1;
+ }
+ return 0;
}
static apudata_t *
@@ -130,20 +140,33 @@ apu_dequeue (void)
ASSERT (apu);
- if (APU_QEMPTY ())
+ if (APU_QEMPTY ()) {
log_printf ("apu: queue empty\n");
-
+ SET_APU_ERROR (apu, "queue empty");
+ /* $$$ ben : should return 0 ??? */
+ }
loc = apu->q_tail;
apu->q_tail = (apu->q_tail + 1) & APUQUEUE_MASK;
return &apu->queue[loc];
}
-void
+int
apu_setchan (int chan, boolean enabled)
{
+ const unsigned int max = 6;
+ int old;
+
ASSERT (apu);
- apu->mix_enable[chan] = enabled;
+ if ((unsigned int) chan >= max) {
+ SET_APU_ERROR (apu, "channel out of range");
+ return -1;
+ }
+ old = (apu->mix_enable >> chan) & 1;
+ if (enabled != (boolean) - 1) {
+ apu->mix_enable = (apu->mix_enable & ~(1 << chan)) | ((!!enabled) << chan);
+ }
+ return old;
}
/* emulation of the 15-bit shift register the
@@ -599,7 +622,7 @@ apu_regwrite (uint32 address, uint8 value)
case APU_WRB2:
chan = (address & 4) ? 1 : 0;
apu->rectangle[chan].regs[2] = value;
-// if (apu->rectangle[chan].enabled)
+/* if (apu->rectangle[chan].enabled) */
apu->rectangle[chan].freq =
APU_TO_FIXED ((((apu->rectangle[chan].regs[3] & 7) << 8) + value) +
1);
@@ -610,7 +633,7 @@ apu_regwrite (uint32 address, uint8 value)
chan = (address & 4) ? 1 : 0;
apu->rectangle[chan].regs[3] = value;
-// if (apu->rectangle[chan].enabled)
+/* if (apu->rectangle[chan].enabled) */
{
apu->rectangle[chan].vbl_length = vbl_lut[value >> 3];
apu->rectangle[chan].env_vol = 0;
@@ -640,7 +663,7 @@ apu_regwrite (uint32 address, uint8 value)
apu->triangle.holdnote = (value & 0x80) ? TRUE : FALSE;
-// if (apu->triangle.enabled)
+/* if (apu->triangle.enabled) */
{
if (FALSE == apu->triangle.counter_started && apu->triangle.vbl_length)
apu->triangle.linear_length = trilength_lut[value & 0x7F];
@@ -652,7 +675,7 @@ apu_regwrite (uint32 address, uint8 value)
apu->triangle.regs[1] = value;
-// if (apu->triangle.enabled)
+/* if (apu->triangle.enabled) */
apu->triangle.freq =
APU_TO_FIXED ((((apu->triangle.regs[2] & 7) << 8) + value) + 1);
break;
@@ -683,7 +706,7 @@ apu_regwrite (uint32 address, uint8 value)
else
apu->triangle.countmode = COUNTMODE_LOAD;
*/
-// if (apu->triangle.enabled)
+/* if (apu->triangle.enabled) */
{
apu->triangle.freq =
APU_TO_FIXED ((((value & 7) << 8) + apu->triangle.regs[1]) + 1);
@@ -724,7 +747,7 @@ apu_regwrite (uint32 address, uint8 value)
case APU_WRD3:
apu->noise.regs[2] = value;
-// if (apu->noise.enabled)
+/* if (apu->noise.enabled) */
{
apu->noise.vbl_length = vbl_lut[value >> 3];
apu->noise.env_vol = 0; /* reset envelope */
@@ -843,7 +866,7 @@ apu_read (uint32 address)
if (apu->noise.enabled && apu->noise.vbl_length)
value |= 0x08;
- //if (apu->dmc.dma_length)
+ /* if (apu->dmc.dma_length) */
/* bodge for timestamp queue */
if (apu->dmc.enabled)
value |= 0x10;
@@ -859,8 +882,7 @@ apu_read (uint32 address)
break;
case APU_JOY1:
- value =
- input_get (INP_ZAPPER | INP_JOYPAD1
+ value = input_get (INP_ZAPPER | INP_JOYPAD1
/*| INP_ARKANOID *//*| INP_POWERPAD */ );
break;
#endif /* !NSF_PLAYER */
@@ -975,18 +997,18 @@ apu_process (void *buffer, int num_samples)
elapsed_cycles += APU_FROM_FIXED (apu->cycle_rate);
accum = 0;
- if (apu->mix_enable[0])
+ if (APU_MIX_ENABLE (0))
accum += apu_rectangle (&apu->rectangle[0]);
- if (apu->mix_enable[1])
+ if (APU_MIX_ENABLE (1))
accum += apu_rectangle (&apu->rectangle[1]);
- if (apu->mix_enable[2])
+ if (APU_MIX_ENABLE (2))
accum += apu_triangle (&apu->triangle);
- if (apu->mix_enable[3])
+ if (APU_MIX_ENABLE (3))
accum += apu_noise (&apu->noise);
- if (apu->mix_enable[4])
+ if (APU_MIX_ENABLE (4))
accum += apu_dmc (&apu->dmc);
- if (apu->ext && apu->mix_enable[5])
+ if (apu->ext && APU_MIX_ENABLE (5))
accum += apu->ext->process ();
/* do any filtering */
@@ -1013,15 +1035,11 @@ apu_process (void *buffer, int num_samples)
/* signed 16-bit output, unsigned 8-bit */
if (16 == apu->sample_bits) {
- int16 *t = buffer;
-
- *t++ = (int16) accum;
- buffer = t;
+ *(int16 *) (buffer) = (int16) accum;
+ buffer += sizeof (int16);
} else {
- uint8 *t = buffer;
-
- *t++ = (accum >> 8) ^ 0x80;
- buffer = t;
+ *(uint8 *) (buffer) = (accum >> 8) ^ 0x80;
+ buffer += sizeof (uint8);
}
}
@@ -1030,11 +1048,20 @@ apu_process (void *buffer, int num_samples)
}
/* set the filter type */
-void
+/* $$$ ben :
+ * Add a get feature (filter_type == -1) and returns old filter type
+ */
+int
apu_setfilter (int filter_type)
{
+ int old;
+
ASSERT (apu);
- apu->filter_type = filter_type;
+ old = apu->filter_type;
+ if (filter_type != -1) {
+ apu->filter_type = filter_type;
+ }
+ return old;
}
void
@@ -1100,12 +1127,16 @@ apu_t *
apu_create (int sample_rate, int refresh_rate, int sample_bits, boolean stereo)
{
apu_t *temp_apu;
- int channel;
+
+/* int channel; */
temp_apu = malloc (sizeof (apu_t));
if (NULL == temp_apu)
return NULL;
+ /* $$$ ben : safety net, in case we forgot to init something */
+ memset (temp_apu, 0, sizeof (apu_t));
+ SET_APU_ERROR (temp_apu, "no error");
temp_apu->sample_rate = sample_rate;
temp_apu->refresh_rate = refresh_rate;
temp_apu->sample_bits = sample_bits;
@@ -1124,8 +1155,9 @@ apu_create (int sample_rate, int refresh_rate, int sample_bits, boolean stereo)
apu_setactive (temp_apu);
apu_reset ();
- for (channel = 0; channel < 6; channel++)
- apu_setchan (channel, TRUE);
+ temp_apu->mix_enable = 0x3F;
+/* for (channel = 0; channel < 6; channel++) */
+/* apu_setchan(channel, TRUE); */
apu_setfilter (APU_FILTER_LOWPASS);
@@ -1142,24 +1174,30 @@ void
apu_destroy (apu_t * src_apu)
{
if (src_apu) {
- void *t = src_apu;
-
if (src_apu->ext)
src_apu->ext->shutdown ();
- free (t);
+ free (src_apu);
}
}
-void
+int
apu_setext (apu_t * src_apu, apuext_t * ext)
{
ASSERT (src_apu);
+ /* $$$ ben : seem cleaner like this */
+ if (src_apu->ext) {
+ src_apu->ext->shutdown ();
+ }
+
src_apu->ext = ext;
/* initialize it */
if (src_apu->ext)
src_apu->ext->init ();
+
+ /* $$$ ben : May be one day extension int () will return error code */
+ return 0;
}
/* this exists for external mixing routines */
@@ -1172,10 +1210,8 @@ apu_getcyclerate (void)
/*
** $Log$
-** Revision 1.1 2006/07/13 15:07:28 wtay
-** Based on patches by: Johan Dahlin <johan at gnome dot org>
-** Ronald Bultje <rbultje at ronald dot bitfreak dot net>
-** * configure.ac:
+** Revision 1.2 2008/03/25 15:56:12 slomo
+** Patch by: Andreas Henriksson <andreas at fatal dot set>
** * gst/nsf/Makefile.am:
** * gst/nsf/dis6502.h:
** * gst/nsf/fds_snd.c:
@@ -1183,7 +1219,6 @@ apu_getcyclerate (void)
** * gst/nsf/fmopl.c:
** * gst/nsf/fmopl.h:
** * gst/nsf/gstnsf.c:
-** * gst/nsf/gstnsf.h:
** * gst/nsf/log.c:
** * gst/nsf/log.h:
** * gst/nsf/memguard.c:
@@ -1202,7 +1237,16 @@ apu_getcyclerate (void)
** * gst/nsf/vrc7_snd.h:
** * gst/nsf/vrcvisnd.c:
** * gst/nsf/vrcvisnd.h:
-** Added NSF decoder plugin. Fixes 151192.
+** Update our internal nosefart to nosefart-2.7-mls to fix segfaults
+** on some files. Fixes bug #498237.
+** Remove some // comments, fix some compiler warnings and use pow()
+** instead of a slow, selfmade implementation.
+**
+** Revision 1.2 2003/04/09 14:50:32 ben
+** Clean NSF api.
+**
+** Revision 1.1 2003/04/08 20:53:01 ben
+** Adding more files...
**
** Revision 1.19 2000/07/04 04:53:26 matt
** minor changes, sound amplification