/*************************************************************************** 2610intf.c The YM2610 emulator supports up to 2 chips. Each chip has the following connections: - Status Read / Control Write A - Port Read / Data Write A - Control Write B - Data Write B ***************************************************************************/ #include // for memset #include // for free #include // for NULL #include "mamedef.h" //#include "sndintrf.h" //#include "streams.h" #include "2610intf.h" #include "fm.h" #ifdef ENABLE_ALL_CORES #define EC_MAME 0x01 // AY8910 core from MAME #endif #define EC_EMU2149 0x00 typedef struct _ym2610_state ym2610_state; struct _ym2610_state { //sound_stream * stream; //emu_timer * timer[2]; void * chip; void * psg; int AY_EMU_CORE; //const ym2610_interface *intf; //const device_config *device; }; #define CHTYPE_YM2610 0x22 /*INLINE ym2610_state *get_safe_token(const device_config *device) { assert(device != NULL); assert(device->token != NULL); assert(device->type == SOUND); assert(sound_get_type(device) == SOUND_YM2610 || sound_get_type(device) == SOUND_YM2610B); return (ym2610_state *)device->token; }*/ static void psg_set_clock(void *param, int clock) { ym2610_state *info = (ym2610_state *)param; if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: ay8910_set_clock_ym(info->psg, clock); break; #endif case EC_EMU2149: PSG_set_clock((PSG*)info->psg, clock); break; } } } static void psg_write(void *param, int address, int data) { ym2610_state *info = (ym2610_state *)param; if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: ay8910_write_ym(info->psg, address, data); break; #endif case EC_EMU2149: PSG_writeIO((PSG*)info->psg, address, data); break; } } } static int psg_read(void *param) { ym2610_state *info = (ym2610_state *)param; if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: return ay8910_read_ym(info->psg); #endif case EC_EMU2149: return PSG_readIO((PSG*)info->psg); } } return 0x00; } static void psg_reset(void *param) { ym2610_state *info = (ym2610_state *)param; if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: ay8910_reset_ym(info->psg); break; #endif case EC_EMU2149: PSG_reset((PSG*)info->psg); break; } } } static const ssg_callbacks psgintf = { psg_set_clock, psg_write, psg_read, psg_reset }; /*------------------------- TM2610 -------------------------------*/ /* IRQ Handler */ /*static void IRQHandler(void *param,int irq) { ym2610_state *info = (ym2610_state *)param; //if(info->intf->handler) info->intf->handler(info->device, irq); //if(info->intf->handler) info->intf->handler(irq); }*/ /* Timer overflow callback from timer.c */ /*static TIMER_CALLBACK( timer_callback_0 ) { ym2610_state *info = (ym2610_state *)ptr; ym2610_timer_over(info->chip,0); } static TIMER_CALLBACK( timer_callback_1 ) { ym2610_state *info = (ym2610_state *)ptr; ym2610_timer_over(info->chip,1); }*/ /*static void timer_handler(void *param,int c,int count,int clock) { ym2610_state *info = (ym2610_state *)param; if( count == 0 ) { // Reset FM Timer //timer_enable(info->timer[c], 0); } else { // Start FM Timer //attotime period = attotime_mul(ATTOTIME_IN_HZ(clock), count); //if (!timer_enable(info->timer[c], 1)) // timer_adjust_oneshot(info->timer[c], period, 0); } }*/ static stream_sample_t* DUMMYBUF[0x02] = {NULL, NULL}; /* update request from fm.c */ void ym2610_update_request(void *param) { ym2610_state *info = (ym2610_state *)param; //stream_update(info->stream); ym2610b_update_one(info->chip, DUMMYBUF, 0); // Not necessary. //if (info->psg != NULL) // ay8910_update_one(info->psg, DUMMYBUF, 0); } //static STREAM_UPDATE( ym2610_stream_update ) void ym2610_stream_update(void *param, stream_sample_t **outputs, int samples) { ym2610_state *info = (ym2610_state *)param; ym2610_update_one(info->chip, outputs, samples); } //static STREAM_UPDATE( ym2610b_stream_update ) void ym2610b_stream_update(void *param, stream_sample_t **outputs, int samples) { ym2610_state *info = (ym2610_state *)param; ym2610b_update_one(info->chip, outputs, samples); } void ym2610_stream_update_ay(void *param, stream_sample_t **outputs, int samples) { ym2610_state *info = (ym2610_state *)param; if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: ay8910_update_one(info->psg, outputs, samples); break; #endif case EC_EMU2149: PSG_calc_stereo((PSG*)info->psg, outputs, samples); break; } } else { memset(outputs[0], 0x00, samples * sizeof(stream_sample_t)); memset(outputs[1], 0x00, samples * sizeof(stream_sample_t)); } } //static STATE_POSTLOAD( ym2610_intf_postload ) /*static void ym2610_intf_postload(UINT8 ChipID) { //ym2610_state *info = (ym2610_state *)param; ym2610_state *info = &YM2610Data[ChipID]; ym2610_postload(info->chip); }*/ //static DEVICE_START( ym2610 ) int device_start_ym2610(void **_info, int AY_EMU_CORE, int clock, UINT8 AYDisable, int* AYrate, int CHIP_SAMPLING_MODE, int CHIP_SAMPLE_RATE) { // clock bit 31: 0 - YM2610 // 1 - YM2610B //static const ym2610_interface generic_2610 = { 0 }; #ifdef ENABLE_ALL_CORES static const ay8910_interface generic_ay8910 = { AY8910_LEGACY_OUTPUT | AY8910_SINGLE_OUTPUT, AY8910_DEFAULT_LOADS //DEVCB_NULL, DEVCB_NULL, DEVCB_NULL, DEVCB_NULL }; #endif //const ym2610_interface *intf = device->static_config ? (const ym2610_interface *)device->static_config : &generic_2610; //const ym2610_interface *intf = &generic_2610; int rate; int ay_clock; //void *pcmbufa,*pcmbufb; //int pcmsizea,pcmsizeb; //ym2610_state *info = get_safe_token(device); ym2610_state *info; //astring *name = astring_alloc(); //sound_type type = sound_get_type(device); unsigned char ChipType; #ifdef ENABLE_ALL_CORES if (AY_EMU_CORE >= 0x02) AY_EMU_CORE = EC_EMU2149; #else AY_EMU_CORE = EC_EMU2149; #endif info = (ym2610_state *) calloc(1, sizeof(ym2610_state)); *_info = (void *) info; info->AY_EMU_CORE = AY_EMU_CORE; ChipType = (clock & 0x80000000) ? 0x01 : 0x00; clock &= 0x7FFFFFFF; rate = clock/72; if ((CHIP_SAMPLING_MODE == 0x01 && rate < CHIP_SAMPLE_RATE) || CHIP_SAMPLING_MODE == 0x02) rate = CHIP_SAMPLE_RATE; //info->intf = intf; //info->device = device; //info->psg = ay8910_start_ym(NULL, sound_get_type(device), device, device->clock, &generic_ay8910); if (! AYDisable) { ay_clock = clock / 4; *AYrate = ay_clock / 8; switch(AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: info->psg = ay8910_start_ym(NULL, CHTYPE_YM2610 + ChipType, ay_clock, &generic_ay8910); break; #endif case EC_EMU2149: info->psg = PSG_new(ay_clock, *AYrate); if (info->psg == NULL) return 0; PSG_setVolumeMode((PSG*)info->psg, 1); // YM2149 volume mode break; } } else { info->psg = NULL; *AYrate = 0; } //assert_always(info->psg != NULL, "Error creating YM2610/AY8910 chip"); /* Timer Handler set */ //info->timer[0] = timer_alloc(device->machine, timer_callback_0, info); //info->timer[1] = timer_alloc(device->machine, timer_callback_1, info); /* stream system initialize */ //info->stream = stream_create(device,0,2,rate,info,(type == SOUND_YM2610) ? ym2610_stream_update : ym2610b_stream_update); /* setup adpcm buffers */ //pcmbufa = device->region; //pcmsizea = device->regionbytes; //astring_printf(name, "%s.deltat", device->tag); //pcmbufb = (void *)(memory_region(device->machine, astring_c(name))); //pcmsizeb = memory_region_length(device->machine, astring_c(name)); //astring_free(name); /*if (pcmbufb == NULL || pcmsizeb == 0) { pcmbufb = pcmbufa; pcmsizeb = pcmsizea; }*/ /**** initialize YM2610 ****/ //info->chip = ym2610_init(info,device,device->clock,rate, // pcmbufa,pcmsizea,pcmbufb,pcmsizeb, // timer_handler,IRQHandler,&psgintf); info->chip = ym2610_init(info, clock & 0x7FFFFFFF, rate, NULL, NULL, &psgintf); //assert_always(info->chip != NULL, "Error creating YM2610 chip"); //state_save_register_postload(device->machine, ym2610_intf_postload, info); return rate; } //static DEVICE_STOP( ym2610 ) void device_stop_ym2610(void *_info) { //ym2610_state *info = get_safe_token(device); ym2610_state* info = (ym2610_state *)_info; ym2610_shutdown(info->chip); if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: ay8910_stop_ym(info->psg); break; #endif case EC_EMU2149: PSG_delete((PSG*)info->psg); break; } info->psg = NULL; } free(info); } //static DEVICE_RESET( ym2610 ) void device_reset_ym2610(void *_info) { //ym2610_state *info = get_safe_token(device); ym2610_state* info = (ym2610_state *)_info; ym2610_reset_chip(info->chip); // also resets the AY clock //psg_reset(info); // already done as a callback in ym2610_reset_chip } //READ8_DEVICE_HANDLER( ym2610_r ) UINT8 ym2610_r(void *_info, offs_t offset) { //ym2610_state *info = get_safe_token(device); ym2610_state* info = (ym2610_state *)_info; return ym2610_read(info->chip, offset & 3); } //WRITE8_DEVICE_HANDLER( ym2610_w ) void ym2610_w(void *_info, offs_t offset, UINT8 data) { //ym2610_state *info = get_safe_token(device); ym2610_state* info = (ym2610_state *)_info; ym2610_write(info->chip, offset & 3, data); } //READ8_DEVICE_HANDLER( ym2610_status_port_a_r ) UINT8 ym2610_status_port_a_r(void *info, offs_t offset) { return ym2610_r(info, 0); } //READ8_DEVICE_HANDLER( ym2610_status_port_b_r ) UINT8 ym2610_status_port_b_r(void *info, offs_t offset) { return ym2610_r(info, 2); } //READ8_DEVICE_HANDLER( ym2610_read_port_r ) UINT8 ym2610_read_port_r(void *info, offs_t offset) { return ym2610_r(info, 1); } //WRITE8_DEVICE_HANDLER( ym2610_control_port_a_w ) void ym2610_control_port_a_w(void *info, offs_t offset, UINT8 data) { ym2610_w(info, 0, data); } //WRITE8_DEVICE_HANDLER( ym2610_control_port_b_w ) void ym2610_control_port_b_w(void *info, offs_t offset, UINT8 data) { ym2610_w(info, 2, data); } //WRITE8_DEVICE_HANDLER( ym2610_data_port_a_w ) void ym2610_data_port_a_w(void *info, offs_t offset, UINT8 data) { ym2610_w(info, 1, data); } //WRITE8_DEVICE_HANDLER( ym2610_data_port_b_w ) void ym2610_data_port_b_w(void *info, offs_t offset, UINT8 data) { ym2610_w(info, 3, data); } void ym2610_write_data_pcmrom(void *_info, UINT8 rom_id, offs_t ROMSize, offs_t DataStart, offs_t DataLength, const UINT8* ROMData) { ym2610_state* info = (ym2610_state *)_info; ym2610_write_pcmrom(info->chip, rom_id, ROMSize, DataStart, DataLength, ROMData); } void ym2610_set_mute_mask(void *_info, UINT32 MuteMaskFM, UINT32 MuteMaskAY) { ym2610_state* info = (ym2610_state *)_info; ym2610_set_mutemask(info->chip, MuteMaskFM); if (info->psg != NULL) { switch(info->AY_EMU_CORE) { #ifdef ENABLE_ALL_CORES case EC_MAME: ay8910_set_mute_mask_ym(info->psg, MuteMaskAY); break; #endif case EC_EMU2149: PSG_setMask((PSG*)info->psg, MuteMaskAY); break; } } } /************************************************************************** * Generic get_info **************************************************************************/ /*DEVICE_GET_INFO( ym2610 ) { switch (state) { // --- the following bits of info are returned as 64-bit signed integers --- case DEVINFO_INT_TOKEN_BYTES: info->i = sizeof(ym2610_state); break; // --- the following bits of info are returned as pointers to data or functions --- case DEVINFO_FCT_START: info->start = DEVICE_START_NAME( ym2610 ); break; case DEVINFO_FCT_STOP: info->stop = DEVICE_STOP_NAME( ym2610 ); break; case DEVINFO_FCT_RESET: info->reset = DEVICE_RESET_NAME( ym2610 ); break; // --- the following bits of info are returned as NULL-terminated strings --- case DEVINFO_STR_NAME: strcpy(info->s, "YM2610"); break; case DEVINFO_STR_FAMILY: strcpy(info->s, "Yamaha FM"); break; case DEVINFO_STR_VERSION: strcpy(info->s, "1.0"); break; case DEVINFO_STR_SOURCE_FILE: strcpy(info->s, __FILE__); break; case DEVINFO_STR_CREDITS: strcpy(info->s, "Copyright Nicola Salmoria and the MAME Team"); break; } }*/ /*DEVICE_GET_INFO( ym2610b ) { switch (state) { // --- the following bits of info are returned as NULL-terminated strings --- case DEVINFO_STR_NAME: strcpy(info->s, "YM2610B"); break; default: DEVICE_GET_INFO_CALL(ym2610); break; } }*/