/* * 86Box A hypervisor and IBM PC system emulator that specializes in * running old operating systems and software designed for IBM * PC systems and compatibles from 1981 through fairly recent * system designs based on the PCI bus. * * This file is part of the 86Box distribution. * * OPTi MediaCHIPS 82C929 audio controller emulation. * * * * Authors: Cacodemon345 * Eluan Costa Miranda * * Copyright 2022 Cacodemon345. * Copyright 2020 Eluan Costa Miranda. */ #include #include #include #include #include #include #include #define HAVE_STDARG_H #include <86box/86box.h> #include <86box/device.h> #include <86box/io.h> #include <86box/midi.h> #include <86box/timer.h> #include <86box/nvr.h> #include <86box/pic.h> #include <86box/sound.h> #include <86box/gameport.h> #include <86box/snd_ad1848.h> #include <86box/snd_sb.h> static int optimc_wss_dma[4] = { 0, 0, 1, 3 }; static int optimc_wss_irq[8] = { 5, 7, 9, 10, 11, 12, 14, 15 }; /* W95 only uses 7-10, others may be wrong */ enum optimc_local_flags { OPTIMC_CS4231 = 0x100, OPTIMC_OPL4 = 0x200, /* Unused */ }; typedef struct optimc_t { uint8_t type, fm_type; uint8_t wss_config, reg_enabled; uint16_t cur_addr, cur_wss_addr, cur_mpu401_addr; int cur_irq, cur_dma; int cur_wss_enabled, cur_wss_irq, cur_wss_dma; int cur_mpu401_irq; int cur_mpu401_enabled; void *gameport; uint8_t cur_mode; ad1848_t ad1848; mpu_t *mpu; sb_t *sb; uint8_t regs[6]; } optimc_t, opti_82c929_t; static uint8_t optimc_wss_read(uint16_t addr, void *priv) { optimc_t *optimc = (optimc_t *) priv; if (!(optimc->regs[4] & 0x10) && optimc->cur_mode == 0) return 0xFF; return 4 | (optimc->wss_config & 0x40); } static void optimc_wss_write(uint16_t addr, uint8_t val, void *priv) { optimc_t *optimc = (optimc_t *) priv; if (!(optimc->regs[4] & 0x10) && optimc->cur_mode == 0) return; optimc->wss_config = val; ad1848_setdma(&optimc->ad1848, optimc_wss_dma[val & 3]); ad1848_setirq(&optimc->ad1848, optimc_wss_irq[(val >> 3) & 7]); } static void optimc_get_buffer(int32_t *buffer, int len, void *p) { optimc_t *optimc = (optimc_t *) p; int c; if (optimc->regs[3] & 0x4) return; /* wss part */ ad1848_update(&optimc->ad1848); for (c = 0; c < len * 2; c++) buffer[c] += (optimc->ad1848.buffer[c] / 2); optimc->ad1848.pos = 0; /* sbprov2 part */ sb_get_buffer_sbpro(buffer, len, optimc->sb); } static void optimc_remove_opl(optimc_t *optimc) { io_removehandler(optimc->cur_addr + 0, 0x0004, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_removehandler(optimc->cur_addr + 8, 0x0002, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_removehandler(0x0388, 0x0004, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); } static void optimc_add_opl(optimc_t *optimc) { fm_driver_get(FM_YMF262, &optimc->sb->opl); /* DSP I/O handler is activated in sb_dsp_setaddr */ io_sethandler(optimc->cur_addr + 0, 0x0004, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_sethandler(optimc->cur_addr + 8, 0x0002, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_sethandler(0x0388, 0x0004, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); } static void optimc_reload_opl(optimc_t *optimc) { optimc_remove_opl(optimc); optimc_add_opl(optimc); } static void optimc_reg_write(uint16_t addr, uint8_t val, void *p) { optimc_t *optimc = (optimc_t *) p; uint16_t idx = addr - 0xF8D; uint8_t old = optimc->regs[idx]; static uint8_t reg_enable_phase = 0; if (optimc->reg_enabled) { switch (idx) { case 0: /* MC1 */ { optimc->regs[0] = val; if (val != old) { optimc->cur_mode = optimc->cur_wss_enabled = !!(val & 0x80); io_removehandler(optimc->cur_wss_addr, 0x0004, optimc_wss_read, NULL, NULL, optimc_wss_write, NULL, NULL, optimc); io_removehandler(optimc->cur_wss_addr + 0x0004, 0x0004, ad1848_read, NULL, NULL, ad1848_write, NULL, NULL, &optimc->ad1848); switch ((val >> 4) & 0x3) { case 0: /* WSBase = 0x530 */ { optimc->cur_wss_addr = 0x530; break; } case 1: /* WSBase = 0xE80 */ { optimc->cur_wss_addr = 0xE80; break; } case 2: /* WSBase = 0xF40 */ { optimc->cur_wss_addr = 0xF40; break; } case 3: /* WSBase = 0x604 */ { optimc->cur_wss_addr = 0x604; break; } } io_sethandler(optimc->cur_wss_addr, 0x0004, optimc_wss_read, NULL, NULL, optimc_wss_write, NULL, NULL, optimc); io_sethandler(optimc->cur_wss_addr + 0x0004, 0x0004, ad1848_read, NULL, NULL, ad1848_write, NULL, NULL, &optimc->ad1848); gameport_remap(optimc->gameport, (optimc->regs[0] & 0x1) ? 0x00 : 0x200); } break; } case 1: /* MC2 */ optimc->regs[1] = val; if (old != val) optimc_reload_opl(optimc); break; case 2: /* MC3 */ if (val == optimc->type) break; optimc->regs[2] = val; if (old != val) { io_removehandler(optimc->cur_addr + 4, 0x0002, sb_ct1345_mixer_read, NULL, NULL, sb_ct1345_mixer_write, NULL, NULL, optimc->sb); optimc_remove_opl(optimc); optimc->cur_addr = (val & 0x4) ? 0x240 : 0x220; switch ((val >> 4) & 0x3) { case 0: optimc->cur_dma = 1; break; case 1: optimc->cur_dma = 0; break; case 2: default: optimc->cur_dma = 3; break; } switch ((val >> 6) & 0x3) { case 0: optimc->cur_irq = 7; break; case 1: optimc->cur_irq = 10; break; case 2: default: optimc->cur_irq = 5; break; } sb_dsp_setaddr(&optimc->sb->dsp, optimc->cur_addr); sb_dsp_setirq(&optimc->sb->dsp, optimc->cur_irq); sb_dsp_setdma8(&optimc->sb->dsp, optimc->cur_dma); optimc_add_opl(optimc); io_sethandler(optimc->cur_addr + 4, 0x0002, sb_ct1345_mixer_read, NULL, NULL, sb_ct1345_mixer_write, NULL, NULL, optimc->sb); } break; case 3: /* MC4 */ optimc->regs[3] = val; if ((old & 0x8) != (val & 0x8)) optimc_reload_opl(optimc); break; case 4: /* MC5 */ optimc->regs[4] = val; break; case 5: /* MC6 */ optimc->regs[5] = val; if (old != val) { switch ((val >> 3) & 0x3) { case 0: optimc->cur_mpu401_irq = 9; break; case 1: optimc->cur_mpu401_irq = 10; break; case 2: optimc->cur_mpu401_irq = 5; break; case 3: optimc->cur_mpu401_irq = 7; break; } switch ((val >> 5) & 0x3) { case 0: optimc->cur_mpu401_addr = 0x330; break; case 1: optimc->cur_mpu401_addr = 0x320; break; case 2: optimc->cur_mpu401_addr = 0x310; break; case 3: optimc->cur_mpu401_addr = 0x300; break; } mpu401_change_addr(optimc->mpu, optimc->cur_mpu401_addr); mpu401_setirq(optimc->mpu, optimc->cur_mpu401_irq); } break; } } if (optimc->reg_enabled) optimc->reg_enabled = 0; if (addr == 0xF8F && (val == optimc->type || val == 0x00)) { if (addr == 0xF8F && val == optimc->type && !optimc->reg_enabled) { optimc->reg_enabled = 1; } if (reg_enable_phase) { switch (reg_enable_phase) { case 1: if (val == optimc->type) { reg_enable_phase++; } break; case 2: if (val == 0x00) { reg_enable_phase++; } break; case 3: if (val == optimc->type) { optimc->regs[2] = 0x2; reg_enable_phase = 1; } break; } } else reg_enable_phase = 1; return; } } static uint8_t optimc_reg_read(uint16_t addr, void *p) { optimc_t *optimc = (optimc_t *) p; uint8_t temp = 0xFF; if (optimc->reg_enabled) { switch (addr - 0xF8D) { case 0: /* MC1 */ case 1: /* MC2 */ case 3: /* MC4 */ case 4: /* MC5 */ temp = optimc->regs[addr - 0xF8D]; case 5: /* MC6 (not readable) */ break; case 2: /* MC3 */ temp = (optimc->regs[2] & ~0x3) | 0x2; break; } optimc->reg_enabled = 0; } return temp; } static void * optimc_init(const device_t *info) { optimc_t *optimc = calloc(1, sizeof(optimc_t)); optimc->type = info->local & 0xFF; optimc->cur_wss_addr = 0x530; optimc->cur_mode = 0; optimc->cur_addr = 0x220; optimc->cur_irq = 7; optimc->cur_wss_enabled = 0; optimc->cur_dma = 1; optimc->cur_mpu401_irq = 9; optimc->cur_mpu401_addr = 0x330; optimc->cur_mpu401_enabled = 1; optimc->regs[0] = ((device_get_config_int("gameport")) ? 0x01 : 0x00); optimc->regs[1] = 0x03; optimc->regs[2] = 0x00; optimc->regs[3] = 0x00; optimc->regs[4] = 0x2F; optimc->regs[5] = 0x83; optimc->gameport = gameport_add(&gameport_device); gameport_remap(optimc->gameport, (optimc->regs[0] & 0x1) ? 0x00 : 0x200); if (info->local & 0x100) ad1848_init(&optimc->ad1848, AD1848_TYPE_CS4231); else ad1848_init(&optimc->ad1848, AD1848_TYPE_DEFAULT); ad1848_setirq(&optimc->ad1848, optimc->cur_wss_irq); ad1848_setdma(&optimc->ad1848, optimc->cur_wss_dma); io_sethandler(0xF8D, 6, optimc_reg_read, NULL, NULL, optimc_reg_write, NULL, NULL, optimc); io_sethandler(optimc->cur_wss_addr, 0x0004, optimc_wss_read, NULL, NULL, optimc_wss_write, NULL, NULL, optimc); io_sethandler(optimc->cur_wss_addr + 0x0004, 0x0004, ad1848_read, NULL, NULL, ad1848_write, NULL, NULL, &optimc->ad1848); optimc->sb = calloc(1, sizeof(sb_t)); optimc->sb->opl_enabled = 1; sb_dsp_init(&optimc->sb->dsp, SBPRO2, SB_SUBTYPE_DEFAULT, optimc); sb_dsp_setaddr(&optimc->sb->dsp, optimc->cur_addr); sb_dsp_setirq(&optimc->sb->dsp, optimc->cur_irq); sb_dsp_setdma8(&optimc->sb->dsp, optimc->cur_dma); sb_ct1345_mixer_reset(optimc->sb); optimc->fm_type = (info->local & OPTIMC_OPL4) ? FM_YMF289B : FM_YMF262; fm_driver_get(optimc->fm_type, &optimc->sb->opl); io_sethandler(optimc->cur_addr + 0, 0x0004, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_sethandler(optimc->cur_addr + 8, 0x0002, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_sethandler(0x0388, 0x0004, optimc->sb->opl.read, NULL, NULL, optimc->sb->opl.write, NULL, NULL, optimc->sb->opl.priv); io_sethandler(optimc->cur_addr + 4, 0x0002, sb_ct1345_mixer_read, NULL, NULL, sb_ct1345_mixer_write, NULL, NULL, optimc->sb); sound_add_handler(optimc_get_buffer, optimc); sound_set_cd_audio_filter(sbpro_filter_cd_audio, optimc->sb); optimc->mpu = (mpu_t *) malloc(sizeof(mpu_t)); memset(optimc->mpu, 0, sizeof(mpu_t)); mpu401_init(optimc->mpu, optimc->cur_mpu401_addr, optimc->cur_mpu401_irq, M_UART, device_get_config_int("receive_input401")); if (device_get_config_int("receive_input")) midi_in_handler(1, sb_dsp_input_msg, sb_dsp_input_sysex, &optimc->sb->dsp); return optimc; } static void optimc_close(void *p) { optimc_t* optimc = (optimc_t*)p; sb_close(optimc->sb); free(optimc->mpu); free(p); } static void optimc_speed_changed(void *p) { optimc_t *optimc = (optimc_t *) p; ad1848_speed_changed(&optimc->ad1848); sb_speed_changed(optimc->sb); } static const device_config_t acermagic_s20_config[] = { // clang-format off { .name = "gameport", .description = "Gameport", .type = CONFIG_BINARY, .default_int = 0 }, { .name = "receive_input", .description = "Receive input (SB MIDI)", .type = CONFIG_BINARY, .default_string = "", .default_int = 1 }, { .name = "receive_input401", .description = "Receive input (MPU-401)", .type = CONFIG_BINARY, .default_string = "", .default_int = 0 }, { .name = "", .description = "", .type = CONFIG_END } // clang-format on }; const device_t acermagic_s20_device = { .name = "AcerMagic S20", .internal_name = "acermagic_s20", .flags = DEVICE_ISA | DEVICE_AT, .local = 0xE3 | 0x100, .init = optimc_init, .close = optimc_close, .reset = NULL, { .available = NULL }, .speed_changed = optimc_speed_changed, .force_redraw = NULL, .config = acermagic_s20_config };