mame/src/mame/audio/seibu.c
Aaron Giles 0069237f20 Memory banks are now referenced by tag rather than index.
Changed all memory_bank_* functions to specify a tag.
Bulk-converted existing banks to be tagged "bank##" in
order to ensure consistency. However, going forward, the
tags don't matter, so please name them something useful.

Added AM_BANK_READ/AM_BANK_WRITE macros to let you specify
bank tags. Also changed AM_ROMBANK and AM_RAMBANK macros to
accept tags as well.

Added new functions memory_install_read_bank_handler and
memory_install_write_bank_handler to install banks by tag
name, similar to input ports.

Changed internals of memory system to dynamically allocate
all banks. The first time a bank with an unknown tag is
installed, a new bank object is created and tracked 
internally. Removed all SMH_BANK(n) references outside of
the main code; these should never, ever be useful anymore.
2009-12-03 08:16:38 +00:00

566 lines
16 KiB
C

/***************************************************************************
Seibu Sound System v1.02, designed 1986 by Seibu Kaihatsu
The Seibu sound system comprises of a Z80A, a YM3812, a YM3931*, and
an Oki MSM6295. As well as sound the Z80 can controls coins and pass
data to the main cpu. There are a few little quirks that make it
worthwhile emulating in a seperate file:
* The YM3812 generates interrupt RST10, by asserting the interrupt line,
and placing 0xd7 on the data bus.
* The main cpu generates interrupt RST18, by asserting the interrupt line,
and placing 0xdf on the data bus.
A problem can occur if both the YM3812 and the main cpu try to assert
the interrupt line at the same time. The effect in the old Mame
emulation would be for sound to stop playing - this is because a RST18
cancelled out a RST10, and if a single RST10 is dropped sound stops
as the YM3812 timer is not reset. The problem occurs because even
if both interrupts happen at the same time, there can only be one value
on the data bus. Obviously the real hardware must have some circuit
to prevent this. It is emulated by user timers to control the z80
interrupt vector.
* The YM3931 is the main/sub cpu interface, similar to Konami's K054986A
or Taito's TC0140SYT. It also provides the Z80 memory map and
interrupt control. It's not a Yamaha chip :-)
Emulation by Bryan McPhail, mish@tendril.co.uk
ADPCM by R. Belmont and Jarek Burczynski
***************************************************************************/
#include "driver.h"
#include "streams.h"
#include "audio/seibu.h"
#include "sound/3812intf.h"
#include "sound/2151intf.h"
#include "sound/2203intf.h"
#include "sound/okim6295.h"
/*
Games using encrypted sound cpu:
Air Raid 1987 "START UP PROGRAM V1.02 (C)1986 SEIBU KAIHATSU INC."
Cabal 1988 "Michel/Seibu sound 11/04/88"
Dead Angle 1988? "START UP PROGRAM V1.02 (C)1986 SEIBU KAIHATSU INC."
Dynamite Duke 1989 "START UP PROGRAM V1.02 (C)1986 SEIBU KAIHATSU INC."
Toki 1989 "START UP PROGRAM V1.02 (C)1986 SEIBU KAIHATSU INC."
Raiden (alt) 1990 "START UP PROGRAM V1.02 (C)1986 SEIBU KAIHATSU INC."
raiden and the decrypted raidena are not identical, there are vast sections of different data.
However, there are a few different bytes in the middle of identical data, suggesting a possible
error in the decryption scheme: they all require an additional XOR with 0x20 and are located at
similar addresses.
00002422: 03 23
000024A1: 00 20
000024A2: 09 29
00002822: 48 68
000028A1: 06 26
00002A21: 17 37
00002A22: 00 20
00002AA1: 12 32
00002C21: 02 22
00002CA1: 02 22
00002CA2: 17 37
*/
static UINT8 decrypt_data(int a,int src)
{
if ( BIT(a,9) & BIT(a,8)) src ^= 0x80;
if ( BIT(a,11) & BIT(a,4) & BIT(a,1)) src ^= 0x40;
if ( BIT(a,11) & ~BIT(a,8) & BIT(a,1)) src ^= 0x04;
if ( BIT(a,13) & ~BIT(a,6) & BIT(a,4)) src ^= 0x02;
if (~BIT(a,11) & BIT(a,9) & BIT(a,2)) src ^= 0x01;
if (BIT(a,13) & BIT(a,4)) src = BITSWAP8(src,7,6,5,4,3,2,0,1);
if (BIT(a, 8) & BIT(a,4)) src = BITSWAP8(src,7,6,5,4,2,3,1,0);
return src;
}
static UINT8 decrypt_opcode(int a,int src)
{
if ( BIT(a,9) & BIT(a,8)) src ^= 0x80;
if ( BIT(a,11) & BIT(a,4) & BIT(a,1)) src ^= 0x40;
if (~BIT(a,13) & BIT(a,12)) src ^= 0x20;
if (~BIT(a,6) & BIT(a,1)) src ^= 0x10;
if (~BIT(a,12) & BIT(a,2)) src ^= 0x08;
if ( BIT(a,11) & ~BIT(a,8) & BIT(a,1)) src ^= 0x04;
if ( BIT(a,13) & ~BIT(a,6) & BIT(a,4)) src ^= 0x02;
if (~BIT(a,11) & BIT(a,9) & BIT(a,2)) src ^= 0x01;
if (BIT(a,13) & BIT(a,4)) src = BITSWAP8(src,7,6,5,4,3,2,0,1);
if (BIT(a, 8) & BIT(a,4)) src = BITSWAP8(src,7,6,5,4,2,3,1,0);
if (BIT(a,12) & BIT(a,9)) src = BITSWAP8(src,7,6,4,5,3,2,1,0);
if (BIT(a,11) & ~BIT(a,6)) src = BITSWAP8(src,6,7,5,4,3,2,1,0);
return src;
}
void seibu_sound_decrypt(running_machine *machine,const char *cpu,int length)
{
const address_space *space = cputag_get_address_space(machine, cpu, ADDRESS_SPACE_PROGRAM);
UINT8 *decrypt = auto_alloc_array(machine, UINT8, length);
UINT8 *rom = memory_region(machine, cpu);
int i;
memory_set_decrypted_region(space, 0x0000, (length < 0x10000) ? (length - 1) : 0x1fff, decrypt);
for (i = 0;i < length;i++)
{
UINT8 src = rom[i];
rom[i] = decrypt_data(i,src);
decrypt[i] = decrypt_opcode(i,src);
}
if (length > 0x10000)
memory_configure_bank_decrypted(machine, "bank1", 0, (length - 0x10000) / 0x8000, decrypt + 0x10000, 0x8000);
}
/***************************************************************************/
/*
Handlers for early Seibu/Tad games with dual channel ADPCM
*/
typedef struct _seibu_adpcm_state seibu_adpcm_state;
struct _seibu_adpcm_state
{
struct adpcm_state adpcm;
sound_stream *stream;
UINT32 current, end;
UINT8 nibble;
UINT8 playing;
UINT8 allocated;
UINT8 *base;
};
static STREAM_UPDATE( seibu_adpcm_callback )
{
seibu_adpcm_state *state = (seibu_adpcm_state *)param;
stream_sample_t *dest = outputs[0];
while (state->playing && samples > 0)
{
int val = (state->base[state->current] >> state->nibble) & 15;
state->nibble ^= 4;
if (state->nibble == 4)
{
state->current++;
if (state->current >= state->end)
state->playing = 0;
}
*dest++ = clock_adpcm(&state->adpcm, val) << 4;
samples--;
}
while (samples > 0)
{
*dest++ = 0;
samples--;
}
}
static DEVICE_START( seibu_adpcm )
{
running_machine *machine = device->machine;
seibu_adpcm_state *state = (seibu_adpcm_state *)device->token;
state->playing = 0;
state->stream = stream_create(device, 0, 1, device->clock, state, seibu_adpcm_callback);
state->base = memory_region(machine, "adpcm");
reset_adpcm(&state->adpcm);
}
DEVICE_GET_INFO( seibu_adpcm )
{
switch (state)
{
/* --- the following bits of info are returned as 64-bit signed integers --- */
case DEVINFO_INT_TOKEN_BYTES: info->i = sizeof(seibu_adpcm_state); break;
/* --- the following bits of info are returned as pointers to data or functions --- */
case DEVINFO_FCT_START: info->start = DEVICE_START_NAME(seibu_adpcm); break;
/* --- the following bits of info are returned as NULL-terminated strings --- */
case DEVINFO_STR_NAME: strcpy(info->s, "Seibu ADPCM"); break;
case DEVINFO_STR_SOURCE_FILE: strcpy(info->s, __FILE__); break;
}
}
// "decrypt" is a bit flowery here, as it's probably just line-swapping to
// simplify PCB layout/routing rather than intentional protection, but it
// still fits, especially since the Z80s for all these games are truly encrypted.
void seibu_adpcm_decrypt(running_machine *machine, const char *region)
{
UINT8 *ROM = memory_region(machine, region);
int len = memory_region_length(machine, region);
int i;
for (i = 0; i < len; i++)
{
ROM[i] = BITSWAP8(ROM[i], 7, 5, 3, 1, 6, 4, 2, 0);
}
}
WRITE8_DEVICE_HANDLER( seibu_adpcm_adr_w )
{
seibu_adpcm_state *state = (seibu_adpcm_state *)device->token;
if (state->stream)
stream_update(state->stream);
if (offset)
{
state->end = data<<8;
}
else
{
state->current = data<<8;
state->nibble = 4;
}
}
WRITE8_DEVICE_HANDLER( seibu_adpcm_ctl_w )
{
seibu_adpcm_state *state = (seibu_adpcm_state *)device->token;
// sequence is 00 02 01 each time.
if (state->stream)
stream_update(state->stream);
switch (data)
{
case 0:
state->playing = 0;
break;
case 2:
break;
case 1:
state->playing = 1;
break;
}
}
/***************************************************************************/
static const device_config *sound_cpu;
enum
{
VECTOR_INIT,
RST10_ASSERT,
RST10_CLEAR,
RST18_ASSERT,
RST18_CLEAR
};
static void update_irq_lines(running_machine *machine, int param)
{
static int irq1,irq2;
switch(param)
{
case VECTOR_INIT:
irq1 = irq2 = 0xff;
break;
case RST10_ASSERT:
irq1 = 0xd7;
break;
case RST10_CLEAR:
irq1 = 0xff;
break;
case RST18_ASSERT:
irq2 = 0xdf;
break;
case RST18_CLEAR:
irq2 = 0xff;
break;
}
if ((irq1 & irq2) == 0xff) /* no IRQs pending */
cpu_set_input_line(sound_cpu,0,CLEAR_LINE);
else /* IRQ pending */
cpu_set_input_line_and_vector(sound_cpu,0,ASSERT_LINE,irq1 & irq2);
}
WRITE8_HANDLER( seibu_irq_clear_w )
{
update_irq_lines(space->machine, VECTOR_INIT);
}
WRITE8_HANDLER( seibu_rst10_ack_w )
{
/* Unused for now */
}
WRITE8_HANDLER( seibu_rst18_ack_w )
{
update_irq_lines(space->machine, RST18_CLEAR);
}
void seibu_ym3812_irqhandler(const device_config *device, int linestate)
{
update_irq_lines(device->machine, linestate ? RST10_ASSERT : RST10_CLEAR);
}
void seibu_ym2151_irqhandler(const device_config *device, int linestate)
{
update_irq_lines(device->machine, linestate ? RST10_ASSERT : RST10_CLEAR);
}
void seibu_ym2203_irqhandler(const device_config *device, int linestate)
{
update_irq_lines(device->machine, linestate ? RST10_ASSERT : RST10_CLEAR);
}
/***************************************************************************/
MACHINE_RESET( seibu_sound )
{
int romlength = memory_region_length(machine, "audiocpu");
UINT8 *rom = memory_region(machine, "audiocpu");
sound_cpu=cputag_get_cpu(machine, "audiocpu");
update_irq_lines(machine, VECTOR_INIT);
if (romlength > 0x10000)
memory_configure_bank(machine, "bank1", 0, (romlength - 0x10000) / 0x8000, rom + 0x10000, 0x8000);
}
/***************************************************************************/
static UINT8 main2sub[2],sub2main[2];
static int main2sub_pending,sub2main_pending;
WRITE8_HANDLER( seibu_bank_w )
{
memory_set_bank(space->machine, "bank1", data & 1);
}
WRITE8_HANDLER( seibu_coin_w )
{
coin_counter_w(space->machine, 0,data & 1);
coin_counter_w(space->machine, 1,data & 2);
}
READ8_HANDLER( seibu_soundlatch_r )
{
return main2sub[offset];
}
READ8_HANDLER( seibu_main_data_pending_r )
{
return sub2main_pending ? 1 : 0;
}
WRITE8_HANDLER( seibu_main_data_w )
{
sub2main[offset] = data;
}
static WRITE8_HANDLER( seibu_pending_w )
{
/* just a guess */
main2sub_pending = 0;
sub2main_pending = 1;
}
READ16_HANDLER( seibu_main_word_r )
{
//logerror("%06x: seibu_main_word_r(%x)\n",cpu_get_pc(space->cpu),offset);
switch (offset)
{
case 2:
case 3:
return sub2main[offset-2];
case 5:
return main2sub_pending ? 1 : 0;
default:
//logerror("%06x: seibu_main_word_r(%x)\n",cpu_get_pc(space->cpu),offset);
return 0xffff;
}
}
WRITE16_HANDLER( seibu_main_word_w )
{
//printf("%06x: seibu_main_word_w(%x,%02x)\n",cpu_get_pc(space->cpu),offset,data);
if (ACCESSING_BITS_0_7)
{
switch (offset)
{
case 0:
case 1:
main2sub[offset] = data;
break;
case 4:
if (strcmp(space->machine->gamedrv->name, "sdgndmps") == 0)
update_irq_lines(space->machine, RST10_ASSERT);
update_irq_lines(space->machine, RST18_ASSERT);
break;
case 2: //Sengoku Mahjong writes here
case 6:
/* just a guess */
sub2main_pending = 0;
main2sub_pending = 1;
break;
default:
//logerror("%06x: seibu_main_word_w(%x,%02x)\n",cpu_get_pc(space->cpu),offset,data);
break;
}
}
}
READ8_HANDLER( seibu_main_v30_r )
{
return seibu_main_word_r(space,offset/2,0xffff) >> (8 * (offset & 1));
}
WRITE8_HANDLER( seibu_main_v30_w )
{
seibu_main_word_w(space,offset/2,data << (8 * (offset & 1)),0x00ff << (8 * (offset & 1)));
}
WRITE16_HANDLER( seibu_main_mustb_w )
{
main2sub[0] = data&0xff;
main2sub[1] = data>>8;
// logerror("seibu_main_mustb_w: %x -> %x %x\n", data, main2sub[0], main2sub[1]);
update_irq_lines(space->machine, RST18_ASSERT);
}
/***************************************************************************/
const ym3812_interface seibu_ym3812_interface =
{
seibu_ym3812_irqhandler
};
const ym2151_interface seibu_ym2151_interface =
{
seibu_ym2151_irqhandler
};
const ym2203_interface seibu_ym2203_interface =
{
{
AY8910_LEGACY_OUTPUT,
AY8910_DEFAULT_LOADS,
DEVCB_NULL, DEVCB_NULL, DEVCB_NULL, DEVCB_NULL
},
seibu_ym2203_irqhandler
};
/***************************************************************************/
ADDRESS_MAP_START( seibu_sound_map, ADDRESS_SPACE_PROGRAM, 8 )
AM_RANGE(0x0000, 0x1fff) AM_ROM
AM_RANGE(0x2000, 0x27ff) AM_RAM
AM_RANGE(0x4000, 0x4000) AM_WRITE(seibu_pending_w)
AM_RANGE(0x4001, 0x4001) AM_WRITE(seibu_irq_clear_w)
AM_RANGE(0x4002, 0x4002) AM_WRITE(seibu_rst10_ack_w)
AM_RANGE(0x4003, 0x4003) AM_WRITE(seibu_rst18_ack_w)
AM_RANGE(0x4007, 0x4007) AM_WRITE(seibu_bank_w)
AM_RANGE(0x4008, 0x4009) AM_DEVREADWRITE("ymsnd", ym3812_r, ym3812_w)
AM_RANGE(0x4010, 0x4011) AM_READ(seibu_soundlatch_r)
AM_RANGE(0x4012, 0x4012) AM_READ(seibu_main_data_pending_r)
AM_RANGE(0x4013, 0x4013) AM_READ_PORT("COIN")
AM_RANGE(0x4018, 0x4019) AM_WRITE(seibu_main_data_w)
AM_RANGE(0x401b, 0x401b) AM_WRITE(seibu_coin_w)
AM_RANGE(0x6000, 0x6000) AM_DEVREADWRITE("oki", okim6295_r, okim6295_w)
AM_RANGE(0x8000, 0xffff) AM_ROMBANK("bank1")
ADDRESS_MAP_END
ADDRESS_MAP_START( seibu2_sound_map, ADDRESS_SPACE_PROGRAM, 8 )
AM_RANGE(0x0000, 0x1fff) AM_ROM
AM_RANGE(0x2000, 0x27ff) AM_RAM
AM_RANGE(0x4000, 0x4000) AM_WRITE(seibu_pending_w)
AM_RANGE(0x4001, 0x4001) AM_WRITE(seibu_irq_clear_w)
AM_RANGE(0x4002, 0x4002) AM_WRITE(seibu_rst10_ack_w)
AM_RANGE(0x4003, 0x4003) AM_WRITE(seibu_rst18_ack_w)
AM_RANGE(0x4007, 0x4007) AM_WRITE(seibu_bank_w)
AM_RANGE(0x4008, 0x4009) AM_DEVREADWRITE("ymsnd", ym2151_r, ym2151_w)
AM_RANGE(0x4010, 0x4011) AM_READ(seibu_soundlatch_r)
AM_RANGE(0x4012, 0x4012) AM_READ(seibu_main_data_pending_r)
AM_RANGE(0x4013, 0x4013) AM_READ_PORT("COIN")
AM_RANGE(0x4018, 0x4019) AM_WRITE(seibu_main_data_w)
AM_RANGE(0x401b, 0x401b) AM_WRITE(seibu_coin_w)
AM_RANGE(0x6000, 0x6000) AM_DEVREADWRITE("oki", okim6295_r, okim6295_w)
AM_RANGE(0x8000, 0xffff) AM_ROMBANK("bank1")
ADDRESS_MAP_END
ADDRESS_MAP_START( seibu2_raiden2_sound_map, ADDRESS_SPACE_PROGRAM, 8 )
AM_RANGE(0x0000, 0x1fff) AM_ROM
AM_RANGE(0x2000, 0x27ff) AM_RAM
AM_RANGE(0x4000, 0x4000) AM_WRITE(seibu_pending_w)
AM_RANGE(0x4001, 0x4001) AM_WRITE(seibu_irq_clear_w)
AM_RANGE(0x4002, 0x4002) AM_WRITE(seibu_rst10_ack_w)
AM_RANGE(0x4003, 0x4003) AM_WRITE(seibu_rst18_ack_w)
AM_RANGE(0x4007, 0x4007) AM_WRITE(seibu_bank_w)
AM_RANGE(0x4008, 0x4009) AM_DEVREADWRITE("ymsnd", ym2151_r, ym2151_w)
AM_RANGE(0x4010, 0x4011) AM_READ(seibu_soundlatch_r)
AM_RANGE(0x4012, 0x4012) AM_READ(seibu_main_data_pending_r)
AM_RANGE(0x4013, 0x4013) AM_READ_PORT("COIN")
AM_RANGE(0x4018, 0x4019) AM_WRITE(seibu_main_data_w)
AM_RANGE(0x401b, 0x401b) AM_WRITE(seibu_coin_w)
AM_RANGE(0x6000, 0x6000) AM_DEVREADWRITE("oki1", okim6295_r, okim6295_w)
AM_RANGE(0x6002, 0x6002) AM_DEVREADWRITE("oki2", okim6295_r, okim6295_w)
AM_RANGE(0x8000, 0xffff) AM_ROMBANK("bank1")
ADDRESS_MAP_END
ADDRESS_MAP_START( seibu3_sound_map, ADDRESS_SPACE_PROGRAM, 8 )
AM_RANGE(0x0000, 0x1fff) AM_ROM
AM_RANGE(0x2000, 0x27ff) AM_RAM
AM_RANGE(0x4000, 0x4000) AM_WRITE(seibu_pending_w)
AM_RANGE(0x4001, 0x4001) AM_WRITE(seibu_irq_clear_w)
AM_RANGE(0x4002, 0x4002) AM_WRITE(seibu_rst10_ack_w)
AM_RANGE(0x4003, 0x4003) AM_WRITE(seibu_rst18_ack_w)
AM_RANGE(0x4007, 0x4007) AM_WRITE(seibu_bank_w)
AM_RANGE(0x4008, 0x4009) AM_DEVREADWRITE("ym1", ym2203_r, ym2203_w)
AM_RANGE(0x4010, 0x4011) AM_READ(seibu_soundlatch_r)
AM_RANGE(0x4012, 0x4012) AM_READ(seibu_main_data_pending_r)
AM_RANGE(0x4013, 0x4013) AM_READ_PORT("COIN")
AM_RANGE(0x4018, 0x4019) AM_WRITE(seibu_main_data_w)
AM_RANGE(0x401b, 0x401b) AM_WRITE(seibu_coin_w)
AM_RANGE(0x6008, 0x6009) AM_DEVREADWRITE("ym2", ym2203_r, ym2203_w)
AM_RANGE(0x8000, 0xffff) AM_ROMBANK("bank1")
ADDRESS_MAP_END
ADDRESS_MAP_START( seibu3_adpcm_sound_map, ADDRESS_SPACE_PROGRAM, 8 )
AM_RANGE(0x0000, 0x1fff) AM_ROM
AM_RANGE(0x2000, 0x27ff) AM_RAM
AM_RANGE(0x4000, 0x4000) AM_WRITE(seibu_pending_w)
AM_RANGE(0x4001, 0x4001) AM_WRITE(seibu_irq_clear_w)
AM_RANGE(0x4002, 0x4002) AM_WRITE(seibu_rst10_ack_w)
AM_RANGE(0x4003, 0x4003) AM_WRITE(seibu_rst18_ack_w)
AM_RANGE(0x4005, 0x4006) AM_DEVWRITE("adpcm1", seibu_adpcm_adr_w)
AM_RANGE(0x4007, 0x4007) AM_WRITE(seibu_bank_w)
AM_RANGE(0x4008, 0x4009) AM_DEVREADWRITE("ym1", ym2203_r, ym2203_w)
AM_RANGE(0x4010, 0x4011) AM_READ(seibu_soundlatch_r)
AM_RANGE(0x4012, 0x4012) AM_READ(seibu_main_data_pending_r)
AM_RANGE(0x4013, 0x4013) AM_READ_PORT("COIN")
AM_RANGE(0x4018, 0x4019) AM_WRITE(seibu_main_data_w)
AM_RANGE(0x401a, 0x401a) AM_DEVWRITE("adpcm1", seibu_adpcm_ctl_w)
AM_RANGE(0x401b, 0x401b) AM_WRITE(seibu_coin_w)
AM_RANGE(0x6005, 0x6006) AM_DEVWRITE("adpcm2", seibu_adpcm_adr_w)
AM_RANGE(0x6008, 0x6009) AM_DEVREADWRITE("ym2", ym2203_r, ym2203_w)
AM_RANGE(0x601a, 0x601a) AM_DEVWRITE("adpcm2", seibu_adpcm_ctl_w)
AM_RANGE(0x8000, 0xffff) AM_ROMBANK("bank1")
ADDRESS_MAP_END