/* * Copyright (C) 2022-2023 nukeykt * * This file is part of YMF276-LLE. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * YMF276/YM3438 emulator. * Thanks: * John McMaster (siliconpr0n.org): * Yamaha YM3438 & YM2610 decap and die shot. * org, andkorzh, HardWareMan (emu-russia): * help & support, YMF276 and YM2612 decap. * */ // YMF276/YM3438 core #include #include #include "fmopn2.h" enum { eg_state_attack = 0, eg_state_decay, eg_state_sustain, eg_state_release }; static const int fm_algorithm[4][6][8] = { { { 1, 1, 1, 1, 1, 1, 1, 1 }, /* OP1_0 */ { 1, 1, 1, 1, 1, 1, 1, 1 }, /* OP1_1 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* OP2 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* Last operator */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* Last operator */ { 0, 0, 0, 0, 0, 0, 0, 1 } /* Out */ }, { { 0, 1, 0, 0, 0, 1, 0, 0 }, /* OP1_0 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* OP1_1 */ { 1, 1, 1, 0, 0, 0, 0, 0 }, /* OP2 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* Last operator */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* Last operator */ { 0, 0, 0, 0, 0, 1, 1, 1 } /* Out */ }, { { 0, 0, 0, 0, 0, 0, 0, 0 }, /* OP1_0 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* OP1_1 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* OP2 */ { 1, 0, 0, 1, 1, 1, 1, 0 }, /* Last operator */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* Last operator */ { 0, 0, 0, 0, 1, 1, 1, 1 } /* Out */ }, { { 0, 0, 1, 0, 0, 1, 0, 0 }, /* OP1_0 */ { 0, 0, 0, 0, 0, 0, 0, 0 }, /* OP1_1 */ { 0, 0, 0, 1, 0, 0, 0, 0 }, /* OP2 */ { 1, 1, 0, 1, 1, 0, 0, 0 }, /* Last operator */ { 0, 0, 1, 0, 0, 0, 0, 0 }, /* Last operator */ { 1, 1, 1, 1, 1, 1, 1, 1 } /* Out */ } }; void FMOPN2_ClockPhase1(fmopn2_t *chip); void FMOPN2_Prescaler(fmopn2_t *chip) { if (!chip->pinput.phi) { int ic_check; chip->ic_latch[0] = chip->ic_latch[1] << 1; chip->ic_latch[0] |= chip->pinput.ic; ic_check = (chip->ic_latch[1] & 0x800) == 0 && chip->pinput.ic; chip->prescaler_latch[0] = chip->prescaler_latch[1] << 1; chip->prescaler_latch[0] |= !ic_check && (chip->prescaler_latch[1] & 0x1f) == 0; chip->ic_check_latch[0] = chip->ic_check_latch[1] << 1; chip->ic_check_latch[0] |= ic_check; chip->phi1_latch[0] = (chip->prescaler_latch[1] & 0x21) != 0; chip->phi2_latch[0] = (chip->prescaler_latch[1] & 0xc) != 0; if (!(chip->flags & fmopn2_flags_ym3438)) { chip->dphi1_latch[0] = (chip->prescaler_latch[1] & 0x9) != 0; chip->dphi2_latch[0] = (chip->prescaler_latch[1] & 0x24) != 0; chip->dphi1_latch[2] = chip->dphi1_latch[1]; chip->dphi2_latch[2] = chip->dphi2_latch[1]; } } else { chip->ic_latch[1] = chip->ic_latch[0] & 0xfff; chip->ic_check_latch[1] = chip->ic_check_latch[0] & 0xf; chip->prescaler_latch[1] = chip->prescaler_latch[0] & 0x3f; chip->phi1_latch[1] = chip->phi1_latch[0] & 0x1; chip->phi2_latch[1] = chip->phi2_latch[0] & 0x1; if (!(chip->flags & fmopn2_flags_ym3438)) { chip->dphi1_latch[1] = chip->dphi1_latch[0]; chip->dphi2_latch[1] = chip->dphi2_latch[0]; chip->dphi1_latch[3] = chip->dphi1_latch[2]; } } chip->fsm_reset = (chip->ic_check_latch[1] & 16) != 0; if (!(chip->flags & fmopn2_flags_ym3438)) { chip->dclk1 = chip->dphi1_latch[1] && !chip->dphi1_latch[2]; chip->dclk2 = chip->dphi2_latch[1] && !chip->dphi2_latch[2]; } } void FMOPN2_YMF276DAC(fmopn2_t *chip) { chip->fsm_load_l = (chip->fsm_shifter_ctrl[1] & 0x20) != 0 && chip->i_phi2; chip->fsm_load_r = (chip->fsm_shifter_ctrl[1] & 0x20000) != 0 && chip->i_phi2; if (chip->dclk1) { chip->dac_shifter[0] = chip->dac_shifter[1] << 1; if (chip->fsm_load_l) chip->dac_shifter[0] |= (chip->ch_accm_l[1] >> 1); if (chip->fsm_load_r) chip->dac_shifter[0] |= (chip->ch_accm_r[1] >> 1); } if (chip->dclk2) { chip->dac_shifter[1] = chip->dac_shifter[0]; } if (!chip->pinput.phi) { chip->fsm_lro_l2[0] = chip->fsm_lro_l2[1] << 1; if (chip->fsm_lro_l[1] & 2) chip->fsm_lro_l2[0] |= 1; chip->fsm_wco_l2[0] = chip->fsm_wco_l2[1] << 1; if (chip->fsm_wco_l[1] & 2) chip->fsm_wco_l2[0] |= 1; chip->dac_so_l[0] = chip->dac_so_l[1] << 1; if (chip->dac_shifter[1] & 0x8000) chip->dac_so_l[0] |= 1; } else { chip->fsm_lro_l2[1] = chip->fsm_lro_l2[0]; chip->fsm_wco_l2[1] = chip->fsm_wco_l2[0]; chip->dac_so_l[1] = chip->dac_so_l[0]; } chip->o_bco = chip->dphi1_latch[2] || chip->dphi1_latch[3]; chip->o_wco = (chip->fsm_wco_l2[1] & 32) != 0; chip->o_lro = (chip->fsm_lro_l2[1] & 32) != 0; chip->o_so = (chip->dac_so_l[1] & 4) != 0; } void FMOPN2_HandleIO(fmopn2_t *chip) { int write_data = chip->input.cs && chip->input.wr && (chip->input.address & 1) == 1 && !chip->input.ic; int write_addr = (chip->input.cs && chip->input.wr && (chip->input.address & 1) == 0) || chip->input.ic; int read_enable = chip->input.cs && chip->input.rd && !chip->input.ic; int io_dir = chip->input.cs && chip->input.rd && !chip->input.ic; int data_enable = !io_dir && !chip->input.ic; if (chip->input.cs && chip->input.wr) { chip->data_latch = chip->input.data; chip->bank_latch = (chip->input.address >> 1) & 1; } if (write_addr) chip->write_addr_trig = 1; if (write_data) chip->write_data_trig = 1; if (!read_enable) { chip->status_timer_a_dlatch = chip->timer_a_status[1]; chip->status_timer_b_dlatch = chip->timer_b_status[1]; } } int FMOPN2_GetBus(fmopn2_t *chip) { int data = 0; int io_dir = chip->input.cs && chip->input.rd && !chip->input.ic; int data_enable = !io_dir && !chip->input.ic; if (data_enable) data = chip->data_latch; if (chip->io_ic_latch[1]) data = 0; return data; } int FMOPN2_ReadTest(fmopn2_t *chip) { if (chip->mode_test_2c[1] & 128) return chip->fsm_sel23; return 0; // FIXME: high impedance } int FMOPN2_ReadStatus(fmopn2_t *chip) { int io_dir = chip->input.cs && chip->input.rd && !chip->input.ic; int read_enable = chip->input.cs && chip->input.rd && !chip->input.ic && (chip->input.address & 3) == 0; int status; int testdata = 0; if (!io_dir) return 0; if (!read_enable) { return 0; // FIXME: floating bus } if (chip->mode_test_21[1] & 64) { testdata |= (chip->pg_debug[1] & 1) << 15; if (chip->mode_test_21[1] & 1) testdata |= ((chip->eg_debug[1] >> 9) & 1) << 14; else testdata |= (chip->eg_incsh_nonzero[1] & 1) << 14; if (chip->mode_test_2c[1] & 16) testdata |= chip->ch_out_debug[1] & 0x1ff; else testdata |= chip->op_output[1] & 0x3fff; if (chip->mode_test_21[1] & 128) status = testdata & 255; else status = testdata >> 8; } else { status = (chip->busy_latch[1] << 7) | (chip->status_timer_b_dlatch << 1) | chip->status_timer_a_dlatch; } return status; } void FMOPN2_FSM1(fmopn2_t *chip) { int i; int connect = 0; int reset = chip->input.i_fsm_reset; chip->fsm_cnt1[0] = chip->fsm_cnt1[1] + 1; if (reset || (chip->fsm_cnt1[1] & 2) != 0) chip->fsm_cnt1[0] = 0; chip->fsm_cnt2[0] = chip->fsm_cnt2[1]; if ((chip->fsm_cnt1[1] & 2) != 0) chip->fsm_cnt2[0]++; if (reset) chip->fsm_cnt2[0] = 0; if (!(chip->flags & fmopn2_flags_ym3438)) { int cnt_comb = (chip->fsm_cnt2[1] << 2) | chip->fsm_cnt1[1]; chip->fsm_clock_eg_l = cnt_comb == 30; chip->fsm_op1_sel_l = cnt_comb == 6 || cnt_comb == 8 || cnt_comb == 9 || cnt_comb == 10 || cnt_comb == 12 || cnt_comb == 13; chip->fsm_sel1_l = cnt_comb == 0; chip->fsm_sel2_l = cnt_comb == 1; chip->fsm_sel23_l = cnt_comb == 29; chip->fsm_ch3_sel_l = cnt_comb == 1 || cnt_comb == 9 || cnt_comb == 17 || cnt_comb == 25; chip->fsm_dac_load_l = cnt_comb == 30 || cnt_comb == 4 || cnt_comb == 9 || cnt_comb == 14 || cnt_comb == 20 || cnt_comb == 9; chip->fsm_dac_out_sel_l = cnt_comb == 14 || cnt_comb == 16 || cnt_comb == 17 || cnt_comb == 18 || cnt_comb == 20 || cnt_comb == 21 || cnt_comb == 22 || cnt_comb == 24 || cnt_comb == 25 || cnt_comb == 26 || cnt_comb == 28 || cnt_comb == 29; chip->fsm_dac_ch6_l = cnt_comb == 4 || cnt_comb == 5 || cnt_comb == 6 || cnt_comb == 8; chip->fsm_wco_l[0] = (chip->fsm_wco_l[1] << 1) | ((cnt_comb & 8) == 0); chip->fsm_lro_l[0] = chip->fsm_lro_l[1] << 1; if (((cnt_comb >> 3) ^ (cnt_comb >> 4)) & 1) chip->fsm_lro_l[0] |= 1; } } void FMOPN2_FSM2(fmopn2_t *chip) { int i, connect = 0; int cnt_comb; chip->fsm_cnt1[1] = chip->fsm_cnt1[0] & 0x3; chip->fsm_cnt2[1] = chip->fsm_cnt2[0] & 0x7; cnt_comb = (chip->fsm_cnt2[1] << 2) | chip->fsm_cnt1[1]; if (!(chip->flags & fmopn2_flags_ym3438)) { chip->fsm_clock_eg = chip->fsm_clock_eg_l; chip->fsm_op4_sel = cnt_comb == 0 || cnt_comb == 1 || cnt_comb == 2 || cnt_comb == 4 || cnt_comb == 5 || cnt_comb == 6; chip->fsm_op1_sel = chip->fsm_op1_sel_l; chip->fsm_op3_sel = cnt_comb == 16 || cnt_comb == 17 || cnt_comb == 18 || cnt_comb == 20 || cnt_comb == 21 || cnt_comb == 22; chip->fsm_op2_sel = cnt_comb == 24 || cnt_comb == 25 || cnt_comb == 26 || cnt_comb == 28 || cnt_comb == 29 || cnt_comb == 30; chip->fsm_sel2 = chip->fsm_sel2_l; chip->fsm_sel23 = chip->fsm_sel23_l; chip->fsm_ch3_sel = chip->fsm_ch3_sel_l; chip->fsm_dac_load = chip->fsm_dac_load_l; chip->fsm_dac_out_sel = chip->fsm_dac_out_sel_l; chip->fsm_dac_ch6 = chip->fsm_dac_ch6_l; chip->fsm_clock_timers = chip->fsm_sel2_l; chip->fsm_clock_timers1 = chip->fsm_sel1_l; chip->fsm_wco_l[1] = chip->fsm_wco_l[0]; chip->fsm_lro_l[1] = chip->fsm_lro_l[0]; } else { chip->fsm_clock_eg = cnt_comb == 0; chip->fsm_op4_sel = cnt_comb == 0 || cnt_comb == 1 || cnt_comb == 2 || cnt_comb == 4 || cnt_comb == 5 || cnt_comb == 6; chip->fsm_op1_sel = cnt_comb == 8 || cnt_comb == 9 || cnt_comb == 10 || cnt_comb == 12 || cnt_comb == 13 || cnt_comb == 14; chip->fsm_op3_sel = cnt_comb == 16 || cnt_comb == 17 || cnt_comb == 18 || cnt_comb == 20 || cnt_comb == 21 || cnt_comb == 22; chip->fsm_op2_sel = cnt_comb == 24 || cnt_comb == 25 || cnt_comb == 26 || cnt_comb == 28 || cnt_comb == 29 || cnt_comb == 30; chip->fsm_sel2 = cnt_comb == 2; chip->fsm_sel23 = cnt_comb == 30; chip->fsm_ch3_sel = cnt_comb == 2 || cnt_comb == 10 || cnt_comb == 18 || cnt_comb == 26; chip->fsm_dac_load = cnt_comb == 0 || cnt_comb == 5 || cnt_comb == 10 || cnt_comb == 16 || cnt_comb == 21 || cnt_comb == 26; chip->fsm_dac_out_sel = cnt_comb == 16 || cnt_comb == 17 || cnt_comb == 18 || cnt_comb == 20 || cnt_comb == 21 || cnt_comb == 22 || cnt_comb == 24 || cnt_comb == 25 || cnt_comb == 26 || cnt_comb == 28 || cnt_comb == 29 || cnt_comb == 30; chip->fsm_dac_ch6 = cnt_comb == 5 || cnt_comb == 6 || cnt_comb == 8 || cnt_comb == 9; chip->fsm_clock_timers = cnt_comb == 2; chip->fsm_clock_timers1 = cnt_comb == 1; } for (i = 0; i < 3; i++) connect |= ((chip->chan_connect[i][1] >> 5) & 1) << i; chip->alg_mod_op1_0 = 0; chip->alg_mod_op1_1 = 0; chip->alg_mod_op2 = 0; chip->alg_mod_prev_0 = 0; chip->alg_mod_prev_1 = 0; chip->alg_output = 0; if (chip->fsm_op2_sel) { chip->alg_mod_op1_0 |= fm_algorithm[0][0][connect]; chip->alg_mod_op1_1 |= fm_algorithm[0][1][connect]; chip->alg_mod_op2 |= fm_algorithm[0][2][connect]; chip->alg_mod_prev_0 |= fm_algorithm[0][3][connect]; chip->alg_mod_prev_1 |= fm_algorithm[0][4][connect]; chip->alg_output |= fm_algorithm[2][5][connect]; } if (chip->fsm_op4_sel) { chip->alg_mod_op1_0 |= fm_algorithm[1][0][connect]; chip->alg_mod_op1_1 |= fm_algorithm[1][1][connect]; chip->alg_mod_op2 |= fm_algorithm[1][2][connect]; chip->alg_mod_prev_0 |= fm_algorithm[1][3][connect]; chip->alg_mod_prev_1 |= fm_algorithm[1][4][connect]; chip->alg_output |= fm_algorithm[3][5][connect]; } if (chip->fsm_op1_sel) { chip->alg_mod_op1_0 |= fm_algorithm[2][0][connect]; chip->alg_mod_op1_1 |= fm_algorithm[2][1][connect]; chip->alg_mod_op2 |= fm_algorithm[2][2][connect]; chip->alg_mod_prev_0 |= fm_algorithm[2][3][connect]; chip->alg_mod_prev_1 |= fm_algorithm[2][4][connect]; chip->alg_output |= fm_algorithm[0][5][connect]; } if (chip->fsm_op3_sel) { chip->alg_mod_op1_0 |= fm_algorithm[3][0][connect]; chip->alg_mod_op1_1 |= fm_algorithm[3][1][connect]; chip->alg_mod_op2 |= fm_algorithm[3][2][connect]; chip->alg_mod_prev_0 |= fm_algorithm[3][3][connect]; chip->alg_mod_prev_1 |= fm_algorithm[3][4][connect]; chip->alg_output |= fm_algorithm[1][5][connect]; } } void FMOPN2_HandleIO1(fmopn2_t *chip) { int write_data_en = !chip->write_data_sr[1] && chip->write_data_dlatch; int write_addr_en = !chip->write_addr_sr[1] && chip->write_addr_dlatch; int busy_cnt = chip->busy_cnt[1] + chip->busy_latch[1]; int busy_of = (busy_cnt >> 5) & 1; chip->write_addr_trig_sync = chip->write_addr_trig; chip->write_data_trig_sync = chip->write_data_trig; chip->write_addr_sr[0] = chip->write_addr_dlatch; chip->write_data_sr[0] = chip->write_data_dlatch; chip->busy_latch[0] = write_data_en || (chip->busy_latch[1] && !(chip->input.ic || busy_of)); if (chip->input.ic) busy_cnt = 0; chip->busy_cnt[0] = busy_cnt & 31; chip->io_ic_latch[0] = chip->input.ic; } void FMOPN2_HandleIO2(fmopn2_t *chip) { chip->write_addr_dlatch = chip->write_addr_trig_sync; if (chip->write_addr_dlatch) chip->write_addr_trig = 0; chip->write_data_dlatch = chip->write_data_trig_sync; if (chip->write_data_dlatch) chip->write_data_trig = 0; chip->write_addr_sr[1] = chip->write_addr_sr[0] & 1; chip->write_data_sr[1] = chip->write_data_sr[0] & 1; chip->busy_cnt[1] = chip->busy_cnt[0] & 31; chip->busy_latch[1] = chip->busy_latch[0] & 1; chip->io_ic_latch[1] = chip->io_ic_latch[0] & 1; } void FMOPN2_DoShiftRegisters(fmopn2_t *chip, int sel) { int i, j; int to = sel; int from = sel ^ 1; int rot = sel == 0 ? 1 : 0; #define SLOT_ROTATE(x) rot ? ((x << 1) | ((x >> 11) & 1)) : x #define CH_ROTATE(x) rot ? ((x << 1) | ((x >> 5) & 1)) : x // slot registers for (i = 0; i < 2; i++) { // multi for (j = 0; j < 4; j++) chip->slot_multi[i][j][to] = SLOT_ROTATE(chip->slot_multi[i][j][from]); // dt for (j = 0; j < 3; j++) chip->slot_dt[i][j][to] = SLOT_ROTATE(chip->slot_dt[i][j][from]); // tl for (j = 0; j < 7; j++) chip->slot_tl[i][j][to] = SLOT_ROTATE(chip->slot_tl[i][j][from]); // ar for (j = 0; j < 5; j++) chip->slot_ar[i][j][to] = SLOT_ROTATE(chip->slot_ar[i][j][from]); // ks for (j = 0; j < 2; j++) chip->slot_ks[i][j][to] = SLOT_ROTATE(chip->slot_ks[i][j][from]); // dr for (j = 0; j < 5; j++) chip->slot_dr[i][j][to] = SLOT_ROTATE(chip->slot_dr[i][j][from]); // am for (j = 0; j < 1; j++) chip->slot_am[i][j][to] = SLOT_ROTATE(chip->slot_am[i][j][from]); // sr for (j = 0; j < 5; j++) chip->slot_sr[i][j][to] = SLOT_ROTATE(chip->slot_sr[i][j][from]); // rr for (j = 0; j < 4; j++) chip->slot_rr[i][j][to] = SLOT_ROTATE(chip->slot_rr[i][j][from]); // sl for (j = 0; j < 4; j++) chip->slot_sl[i][j][to] = SLOT_ROTATE(chip->slot_sl[i][j][from]); // ssg eg for (j = 0; j < 4; j++) chip->slot_ssg_eg[i][j][to] = SLOT_ROTATE(chip->slot_ssg_eg[i][j][from]); } // channel registers // fnum for (j = 0; j < 11; j++) chip->chan_fnum[j][to] = CH_ROTATE(chip->chan_fnum[j][from]); // fnum ch3 for (j = 0; j < 11; j++) chip->chan_fnum_ch3[j][to] = CH_ROTATE(chip->chan_fnum_ch3[j][from]); // block for (j = 0; j < 3; j++) chip->chan_block[j][to] = CH_ROTATE(chip->chan_block[j][from]); // block ch3 for (j = 0; j < 3; j++) chip->chan_block_ch3[j][to] = CH_ROTATE(chip->chan_block_ch3[j][from]); // connect for (j = 0; j < 3; j++) chip->chan_connect[j][to] = CH_ROTATE(chip->chan_connect[j][from]); // fb for (j = 0; j < 3; j++) chip->chan_fb[j][to] = CH_ROTATE(chip->chan_fb[j][from]); // pms for (j = 0; j < 3; j++) chip->chan_pms[j][to] = CH_ROTATE(chip->chan_pms[j][from]); // ams for (j = 0; j < 2; j++) chip->chan_ams[j][to] = CH_ROTATE(chip->chan_ams[j][from]); // pan for (j = 0; j < 2; j++) chip->chan_pan[j][to] = CH_ROTATE(chip->chan_pan[j][from]); #undef SLOT_ROTATE #undef CH_ROTATE } void FMOPN2_FMRegisters1(fmopn2_t *chip) { int i, j; int write_data_en = !chip->write_data_sr[1] && chip->write_data_dlatch; int write_addr_en = !chip->write_addr_sr[1] && chip->write_addr_dlatch; int bus = FMOPN2_GetBus(chip); int address = bus | (chip->bank_latch << 8); int fm_write = (bus & 0xf0) != 0; if (write_addr_en) chip->write_fm_address[0] = fm_write; else chip->write_fm_address[0] = chip->write_fm_address[1]; if (chip->input.ic) chip->fm_address[0] = 0; else if (fm_write && write_addr_en) chip->fm_address[0] = address; else chip->fm_address[0] = chip->fm_address[1]; chip->write_fm_data[0] = (chip->write_fm_address[1] && write_data_en) || (chip->write_fm_data[1] && !write_addr_en); if (chip->input.ic) chip->fm_data[0] = 0; else if (chip->write_fm_address[1] && write_data_en) chip->fm_data[0] = bus; else chip->fm_data[0] = chip->fm_data[1]; if (write_addr_en) { chip->write_mode_21[0] = address == 0x21; chip->write_mode_22[0] = address == 0x22; chip->write_mode_24[0] = address == 0x24; chip->write_mode_25[0] = address == 0x25; chip->write_mode_26[0] = address == 0x26; chip->write_mode_27[0] = address == 0x27; chip->write_mode_28[0] = address == 0x28; chip->write_mode_2a[0] = address == 0x2a; chip->write_mode_2b[0] = address == 0x2b; chip->write_mode_2c[0] = address == 0x2c; } else { chip->write_mode_21[0] = chip->write_mode_21[1]; chip->write_mode_22[0] = chip->write_mode_22[1]; chip->write_mode_24[0] = chip->write_mode_24[1]; chip->write_mode_25[0] = chip->write_mode_25[1]; chip->write_mode_26[0] = chip->write_mode_26[1]; chip->write_mode_27[0] = chip->write_mode_27[1]; chip->write_mode_28[0] = chip->write_mode_28[1]; chip->write_mode_2a[0] = chip->write_mode_2a[1]; chip->write_mode_2b[0] = chip->write_mode_2b[1]; chip->write_mode_2c[0] = chip->write_mode_2c[1]; } if (chip->input.ic) { chip->mode_test_21[0] = 0; chip->mode_lfo_en[0] = 0; chip->mode_lfo_freq[0] = 0; chip->mode_timer_a_reg[0] = 0; chip->mode_timer_b_reg[0] = 0; chip->mode_ch3[0] = 0; chip->mode_timer_a_load[0] = 0; chip->mode_timer_a_enable[0] = 0; chip->mode_timer_a_reset[0] = 0; chip->mode_timer_b_load[0] = 0; chip->mode_timer_b_enable[0] = 0; chip->mode_timer_b_reset[0] = 0; chip->mode_kon_operator[0] = 0; chip->mode_kon_channel[0] = 0; chip->mode_dac_data[0] = 0; chip->mode_dac_en[0] = 0; chip->mode_test_2c[0] = 0; // slot registers for (i = 0; i < 2; i++) { // multi for (j = 0; j < 4; j++) chip->slot_multi[i][j][0] &= ~1; // dt for (j = 0; j < 3; j++) chip->slot_dt[i][j][0] &= ~1; // tl for (j = 0; j < 7; j++) chip->slot_tl[i][j][0] &= ~1; // ar for (j = 0; j < 5; j++) chip->slot_ar[i][j][0] &= ~1; // ks for (j = 0; j < 2; j++) chip->slot_ks[i][j][0] &= ~1; // dr for (j = 0; j < 5; j++) chip->slot_dr[i][j][0] &= ~1; // am for (j = 0; j < 1; j++) chip->slot_am[i][j][0] &= ~1; // sr for (j = 0; j < 5; j++) chip->slot_sr[i][j][0] &= ~1; // rr for (j = 0; j < 4; j++) chip->slot_rr[i][j][0] &= ~1; // sl for (j = 0; j < 4; j++) chip->slot_sl[i][j][0] &= ~1; // ssg eg for (j = 0; j < 4; j++) chip->slot_ssg_eg[i][j][0] &= ~1; } // channel registers // fn low for (j = 0; j < 11; j++) chip->chan_fnum[j][0] &= ~1; // fn low ch3 for (j = 0; j < 11; j++) chip->chan_fnum_ch3[j][0] &= ~1; // block fn high for (j = 0; j < 3; j++) chip->chan_block[j][0] &= ~1; // block fn high ch3 for (j = 0; j < 3; j++) chip->chan_block_ch3[j][0] &= ~1; // connect for (j = 0; j < 3; j++) chip->chan_connect[j][0] &= ~1; // fb for (j = 0; j < 3; j++) chip->chan_fb[j][0] &= ~1; // pms for (j = 0; j < 3; j++) chip->chan_pms[j][0] &= ~1; // ams for (j = 0; j < 2; j++) chip->chan_ams[j][0] &= ~1; // pan for (j = 0; j < 2; j++) chip->chan_pan[j][0] &= ~1; chip->chan_a4[0] = 0; chip->chan_ac[0] = 0; } else { if (write_data_en && chip->write_mode_21[1] && !chip->bank_latch) chip->mode_test_21[0] = bus & 255; else chip->mode_test_21[0] = chip->mode_test_21[1]; if (write_data_en && chip->write_mode_22[1] && !chip->bank_latch) { chip->mode_lfo_en[0] = (bus >> 3) & 1; chip->mode_lfo_freq[0] = bus & 7; } else { chip->mode_lfo_en[0] = chip->mode_lfo_en[1]; chip->mode_lfo_freq[0] = chip->mode_lfo_freq[1]; } chip->mode_timer_a_reg[0] = 0; if (write_data_en && chip->write_mode_24[1] && !chip->bank_latch) chip->mode_timer_a_reg[0] |= (bus & 255) << 2; else chip->mode_timer_a_reg[0] |= chip->mode_timer_a_reg[1] & 0x3fc; if (write_data_en && chip->write_mode_25[1] && !chip->bank_latch) chip->mode_timer_a_reg[0] |= bus & 3; else chip->mode_timer_a_reg[0] |= chip->mode_timer_a_reg[1] & 0x3; if (write_data_en && chip->write_mode_26[1] && !chip->bank_latch) chip->mode_timer_b_reg[0] = bus & 255; else chip->mode_timer_b_reg[0] = chip->mode_timer_b_reg[1]; if (write_data_en && chip->write_mode_27[1] && !chip->bank_latch) { chip->mode_ch3[0] = (bus >> 6) & 3; chip->mode_timer_a_load[0] = bus & 1; chip->mode_timer_a_enable[0] = (bus >> 2) & 1; chip->mode_timer_a_reset[0] = (bus >> 4) & 1; chip->mode_timer_b_load[0] = (bus >> 1) & 1; chip->mode_timer_b_enable[0] = (bus >> 3) & 1; chip->mode_timer_b_reset[0] = (bus >> 5) & 1; } else { chip->mode_ch3[0] = chip->mode_ch3[1]; chip->mode_timer_a_load[0] = chip->mode_timer_a_load[1]; chip->mode_timer_a_enable[0] = chip->mode_timer_a_enable[1]; chip->mode_timer_a_reset[0] = 0; chip->mode_timer_b_load[0] = chip->mode_timer_b_load[1]; chip->mode_timer_b_enable[0] = chip->mode_timer_b_enable[1]; chip->mode_timer_b_reset[0] = 0; } if (write_data_en && chip->write_mode_28[1] && !chip->bank_latch) { chip->mode_kon_operator[0] = (bus >> 4) & 15; chip->mode_kon_channel[0] = bus & 15; } else { chip->mode_kon_operator[0] = chip->mode_kon_operator[1]; chip->mode_kon_channel[0] = chip->mode_kon_channel[1]; } if (write_data_en && chip->write_mode_2a[1] && !chip->bank_latch) chip->mode_dac_data[0] = (bus & 255) ^ 128; else chip->mode_dac_data[0] = chip->mode_dac_data[1]; if (write_data_en && chip->write_mode_2b[1] && !chip->bank_latch) chip->mode_dac_en[0] = (bus >> 7) & 1; else chip->mode_dac_en[0] = chip->mode_dac_en[1]; if (write_data_en && chip->write_mode_2c[1] && !chip->bank_latch) chip->mode_test_2c[0] = bus & 0xf8; else chip->mode_test_2c[0] = chip->mode_test_2c[1]; } if (chip->write_fm_data[1] && (chip->fm_address[1]&3) == chip->reg_cnt1[1] && ((chip->fm_address[1]>>2)&1) == ((chip->reg_cnt2[1]>>1)&1) && ((chip->fm_address[1]>>8)&1) == (chip->reg_cnt2[1]&1)) { int bank = (chip->fm_address[1]>>3)&1; switch (chip->fm_address[1] & 0xf0) { case 0x30: // multi for (j = 0; j < 4; j++) { chip->slot_multi[bank][j][0] &= ~1; chip->slot_multi[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } // dt for (j = 0; j < 3; j++) { chip->slot_dt[bank][j][0] &= ~1; chip->slot_dt[bank][j][0] |= (chip->fm_data[1] >> (j + 4)) & 1; } break; case 0x40: // tl for (j = 0; j < 7; j++) { chip->slot_tl[bank][j][0] &= ~1; chip->slot_tl[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } break; case 0x50: // ar for (j = 0; j < 5; j++) { chip->slot_ar[bank][j][0] &= ~1; chip->slot_ar[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } // ks for (j = 0; j < 2; j++) { chip->slot_ks[bank][j][0] &= ~1; chip->slot_ks[bank][j][0] |= (chip->fm_data[1] >> (j + 6)) & 1; } break; case 0x60: // dr for (j = 0; j < 5; j++) { chip->slot_dr[bank][j][0] &= ~1; chip->slot_dr[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } // am for (j = 0; j < 1; j++) { chip->slot_am[bank][j][0] &= ~1; chip->slot_am[bank][j][0] |= (chip->fm_data[1] >> (j + 7)) & 1; } break; case 0x70: // sr for (j = 0; j < 5; j++) { chip->slot_sr[bank][j][0] &= ~1; chip->slot_sr[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } break; case 0x80: // rr for (j = 0; j < 4; j++) { chip->slot_rr[bank][j][0] &= ~1; chip->slot_rr[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } // sl for (j = 0; j < 4; j++) { chip->slot_sl[bank][j][0] &= ~1; chip->slot_sl[bank][j][0] |= (chip->fm_data[1] >> (j + 4)) & 1; } break; case 0x90: // ssg eg for (j = 0; j < 4; j++) { chip->slot_ssg_eg[bank][j][0] &= ~1; chip->slot_ssg_eg[bank][j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } break; } } if (chip->write_fm_data[1] && (chip->fm_address[1] & 3) == chip->reg_cnt1[1] && ((chip->fm_address[1] >> 8) & 1) == (chip->reg_cnt2[1] & 1)) { switch (chip->fm_address[1] & 0xfc) { case 0xa0: // fnum for (j = 0; j < 8; j++) { chip->chan_fnum[j][0] &= ~1; chip->chan_fnum[j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } for (j = 0; j < 3; j++) { chip->chan_fnum[8+j][0] &= ~1; chip->chan_fnum[8+j][0] |= (chip->chan_a4[1] >> (j + 0)) & 1; } // block for (j = 0; j < 3; j++) { chip->chan_block[j][0] &= ~1; chip->chan_block[j][0] |= (chip->chan_a4[1] >> (j + 3)) & 1; } break; case 0xa4: chip->chan_a4[0] = chip->fm_data[1] & 0x3f; break; case 0xa8: // fnum for (j = 0; j < 8; j++) { chip->chan_fnum_ch3[j][0] &= ~1; chip->chan_fnum_ch3[j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } for (j = 0; j < 3; j++) { chip->chan_fnum_ch3[8+j][0] &= ~1; chip->chan_fnum_ch3[8+j][0] |= (chip->chan_ac[1] >> (j + 0)) & 1; } // block for (j = 0; j < 3; j++) { chip->chan_block_ch3[j][0] &= ~1; chip->chan_block_ch3[j][0] |= (chip->chan_ac[1] >> (j + 3)) & 1; } break; case 0xac: chip->chan_ac[0] = chip->fm_data[1] & 0x3f; break; case 0xb0: // connect for (j = 0; j < 3; j++) { chip->chan_connect[j][0] &= ~1; chip->chan_connect[j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } // fb for (j = 0; j < 3; j++) { chip->chan_fb[j][0] &= ~1; chip->chan_fb[j][0] |= (chip->fm_data[1] >> (j + 3)) & 1; } break; case 0xb4: // pms for (j = 0; j < 3; j++) { chip->chan_pms[j][0] &= ~1; chip->chan_pms[j][0] |= (chip->fm_data[1] >> (j + 0)) & 1; } // ams for (j = 0; j < 2; j++) { chip->chan_ams[j][0] &= ~1; chip->chan_ams[j][0] |= (chip->fm_data[1] >> (j + 4)) & 1; } // pan for (j = 0; j < 2; j++) { chip->chan_pan[j][0] &= ~1; chip->chan_pan[j][0] |= !((chip->fm_data[1] >> (j + 6)) & 1); } break; } } // keyon chip->mode_kon[0][0] = chip->mode_kon[0][1] << 1; chip->mode_kon[1][0] = chip->mode_kon[1][1] << 1; chip->mode_kon[2][0] = chip->mode_kon[2][1] << 1; chip->mode_kon[3][0] = chip->mode_kon[3][1] << 1; if (chip->reg_cnt2[1] == ((chip->mode_kon_channel[1] >> 2) & 1) && chip->reg_cnt1[1] == (chip->mode_kon_channel[1] & 3)) { chip->mode_kon[0][0] |= (chip->mode_kon_operator[1] >> 0) & 1; chip->mode_kon[1][0] |= (chip->mode_kon_operator[1] >> 3) & 1; chip->mode_kon[2][0] |= (chip->mode_kon_operator[1] >> 1) & 1; chip->mode_kon[3][0] |= (chip->mode_kon_operator[1] >> 2) & 1; } else { if (!chip->input.ic) chip->mode_kon[0][0] |= (chip->mode_kon[3][1] >> 5) & 1; chip->mode_kon[1][0] |= (chip->mode_kon[0][1] >> 5) & 1; chip->mode_kon[2][0] |= (chip->mode_kon[1][1] >> 5) & 1; chip->mode_kon[3][0] |= (chip->mode_kon[2][1] >> 5) & 1; } } void FMOPN2_FMRegisters2(fmopn2_t *chip) { chip->write_fm_address[1] = chip->write_fm_address[0]; chip->write_fm_data[1] = chip->write_fm_data[0]; chip->fm_address[1] = chip->fm_address[0]; chip->fm_data[1] = chip->fm_data[0]; chip->write_mode_21[1] = chip->write_mode_21[0]; chip->write_mode_22[1] = chip->write_mode_22[0]; chip->write_mode_24[1] = chip->write_mode_24[0]; chip->write_mode_25[1] = chip->write_mode_25[0]; chip->write_mode_26[1] = chip->write_mode_26[0]; chip->write_mode_27[1] = chip->write_mode_27[0]; chip->write_mode_28[1] = chip->write_mode_28[0]; chip->write_mode_2a[1] = chip->write_mode_2a[0]; chip->write_mode_2b[1] = chip->write_mode_2b[0]; chip->write_mode_2c[1] = chip->write_mode_2c[0]; chip->mode_test_21[1] = chip->mode_test_21[0]; chip->mode_lfo_en[1] = chip->mode_lfo_en[0]; chip->mode_lfo_freq[1] = chip->mode_lfo_freq[0]; chip->mode_timer_a_reg[1] = chip->mode_timer_a_reg[0]; chip->mode_timer_b_reg[1] = chip->mode_timer_b_reg[0]; chip->mode_ch3[1] = chip->mode_ch3[0]; chip->mode_timer_a_load[1] = chip->mode_timer_a_load[0]; chip->mode_timer_a_enable[1] = chip->mode_timer_a_enable[0]; chip->mode_timer_a_reset[1] = chip->mode_timer_a_reset[0]; chip->mode_timer_b_load[1] = chip->mode_timer_b_load[0]; chip->mode_timer_b_enable[1] = chip->mode_timer_b_enable[0]; chip->mode_timer_b_reset[1] = chip->mode_timer_b_reset[0]; chip->mode_kon_operator[1] = chip->mode_kon_operator[0]; chip->mode_kon_channel[1] = chip->mode_kon_channel[0]; chip->mode_dac_data[1] = chip->mode_dac_data[0]; chip->mode_dac_en[1] = chip->mode_dac_en[0]; chip->mode_test_2c[1] = chip->mode_test_2c[0]; chip->chan_a4[1] = chip->chan_a4[0]; chip->chan_ac[1] = chip->chan_ac[0]; chip->mode_kon[0][1] = chip->mode_kon[0][0]; chip->mode_kon[1][1] = chip->mode_kon[1][0]; chip->mode_kon[2][1] = chip->mode_kon[2][0]; chip->mode_kon[3][1] = chip->mode_kon[3][0]; } void FMOPN2_Misc1(fmopn2_t *chip) { chip->reg_cnt1[0] = chip->reg_cnt1[1] + 1; chip->reg_cnt2[0] = chip->reg_cnt2[1]; if (chip->reg_cnt1[1] & 2) { chip->reg_cnt1[0] = 0; chip->reg_cnt2[0]++; } if (chip->fsm_sel23 || chip->input.ic) { chip->reg_cnt1[0] = 0; chip->reg_cnt2[0] = 0; } } void FMOPN2_Misc2(fmopn2_t *chip) { chip->reg_cnt1[1] = chip->reg_cnt1[0] & 3; chip->reg_cnt2[1] = chip->reg_cnt2[0] & 7; } void FMOPN2_LFO1(fmopn2_t *chip) { static const int lfo_cycles[8] = { 108, 77, 71, 67, 62, 44, 8, 5 }; int inc = (chip->mode_test_21[1] & 2) != 0 || chip->fsm_sel23; int freq = chip->mode_lfo_freq[1]; int of = (chip->lfo_cnt1[1] & lfo_cycles[freq]) == lfo_cycles[freq]; chip->lfo_cnt1[0] = chip->lfo_cnt1[1] + inc; if (chip->input.ic || of) chip->lfo_cnt1[0] = 0; chip->lfo_cnt2[0] = chip->lfo_cnt2[1] + of; if (!chip->mode_lfo_en[1]) chip->lfo_cnt2[0] = 0; chip->lfo_inc_latch[0] = chip->fsm_sel23; chip->lfo_dlatch_load = chip->lfo_inc_latch[1]; } void FMOPN2_LFO2(fmopn2_t *chip) { chip->lfo_cnt1[1] = chip->lfo_cnt1[0] & 127; chip->lfo_cnt2[1] = chip->lfo_cnt2[0] & 127; chip->lfo_inc_latch[1] = chip->lfo_inc_latch[0]; if (chip->lfo_inc_latch[1] && !chip->lfo_dlatch_load) chip->lfo_dlatch = chip->lfo_cnt2[1]; } void FMOPN2_PhaseGenerator1(fmopn2_t *chip) { // Note table static const int fn_note[16] = { 0, 0, 0, 0, 0, 0, 0, 1, 2, 3, 3, 3, 3, 3, 3, 3 }; // LFO shift static const int pg_lfo_sh1[8][8] = { { 7, 7, 7, 7, 7, 7, 7, 7 }, { 7, 7, 7, 7, 7, 7, 7, 7 }, { 7, 7, 7, 7, 7, 7, 1, 1 }, { 7, 7, 7, 7, 1, 1, 1, 1 }, { 7, 7, 7, 1, 1, 1, 1, 0 }, { 7, 7, 1, 1, 0, 0, 0, 0 }, { 7, 7, 1, 1, 0, 0, 0, 0 }, { 7, 7, 1, 1, 0, 0, 0, 0 } }; static const int pg_lfo_sh2[8][8] = { { 7, 7, 7, 7, 7, 7, 7, 7 }, { 7, 7, 7, 7, 2, 2, 2, 2 }, { 7, 7, 7, 2, 2, 2, 7, 7 }, { 7, 7, 2, 2, 7, 7, 2, 2 }, { 7, 7, 2, 7, 7, 7, 2, 7 }, { 7, 7, 7, 2, 7, 7, 2, 1 }, { 7, 7, 7, 2, 7, 7, 2, 1 }, { 7, 7, 7, 2, 7, 7, 2, 1 } }; #if 0 // YM2610 decap static const int pg_lfo_sh2[8][8] = { { 7, 7, 7, 7, 7, 7, 7, 7 }, { 7, 7, 7, 7, 2, 2, 2, 2 }, { 7, 7, 7, 2, 2, 2, 7, 7 }, { 7, 7, 2, 2, 7, 7, 2, 2 }, { 7, 2, 2, 7, 7, 2, 2, 7 }, { 7, 2, 7, 2, 7, 2, 2, 1 }, { 7, 2, 7, 2, 7, 2, 2, 1 }, { 7, 2, 7, 2, 7, 2, 2, 1 } }; #endif static const int pg_detune_add[4] = { 0, 8, 10, 11 }; static const int pg_detune[8] = { 16, 17, 19, 20, 22, 24, 27, 29 }; int i; int fnum = 0; int fnum_h; int block = 0; int pms = 0; int dt = 0; int dt_l = 0; int dt_sum; int dt_sum_l; int dt_sum_h; int kcode; int multi = 0; int lfo; int ch3_sel = chip->reg_cnt1[1] == 1 && (chip->reg_cnt2[1] & 1) == 0 && chip->mode_ch3[1] != 0; int op_sel = chip->reg_cnt2[1] >> 1; int bank = (chip->reg_cnt2[1] >> 2) & 1; int carry = 0; int pg_inc; int reset; if (ch3_sel && op_sel == 0) { for (i = 0; i < 11; i++) fnum |= ((chip->chan_fnum_ch3[i][1] >> 5) & 1) << i; for (i = 0; i < 3; i++) block |= ((chip->chan_block_ch3[i][1] >> 5) & 1) << i; } else if (ch3_sel && op_sel == 1) { for (i = 0; i < 11; i++) fnum |= ((chip->chan_fnum_ch3[i][1] >> 0) & 1) << i; for (i = 0; i < 3; i++) block |= ((chip->chan_block_ch3[i][1] >> 0) & 1) << i; } else if (ch3_sel && op_sel == 2) { for (i = 0; i < 11; i++) fnum |= ((chip->chan_fnum_ch3[i][1] >> 4) & 1) << i; for (i = 0; i < 3; i++) block |= ((chip->chan_block_ch3[i][1] >> 4) & 1) << i; } else { for (i = 0; i < 11; i++) fnum |= ((chip->chan_fnum[i][1] >> 4) & 1) << i; for (i = 0; i < 3; i++) block |= ((chip->chan_block[i][1] >> 4) & 1) << i; } for (i = 0; i < 3; i++) pms |= ((chip->chan_pms[i][1] >> 5) & 1) << i; for (i = 0; i < 3; i++) dt |= ((chip->slot_dt[bank][i][1] >> 11) & 1) << i; for (i = 0; i < 4; i++) multi |= ((chip->slot_multi[bank][i][1] >> 11) & 1) << i; kcode = (block << 2) | fn_note[fnum >> 7]; chip->pg_kcode[0][0] = kcode; chip->pg_kcode[1][0] = chip->pg_kcode[0][1]; chip->pg_fnum[0][0] = fnum; chip->pg_fnum[1][0] = chip->pg_fnum[0][1]; lfo = (chip->lfo_dlatch >> 2) & 7; if (chip->lfo_dlatch & 32) lfo ^= 7; fnum_h = chip->pg_fnum[0][1] >> 4; chip->pg_fnum_lfo1 = fnum_h >> pg_lfo_sh1[pms][lfo]; chip->pg_fnum_lfo2 = fnum_h >> pg_lfo_sh2[pms][lfo]; chip->pg_lfo_shift = 2; if (pms > 5) chip->pg_lfo_shift = 7 - pms; chip->pg_lfo_sign = (chip->lfo_dlatch >> 6) & 1; chip->pg_freq1 = ((chip->pg_fnum[1][1] << 1) + chip->pg_lfo) & 0xfff; block = chip->pg_kcode[1][1] >> 2; chip->pg_block = block; chip->pg_dt[0] = dt; dt_l = chip->pg_dt[1] & 3; kcode = chip->pg_kcode[1][1]; if (kcode > 28) kcode = 28; dt_sum = kcode + ((pg_detune_add[dt_l] + 1) << 2); dt_sum_l = dt_sum & 7; dt_sum_h = dt_sum >> 3; chip->pg_detune[0] = dt_l ? pg_detune[dt_sum_l] >> (9 - dt_sum_h) : 0; if (chip->pg_dt[1] & 0x04) chip->pg_detune[0] = -chip->pg_detune[0]; chip->pg_freq3 = (chip->pg_freq2 + chip->pg_detune[1]) & 0x1ffff; chip->pg_multi[0][0] = multi; chip->pg_multi[1][0] = chip->pg_multi[0][1]; chip->pg_multi2 = chip->pg_multi[1][1]; chip->pg_inc[0] = chip->pg_freq4; chip->pg_inc_mask[0] = chip->pg_inc[1]; if (chip->pg_reset[1] & 2) chip->pg_inc_mask[0] = 0; chip->pg_reset_latch[0] = (chip->pg_reset[1] & 2) != 0; pg_inc = chip->pg_inc_mask[1]; reset = chip->pg_reset_latch[1] || (chip->mode_test_21[1] & 8) != 0; for (i = 0; i < 20; i++) { if (!reset) carry += (chip->pg_phase[i][1] >> 23) & 1; carry += pg_inc & 1; chip->pg_phase[i][0] = (chip->pg_phase[i][1] << 1) | (carry & 1); pg_inc >>= 1; carry >>= 1; } chip->pg_debug[0] = chip->pg_debug[1] >> 1; if (chip->fsm_sel2) { for (i = 0; i < 10; i++) { chip->pg_debug[0] |= ((chip->pg_phase[i][1] >> 23) & 1) << i; } } } void FMOPN2_PhaseGenerator2(fmopn2_t *chip) { int i; int block = chip->pg_kcode[1][0]; chip->pg_fnum[0][1] = chip->pg_fnum[0][0]; chip->pg_fnum[1][1] = chip->pg_fnum[1][0]; chip->pg_kcode[0][1] = chip->pg_kcode[0][0]; chip->pg_kcode[1][1] = chip->pg_kcode[1][0]; chip->pg_lfo = (chip->pg_fnum_lfo1 + chip->pg_fnum_lfo2) >> chip->pg_lfo_shift; if (chip->pg_lfo_sign) chip->pg_lfo = -chip->pg_lfo; chip->pg_freq2 = (chip->pg_freq1 << chip->pg_block) >> 2; chip->pg_dt[1] = chip->pg_dt[0]; chip->pg_detune[1] = chip->pg_detune[0]; chip->pg_multi[0][1] = chip->pg_multi[0][0]; chip->pg_multi[1][1] = chip->pg_multi[1][0]; if (!chip->pg_multi2) chip->pg_freq4 = chip->pg_freq3 >> 1; else chip->pg_freq4 = chip->pg_freq3 * chip->pg_multi2; chip->pg_inc[1] = chip->pg_inc[0]; chip->pg_inc_mask[1] = chip->pg_inc_mask[0]; chip->pg_reset_latch[1] = chip->pg_reset_latch[0]; for (i = 0; i < 20; i++) { chip->pg_phase[i][1] = chip->pg_phase[i][0]; } chip->pg_debug[1] = chip->pg_debug[0]; } void FMOPN2_EnvelopeGenerator1(fmopn2_t *chip) { int i; int sum; int add; int timer_bit; int timer_bit_masked; int bank = (chip->reg_cnt2[1] >> 2) & 1; int rate_sel; int rate = 0; int ks = 0; int b0, b1; int inc1; int exp = 0; int linear = 0; int inc_total = 0; int level = 0; int level2 = 0; int kon; int kon2; int okon; int okon2; int state; int pg_reset; int kon_event; int ssg_eg = 0; int ssg_inv_e = 0; int ssg_dir = 0; int ssg_inv = 0; int ssg_holdup = 0; int ssg_enable; int ssg_pgreset = 0; int ssg_pgrepeat = 0; int eg_off; int eg_slreach; int sl = 0; int nextlevel = 0; int nextstate = eg_state_attack; int tl = 0; int istantattack = 0; int eg_output; int ams = 0; int csm_kon; static const int eg_stephi[4][4] = { { 0, 0, 0, 0 }, { 1, 0, 0, 0 }, { 1, 0, 1, 0 }, { 1, 1, 1, 0 } }; chip->eg_prescaler_clock_l[0] = chip->fsm_clock_eg; chip->eg_prescaler[0] = chip->eg_prescaler[1] + chip->fsm_clock_eg; if (((chip->eg_prescaler[1] & 2) != 0 && chip->fsm_clock_eg) || chip->input.ic) chip->eg_prescaler[0] = 0; chip->eg_step[0] = chip->eg_prescaler[1] >> 1; chip->eg_clock_delay[0] = (chip->eg_clock_delay[1] << 1) | chip->fsm_clock_eg; chip->eg_timer_load = chip->eg_step[1] && chip->eg_prescaler_clock_l[1]; sum = (chip->eg_timer[1] >> 11) & 1; add = chip->eg_timer_carry[1]; if ((chip->eg_prescaler[1] & 2) != 0 && chip->eg_prescaler_clock_l[1]) add = 1; sum += add; chip->eg_timer_carry[0] = sum >> 1; sum &= 1; if (chip->input.ic || (chip->mode_test_21[1] & 32) != 0) sum = 0; chip->eg_timer[0] = (chip->eg_timer[1] << 1) | sum; timer_bit = sum; if (chip->mode_test_2c[1] & 64) { if (chip->mode_test_2c[1] & 128) // Assuming TEST pin is NC timer_bit |= FMOPN2_ReadTest(chip); else timer_bit |= chip->input.test & 1; } chip->eg_timer_mask[0] = timer_bit | chip->eg_timer_mask[1]; if (chip->fsm_clock_eg || ((chip->eg_clock_delay[1]>>11) & 1) != 0 || chip->input.ic) chip->eg_timer_mask[0] = 0; timer_bit_masked = timer_bit; if (chip->eg_timer_mask[1]) timer_bit_masked = 0; chip->eg_timer_masked[0] = (chip->eg_timer_masked[1] << 1) | timer_bit_masked; for (i = 0; i < 10; i++) { level2 |= ((chip->eg_level[i][1] >> 20) & 1) << i; } chip->eg_level_latch[0] = level2; csm_kon = chip->fsm_ch3_sel && chip->timer_csm_key_dlatch; kon2 = ((chip->mode_kon[3][1] >> 5) & 1) | csm_kon; chip->eg_kon_latch[0] = (chip->eg_kon_latch[1] << 1) | kon2; chip->eg_kon_csm[0] = (chip->eg_kon_csm[1] << 1) | csm_kon; kon = (chip->eg_kon_latch[1] >> 1) & 1; okon = (chip->eg_key[1] >> 23) & 1; pg_reset = (kon && !okon) || (chip->eg_ssg_pgreset[1] & 2) != 0; kon_event = (kon && !okon) || (okon && (chip->eg_ssg_pgrepeat[1] & 2) != 0); chip->pg_reset[0] = (chip->pg_reset[1] << 1) | pg_reset; if (chip->eg_ssg_pgreset[1] & 2) chip->pg_reset[0] |= 1; chip->eg_key[0] = (chip->eg_key[1] << 1) | kon; okon2 = (chip->eg_key[1] >> 21) & 1; for (i = 0; i < 4; i++) ssg_eg |= ((chip->slot_ssg_eg[bank][i][1] >> 11) & 1) << i; ssg_enable = (ssg_eg & 8) != 0; ssg_inv_e = ssg_enable && (ssg_eg & 4) != 0; if (ssg_enable) { if (okon2) { ssg_dir = (chip->eg_ssg_dir[1] >> 23) & 1; if (level2 & 512) { if ((ssg_eg & 3) == 2) ssg_dir ^= 1; if ((ssg_eg & 3) == 3) ssg_dir = 1; } } if (kon2) ssg_holdup = (ssg_eg & 7) == 3 || (ssg_eg & 7) == 5; if (level2 & 512) { if ((ssg_eg & 3) == 0) ssg_pgreset = 1; if ((ssg_eg & 1) == 0) ssg_pgrepeat = 1; } } ssg_inv = okon2 & (((chip->eg_ssg_dir[1] >> 23) & 1) ^ ssg_inv_e); chip->eg_ssg_dir[0] = (chip->eg_ssg_dir[1] << 1) | ssg_dir; chip->eg_ssg_inv[0] = ssg_inv; chip->eg_ssg_holdup[0] = (chip->eg_ssg_holdup[1] << 1) | ssg_holdup; chip->eg_ssg_enable[0] = (chip->eg_ssg_enable[1] << 1) | ssg_enable; chip->eg_ssg_pgreset[0] = (chip->eg_ssg_pgreset[1] << 1) | ssg_pgreset; chip->eg_ssg_pgrepeat[0] = (chip->eg_ssg_pgrepeat[1] << 1) | ssg_pgrepeat; chip->eg_level_ssg[0] = eg_output = chip->eg_ssg_inv[1] ? chip->eg_level_latch_inv : chip->eg_level_latch[1]; if (chip->mode_test_21[1] & 32) eg_output = 0; for (i = 0; i < 4; i++) sl |= ((chip->slot_sl[bank][i][1] >> 11) & 1) << i; if (sl == 15) sl |= 16; chip->eg_sl[0][0] = sl; chip->eg_sl[1][0] = chip->eg_sl[0][1]; for (i = 0; i < 7; i++) tl |= ((chip->slot_tl[bank][i][1] >> 11) & 1) << i; chip->eg_tl[0][0] = tl; chip->eg_tl[1][0] = chip->eg_tl[0][1]; b0 = (chip->eg_state[0][1] >> 21) & 1; b1 = (chip->eg_state[1][1] >> 21) & 1; rate_sel = b1 * 2 + b0; if (okon2) { if (ssg_pgrepeat) rate_sel = eg_state_attack; } else { if (kon2) rate_sel = eg_state_attack; } switch (rate_sel) { case eg_state_attack: for (i = 0; i < 5; i++) rate |= ((chip->slot_ar[bank][i][1] >> 11) & 1) << i; break; case eg_state_decay: for (i = 0; i < 5; i++) rate |= ((chip->slot_dr[bank][i][1] >> 11) & 1) << i; break; case eg_state_sustain: for (i = 0; i < 5; i++) rate |= ((chip->slot_sr[bank][i][1] >> 11) & 1) << i; break; case eg_state_release: rate = 1; for (i = 0; i < 4; i++) rate |= ((chip->slot_rr[bank][i][1] >> 11) & 1) << (i + 1); break; } chip->eg_rate_nonzero[0] = rate != 0; chip->eg_rate = rate; for (i = 0; i < 2; i++) ks |= ((chip->slot_ks[bank][i][1] >> 11) & 1) << i; chip->eg_ksv = chip->pg_kcode[0][1] >> (ks ^ 3); rate = chip->eg_rate2; if (rate & 64) rate = 63; sum = (rate >> 2) + chip->eg_shift_lock; inc1 = 0; if (rate < 48 && chip->eg_rate_nonzero[1]) { switch (sum & 15) { case 12: inc1 = rate != 0; break; case 13: inc1 = (rate >> 1) & 1; break; case 14: inc1 = rate & 1; break; } } chip->eg_inc1 = inc1; chip->eg_inc2 = eg_stephi[rate & 3][chip->eg_timer_low_lock]; chip->eg_rate12 = (rate & 60) == 48; chip->eg_rate13 = (rate & 60) == 52; chip->eg_rate14 = (rate & 60) == 56; chip->eg_rate15 = (rate & 60) == 60; chip->eg_maxrate[0] = (rate & 62) == 62; chip->eg_prescaler_l = (chip->eg_prescaler[1] & 2) != 0; chip->eg_incsh_nonzero[0] = chip->eg_incsh0 | chip->eg_incsh1 | chip->eg_incsh2 | chip->eg_incsh3; if (okon && !kon) { level = chip->eg_level_ssg[1]; } else { for (i = 0; i < 10; i++) { level |= ((chip->eg_level[i][1] >> 22) & 1) << i; } } b0 = (chip->eg_state[0][1] >> 23) & 1; b1 = (chip->eg_state[1][1] >> 23) & 1; state = b1 * 2 + b0; exp = kon && (state == eg_state_attack) && !chip->eg_maxrate[1] && level != 0; eg_off = (chip->eg_ssg_enable[1] & 2) != 0 ? (level & 512) != 0 : (level & 0x3f0) == 0x3f0; eg_slreach = (level >> 4) == (chip->eg_sl[1][1] << 1); linear = !kon_event && !eg_off && (state == eg_state_sustain || state == eg_state_release); linear |= !kon_event && !eg_off && !eg_slreach && state == eg_state_decay; if (exp) { if (chip->eg_incsh0) inc_total |= ~level >> 4; if (chip->eg_incsh1) inc_total |= ~level >> 3; if (chip->eg_incsh2) inc_total |= ~level >> 2; if (chip->eg_incsh3) inc_total |= ~level >> 1; } if (linear) { if (chip->eg_ssg_enable[1] & 2) { if (chip->eg_incsh0) inc_total |= 4; if (chip->eg_incsh1) inc_total |= 8; if (chip->eg_incsh2) inc_total |= 16; if (chip->eg_incsh3) inc_total |= 32; } else { if (chip->eg_incsh0) inc_total |= 1; if (chip->eg_incsh1) inc_total |= 2; if (chip->eg_incsh2) inc_total |= 4; if (chip->eg_incsh3) inc_total |= 8; } } chip->eg_inc_total = inc_total; istantattack = chip->eg_maxrate[1] && (!chip->eg_maxrate[1] || kon_event); if (!istantattack) nextlevel |= level; if (chip->eg_kon_csm[1] & 2) nextlevel |= chip->eg_tl[1][1] << 3; if ((!kon_event && eg_off && (chip->eg_ssg_holdup[1] & 2) == 0 && state != eg_state_attack) || chip->input.ic) { nextlevel = 0x3ff; nextstate |= eg_state_release; } if (!kon_event && state == eg_state_sustain) { nextstate |= eg_state_sustain; } if (!kon_event && state == eg_state_decay && !eg_slreach) { nextstate |= eg_state_decay; } if (!kon_event && state == eg_state_decay && eg_slreach) { nextstate |= eg_state_sustain; } if (!kon && !kon_event) { nextstate |= eg_state_release; } if (!kon_event && state == eg_state_release) { nextstate |= eg_state_release; } if (!kon_event && state == eg_state_attack && level == 0) { nextstate |= eg_state_decay; } if (chip->input.ic) { nextstate |= eg_state_release; } chip->eg_nextlevel[0] = nextlevel; b0 = nextstate & 1; b1 = (nextstate >> 1) & 1; chip->eg_state[0][0] = (chip->eg_state[0][1] << 1) | b0; chip->eg_state[1][0] = (chip->eg_state[1][1] << 1) | b1; nextlevel = chip->eg_nextlevel[1]; for (i = 0; i < 10; i++) { chip->eg_level[i][0] = (chip->eg_level[i][1] << 1) | (nextlevel & 1); nextlevel >>= 1; } if (chip->slot_am[bank][0][1] & (1<<11)) for (i = 0; i < 2; i++) ams |= ((chip->chan_ams[i][1] >> 5) & 1) << i; chip->eg_ams = ams; if (chip->lfo_dlatch & 64) chip->eg_lfo[0] = chip->lfo_dlatch & 63; else chip->eg_lfo[0] = chip->lfo_dlatch ^ 63; chip->eg_ch3_latch[0] = chip->fsm_ch3_sel; chip->eg_out_tl = chip->eg_tl[0][1]; if (chip->eg_ch3_latch[1] && chip->mode_ch3[1] == 2) // CSM chip->eg_out_tl = 0; chip->eg_out = eg_output + chip->eg_lfo[1]; chip->eg_debug[0] = chip->eg_debug[1] << 1; if (chip->fsm_sel2) chip->eg_debug[0] |= chip->eg_out_total; } void FMOPN2_EnvelopeGenerator2(fmopn2_t *chip) { int i; int b0, b1, b2, b3; static const int eg_am_shift[4] = { 7, 3, 1, 0 }; chip->eg_prescaler_clock_l[1] = chip->eg_prescaler_clock_l[0]; chip->eg_prescaler[1] = chip->eg_prescaler[0] & 3; chip->eg_step[1] = chip->eg_step[0]; chip->eg_timer[1] = chip->eg_timer[0]; chip->eg_clock_delay[1] = chip->eg_clock_delay[0]; chip->eg_timer_carry[1] = chip->eg_timer_carry[0]; chip->eg_timer_mask[1] = chip->eg_timer_mask[0]; chip->eg_timer_masked[1] = chip->eg_timer_masked[0]; if (!chip->eg_timer_load && chip->eg_step[1] && chip->eg_prescaler_clock_l[1]) { b0 = (chip->eg_timer[1] >> 11) & 1; b1 = (chip->eg_timer[1] >> 10) & 1; chip->eg_timer_low_lock = b1 * 2 + b0; b0 = (chip->eg_timer_masked[1] & 0xaaa) != 0; b1 = (chip->eg_timer_masked[1] & 0x666) != 0; b2 = (chip->eg_timer_masked[1] & 0x1e1) != 0; b3 = (chip->eg_timer_masked[1] & 0x1f) != 0; chip->eg_shift_lock = b3 * 8 + b2 * 4 + b1 * 2 + b0; } chip->eg_rate_nonzero[1] = chip->eg_rate_nonzero[0]; chip->eg_rate2 = (chip->eg_rate << 1) + chip->eg_ksv; chip->eg_maxrate[1] = chip->eg_maxrate[0]; chip->eg_incsh0 = 0; chip->eg_incsh1 = 0; chip->eg_incsh2 = 0; chip->eg_incsh3 = 0; if (chip->eg_prescaler_l) { chip->eg_incsh0 = chip->eg_inc1; chip->eg_incsh3 = chip->eg_rate15; if (!chip->eg_inc2) { chip->eg_incsh0 |= chip->eg_rate12; chip->eg_incsh1 = chip->eg_rate13; chip->eg_incsh2 = chip->eg_rate14; } else { chip->eg_incsh1 = chip->eg_rate12; chip->eg_incsh2 = chip->eg_rate13; chip->eg_incsh3 |= chip->eg_rate14; } } chip->eg_incsh_nonzero[1] = chip->eg_incsh_nonzero[0]; chip->eg_kon_latch[1] = chip->eg_kon_latch[0]; chip->eg_level_ssg[1] = chip->eg_level_ssg[0]; chip->pg_reset[1] = chip->pg_reset[0]; chip->eg_ssg_dir[1] = chip->eg_ssg_dir[0]; chip->eg_ssg_inv[1] = chip->eg_ssg_inv[0]; chip->eg_ssg_holdup[1] = chip->eg_ssg_holdup[0]; chip->eg_ssg_enable[1] = chip->eg_ssg_enable[0]; chip->eg_ssg_pgreset[1] = chip->eg_ssg_pgreset[0]; chip->eg_ssg_pgrepeat[1] = chip->eg_ssg_pgrepeat[0]; chip->eg_level_latch[1] = chip->eg_level_latch[0]; chip->eg_level_latch_inv = (512 - chip->eg_level_latch[0]) & 0x3ff; chip->eg_sl[0][1] = chip->eg_sl[0][0]; chip->eg_sl[1][1] = chip->eg_sl[1][0]; chip->eg_tl[0][1] = chip->eg_tl[0][0]; chip->eg_tl[1][1] = chip->eg_tl[1][0]; chip->eg_nextlevel[1] = chip->eg_nextlevel[0] + chip->eg_inc_total; chip->eg_kon_csm[1] = chip->eg_kon_csm[0]; for (i = 0; i < 10; i++) { chip->eg_level[i][1] = chip->eg_level[i][0]; } chip->eg_state[0][1] = chip->eg_state[0][0]; chip->eg_state[1][1] = chip->eg_state[1][0]; chip->eg_lfo[1] = (chip->eg_lfo[0] << 1) >> eg_am_shift[chip->eg_ams]; chip->eg_ch3_latch[1] = chip->eg_ch3_latch[0]; chip->eg_out_total = (chip->eg_out & 1023) + (chip->eg_out_tl << 3); if ((chip->eg_out & 1024) != 0 || (chip->eg_out_total & 1024) != 0) chip->eg_out_total = 1023; chip->eg_debug[1] = chip->eg_debug[0]; chip->eg_key[1] = chip->eg_key[0]; } void FMOPN2_Operator1(fmopn2_t *chip) { int i; int carry = 0; int phase = 0; int quarter; int index; int atten = 0; int output; int mod1 = 0, mod2 = 0; int mod; int fb = 0; static const int logsin[128] = { 0x6c3, 0x58b, 0x4e4, 0x471, 0x41a, 0x3d3, 0x398, 0x365, 0x339, 0x311, 0x2ed, 0x2cd, 0x2af, 0x293, 0x279, 0x261, 0x24b, 0x236, 0x222, 0x20f, 0x1fd, 0x1ec, 0x1dc, 0x1cd, 0x1be, 0x1b0, 0x1a2, 0x195, 0x188, 0x17c, 0x171, 0x166, 0x15b, 0x150, 0x146, 0x13c, 0x133, 0x129, 0x121, 0x118, 0x10f, 0x107, 0x0ff, 0x0f8, 0x0f0, 0x0e9, 0x0e2, 0x0db, 0x0d4, 0x0cd, 0x0c7, 0x0c1, 0x0bb, 0x0b5, 0x0af, 0x0a9, 0x0a4, 0x09f, 0x099, 0x094, 0x08f, 0x08a, 0x086, 0x081, 0x07d, 0x078, 0x074, 0x070, 0x06c, 0x068, 0x064, 0x060, 0x05c, 0x059, 0x055, 0x052, 0x04e, 0x04b, 0x048, 0x045, 0x042, 0x03f, 0x03c, 0x039, 0x037, 0x034, 0x031, 0x02f, 0x02d, 0x02a, 0x028, 0x026, 0x024, 0x022, 0x020, 0x01e, 0x01c, 0x01a, 0x018, 0x017, 0x015, 0x014, 0x012, 0x011, 0x00f, 0x00e, 0x00d, 0x00c, 0x00a, 0x009, 0x008, 0x007, 0x007, 0x006, 0x005, 0x004, 0x004, 0x003, 0x002, 0x002, 0x001, 0x001, 0x001, 0x001, 0x000, 0x000, 0x000, 0x000 }; static const int logsin_d[128] = { 0x196, 0x07c, 0x04a, 0x035, 0x029, 0x022, 0x01d, 0x019, 0x015, 0x013, 0x012, 0x00f, 0x00e, 0x00d, 0x00d, 0x00c, 0x00b, 0x00a, 0x00a, 0x009, 0x009, 0x009, 0x008, 0x007, 0x007, 0x007, 0x007, 0x006, 0x007, 0x006, 0x006, 0x005, 0x005, 0x005, 0x005, 0x005, 0x004, 0x005, 0x004, 0x004, 0x005, 0x004, 0x004, 0x003, 0x004, 0x003, 0x003, 0x003, 0x003, 0x004, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x002, 0x003, 0x003, 0x003, 0x003, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x002, 0x001, 0x002, 0x002, 0x002, 0x001, 0x001, 0x001, 0x002, 0x002, 0x001, 0x001, 0x002, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x001, 0x000, 0x001, 0x000, 0x001, 0x000, 0x001, 0x001, 0x000, 0x000, 0x001, 0x001, 0x001, 0x001, 0x000, 0x000, 0x000, 0x001, 0x000, 0x000, 0x001, 0x000, 0x001, 0x000, 0x000, 0x000, 0x000, 0x000, 0x000, 0x000 }; static const int pow[128] = { 0x3f5, 0x3ea, 0x3df, 0x3d4, 0x3c9, 0x3bf, 0x3b4, 0x3a9, 0x39f, 0x394, 0x38a, 0x37f, 0x375, 0x36a, 0x360, 0x356, 0x34c, 0x342, 0x338, 0x32e, 0x324, 0x31a, 0x310, 0x306, 0x2fd, 0x2f3, 0x2e9, 0x2e0, 0x2d6, 0x2cd, 0x2c4, 0x2ba, 0x2b1, 0x2a8, 0x29e, 0x295, 0x28c, 0x283, 0x27a, 0x271, 0x268, 0x25f, 0x257, 0x24e, 0x245, 0x23c, 0x234, 0x22b, 0x223, 0x21a, 0x212, 0x209, 0x201, 0x1f9, 0x1f0, 0x1e8, 0x1e0, 0x1d8, 0x1d0, 0x1c8, 0x1c0, 0x1b8, 0x1b0, 0x1a8, 0x1a0, 0x199, 0x191, 0x189, 0x181, 0x17a, 0x172, 0x16b, 0x163, 0x15c, 0x154, 0x14d, 0x146, 0x13e, 0x137, 0x130, 0x129, 0x122, 0x11b, 0x114, 0x10c, 0x106, 0x0ff, 0x0f8, 0x0f1, 0x0ea, 0x0e3, 0x0dc, 0x0d6, 0x0cf, 0x0c8, 0x0c2, 0x0bb, 0x0b5, 0x0ae, 0x0a8, 0x0a1, 0x09b, 0x094, 0x08e, 0x088, 0x082, 0x07b, 0x075, 0x06f, 0x069, 0x063, 0x05d, 0x057, 0x051, 0x04b, 0x045, 0x03f, 0x039, 0x033, 0x02d, 0x028, 0x022, 0x01c, 0x016, 0x011, 0x00b, 0x006, 0x000, }; static const int pow_d[128] = { 0x005, 0x005, 0x005, 0x006, 0x006, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x006, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x004, 0x005, 0x004, 0x004, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x005, 0x004, 0x004, 0x004, 0x005, 0x004, 0x005, 0x004, 0x004, 0x004, 0x005, 0x004, 0x004, 0x005, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x003, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x004, 0x003, 0x004, 0x004, 0x004, 0x003, 0x003, 0x003, 0x003, 0x004, 0x003, 0x003, 0x003, 0x003, 0x003, 0x004, 0x004, 0x003, 0x003, 0x004, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x004, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x003, 0x002, 0x003, 0x003, 0x003, 0x003, 0x003, 0x002, 0x003, }; for (i = 0; i < 10; i++) { carry += (chip->op_mod[i][1] >> 5) & 1; carry += (chip->pg_phase[10 + i][1] >> 19) & 1; phase += (carry & 1) << i; carry >>= 1; } chip->op_phase[0] = phase; chip->op_sign[0] = (chip->op_sign[1] << 1) | ((chip->op_phase[1] >> 9) & 1); quarter = chip->op_phase[1] & 255; if (chip->op_phase[1] & 256) quarter ^= 255; chip->op_logsin_add_delta[0] = (quarter & 1) == 0; chip->op_logsin_base[0] = logsin[quarter >> 1]; chip->op_logsin_delta[0] = logsin_d[quarter >> 1]; chip->op_env[0] = chip->eg_out_total; atten = chip->op_logsin_base[1]; if (chip->op_logsin_add_delta[1]) atten += chip->op_logsin_delta[1]; atten += chip->op_env[1] << 2; chip->op_atten[0] = atten; atten = chip->op_atten[1]; if (atten & 4096) atten = 4095; index = atten & 255; chip->op_shift[0] = atten >> 8; chip->op_pow_add_delta[0] = (index & 1) == 0; chip->op_pow_base[0] = pow[index >> 1]; chip->op_pow_delta[0] = pow_d[index >> 1]; output = chip->op_pow_base[1]; if (chip->op_pow_add_delta[1]) output += chip->op_pow_delta[1]; output |= 0x400; output = (output << 2) >> chip->op_shift[1]; if (chip->mode_test_21[1] & 16) output ^= 1 << 13; if (chip->op_sign[1] & 4) { output ^= 0x3fff; output++; } chip->op_output[0] = output; for (i = 0; i < 14; i++) { chip->op_op1[0][i][0] = chip->op_op1[0][i][1] << 1; chip->op_op1[1][i][0] = chip->op_op1[1][i][1] << 1; chip->op_op2[i][0] = chip->op_op2[i][1] << 1; if (chip->fsm_op1_sel) { chip->op_op1[0][i][0] |= (chip->op_output[1] >> i) & 1; chip->op_op1[1][i][0] |= (chip->op_op1[0][i][1] >> 5) & 1; } else { chip->op_op1[0][i][0] |= (chip->op_op1[0][i][1] >> 5) & 1; chip->op_op1[1][i][0] |= (chip->op_op1[1][i][1] >> 5) & 1; } if (chip->fsm_op2_sel) chip->op_op2[i][0] |= (chip->op_output[1] >> i) & 1; else chip->op_op2[i][0] |= (chip->op_op2[i][1] >> 5) & 1; } if (chip->alg_mod_op1_0) { for (i = 0; i < 14; i++) mod2 |= ((chip->op_op1[0][i][1] >> 5) & 1) << i; } if (chip->alg_mod_op1_1) { for (i = 0; i < 14; i++) mod1 |= ((chip->op_op1[1][i][1] >> 5) & 1) << i; } if (chip->alg_mod_op2) { for (i = 0; i < 14; i++) mod1 |= ((chip->op_op2[i][1] >> 5) & 1) << i; } if (chip->alg_mod_prev_0) { mod2 |= chip->op_output[1]; } if (chip->alg_mod_prev_1) { mod1 |= chip->op_output[1]; } if (mod1 & (1 << 13)) mod1 |= 1 << 14; if (mod2 & (1 << 13)) mod2 |= 1 << 14; mod = (mod1 + mod2) >> 1; mod &= 0x3fff; chip->op_mod_sum[0] = mod; chip->op_dofeedback[0] = chip->fsm_op2_sel; if (chip->op_dofeedback[1]) { for (i = 0; i < 3; i++) fb |= (chip->chan_fb[i][1] & 1) << i; if (!fb) mod = 0; else { mod = chip->op_mod_sum[1]; if (mod & (1 << 13)) mod |= ~0x3fff; mod = mod >> (9 - fb); } } else mod = chip->op_mod_sum[1]; for (i = 0; i < 10; i++) { chip->op_mod[i][0] = (chip->op_mod[i][1] << 1) | (mod & 1); mod >>= 1; } } void FMOPN2_Operator2(fmopn2_t *chip) { int i; for (i = 0; i < 10; i++) { chip->op_mod[i][1] = chip->op_mod[i][0]; } for (i = 0; i < 14; i++) { chip->op_op1[0][i][1] = chip->op_op1[0][i][0]; chip->op_op1[1][i][1] = chip->op_op1[1][i][0]; chip->op_op2[i][1] = chip->op_op2[i][0]; } chip->op_phase[1] = chip->op_phase[0]; chip->op_sign[1] = chip->op_sign[0]; chip->op_logsin_add_delta[1] = chip->op_logsin_add_delta[0]; chip->op_logsin_base[1] = chip->op_logsin_base[0]; chip->op_logsin_delta[1] = chip->op_logsin_delta[0]; chip->op_env[1] = chip->op_env[0]; chip->op_atten[1] = chip->op_atten[0]; chip->op_pow_add_delta[1] = chip->op_pow_add_delta[0]; chip->op_pow_base[1] = chip->op_pow_base[0]; chip->op_pow_delta[1] = chip->op_pow_delta[0]; chip->op_shift[1] = chip->op_shift[0]; chip->op_output[1] = chip->op_output[0]; chip->op_mod_sum[1] = chip->op_mod_sum[0]; chip->op_dofeedback[1] = chip->op_dofeedback[0]; } void FMOPN2_YM3438Accumulator1(fmopn2_t *chip) { int i; int sum; int inp = 0; int acc = 0; int test_dac = (chip->mode_test_2c[1] & 32) != 0; int load = test_dac || chip->fsm_op1_sel; int acc_clear = load && !test_dac; sum = test_dac; if (chip->alg_output && !test_dac) inp = (chip->op_output[1] >> 5) & 511; if (!acc_clear) for (i = 0; i < 9; i++) acc += ((chip->ch_accm[i][1] >> 5) & 1) << i; sum = test_dac + inp + acc; sum &= 511; if ((inp & 256) != 0 && (acc & 256) != 0 && (sum & 256) == 0) sum = 256; else if ((inp & 256) == 0 && (acc & 256) == 0 && (sum & 256) != 0) sum = 255; for (i = 0; i < 9; i++) chip->ch_accm[i][0] = (chip->ch_accm[i][1] << 1) | ((sum >> i) & 1); for (i = 0; i < 9; i++) { chip->ch_out[i][0] = chip->ch_out[i][1] << 1; if (load) chip->ch_out[i][0] |= (chip->ch_accm[i][1] >> 5) & 1; else chip->ch_out[i][0] |= (chip->ch_out[i][1] >> 5) & 1; } chip->ch_dac_load = chip->fsm_dac_load; chip->ch_out_debug[0] = chip->ch_out_dlatch; } void FMOPN2_YM3438Accumulator2(fmopn2_t* chip) { int i; int test_dac = (chip->mode_test_2c[1] & 32) != 0; int do_out = 0; int sign; int out; for (i = 0; i < 9; i++) { chip->ch_accm[i][1] = chip->ch_accm[i][0]; chip->ch_out[i][1] = chip->ch_out[i][0]; } if ((chip->fsm_dac_load && !chip->ch_dac_load) || test_dac) { chip->ch_out_dlatch = 0; if (chip->fsm_dac_out_sel || test_dac) { for (i = 0; i < 9; i++) chip->ch_out_dlatch |= ((chip->ch_out[i][1] >> 5) & 1) << i; } else { for (i = 0; i < 9; i++) chip->ch_out_dlatch |= ((chip->ch_out[i][1] >> 4) & 1) << i; } } if ((chip->fsm_dac_ch6 && chip->mode_dac_en[1]) || test_dac) { chip->dac_val = chip->mode_dac_data[1] << 1; chip->dac_val |= (chip->mode_test_2c[1] & 8) != 0; } else chip->dac_val = chip->ch_out_dlatch; if (chip->fsm_dac_load && !chip->ch_dac_load) { chip->ch_out_pan_dlatch = 0; if (chip->fsm_dac_out_sel) { for (i = 0; i < 2; i++) chip->ch_out_pan_dlatch |= (((chip->chan_pan[i][1] >> 5) & 1) ^ 1) << i; } else { for (i = 0; i < 2; i++) chip->ch_out_pan_dlatch |= (((chip->chan_pan[i][1] >> 4) & 1) ^ 1) << i; } } do_out = test_dac || !chip->fsm_dac_load; if (do_out && (chip->ch_out_pan_dlatch & 2) != 0) chip->out_l = chip->dac_val; else chip->out_l = 0; if (do_out && (chip->ch_out_pan_dlatch & 1) != 0) chip->out_r = chip->dac_val; else chip->out_r = 0; if (chip->out_l & 256) chip->out_l |= ~0x1ff; if (chip->out_r & 256) chip->out_r |= ~0x1ff; chip->ch_out_debug[1] = chip->ch_out_debug[0]; } void FMOPN2_YMF276Accumulator1(fmopn2_t *chip) { int i; int sum1; int sum2; int sum; int inp = 0; int accm = 0; int acc = 0; int c; int test_dac = (chip->mode_test_2c[1] & 32) != 0; int test_dac2 = (chip->mode_test_2c[1] & 8) != 0; int load = test_dac || chip->fsm_op1_sel; int acc_clear = load && !test_dac; int sel_dac = (chip->fsm_op1_sel_l2[1] & 16) != 0 && chip->fsm_op1_sel && chip->mode_dac_en[1]; int sel_fm = chip->fsm_op1_sel && !sel_dac; int out = 0; int pan = 0; int acc_l = 0; int acc_r = 0; for (i = 0; i < 14; i++) accm += ((chip->ch_accm[i][1] >> 5) & 1) << i; if (chip->alg_output && !test_dac) inp = chip->op_output[1] & 0x3fff; if (test_dac2) inp = 0x3fff; if (!acc_clear) acc = accm; sum1 = (acc & 31) + (inp & 31) + (test_dac && !test_dac2); c = ((sum1 & 32) != 0 || test_dac) && !test_dac2; sum2 = (acc >> 5) + (inp >> 5) + c; sum = ((sum2 & 511) << 5) | (sum1 & 31); if ((inp & 0x2000) != 0 && (acc & 0x2000) != 0 && (sum & 0x2000) == 0) sum = 0x2000; else if ((inp & 0x2000) == 0 && (acc & 0x2000) == 0 && (sum & 0x2000) != 0) sum = 0x1fff; for (i = 0; i < 14; i++) chip->ch_accm[i][0] = (chip->ch_accm[i][1] << 1) | ((sum >> i) & 1); for (i = 0; i < 9; i++) { chip->ch_out[i][0] = chip->ch_out[i][1] << 1; if (load) chip->ch_out[i][0] |= (chip->ch_accm[i+5][1] >> 5) & 1; else chip->ch_out[i][0] |= (chip->ch_out[i][1] >> 5) & 1; } chip->ch_dac_load = chip->fsm_dac_load; chip->ch_out_debug[0] = chip->ch_out_dlatch; chip->fsm_op1_sel_l2[0] = (chip->fsm_op1_sel_l2[1] << 1) | chip->fsm_op1_sel; chip->fsm_op1_sel_l3[0] = chip->fsm_op1_sel; if (sel_dac) out |= chip->mode_dac_data[1] << 6; if (sel_fm) out |= accm; if (out & 0x2000) out |= 0x1c000; for (i = 0; i < 2; i++) pan |= (((chip->chan_pan[i][1] >> 5) & 1) ^ 1) << i; chip->fsm_shifter_ctrl[0] = chip->fsm_shifter_ctrl[1] << 1; if (chip->fsm_op1_sel && !chip->fsm_op1_sel_l3[1]) { acc_l = 0; acc_r = 0; chip->fsm_shifter_ctrl[0] |= 1; } else { acc_l = chip->ch_accm_l[1]; acc_r = chip->ch_accm_r[1]; } chip->ch_accm_l[0] = acc_l + ((pan & 2) != 0 ? out : 0); chip->ch_accm_r[0] = acc_r + ((pan & 1) != 0 ? out : 0); } void FMOPN2_YMF276Accumulator2(fmopn2_t *chip) { int i; int test_dac = (chip->mode_test_2c[1] & 32) != 0; for (i = 0; i < 14; i++) { chip->ch_accm[i][1] = chip->ch_accm[i][0]; } for (i = 0; i < 9; i++) { chip->ch_out[i][1] = chip->ch_out[i][0]; } if ((chip->fsm_dac_load && !chip->ch_dac_load) || test_dac) { chip->ch_out_dlatch = 0; if (chip->fsm_dac_out_sel || test_dac) { for (i = 0; i < 9; i++) chip->ch_out_dlatch |= ((chip->ch_out[i][1] >> 5) & 1) << i; } else { for (i = 0; i < 9; i++) chip->ch_out_dlatch |= ((chip->ch_out[i][1] >> 4) & 1) << i; } } chip->ch_out_debug[1] = chip->ch_out_debug[0]; chip->fsm_op1_sel_l2[1] = chip->fsm_op1_sel_l2[0]; chip->fsm_op1_sel_l3[1] = chip->fsm_op1_sel_l3[0]; chip->ch_accm_l[1] = chip->ch_accm_l[0]; chip->ch_accm_r[1] = chip->ch_accm_r[0]; chip->fsm_shifter_ctrl[1] = chip->fsm_shifter_ctrl[0]; } void FMOPN2_Timers1(fmopn2_t *chip) { int time; int test_timers = (chip->mode_test_21[1] & 4) != 0; int reset; int subcnt; if (chip->timer_a_load_latch[1]) time = chip->mode_timer_a_reg[1]; else time = chip->timer_a_cnt[1]; if ((chip->timer_a_load_dlatch && chip->fsm_clock_timers1) || test_timers) time++; reset = chip->mode_timer_a_reset[1] || chip->input.ic; if (reset) chip->timer_a_status[0] = 0; else chip->timer_a_status[0] = chip->timer_a_status[1] || (chip->timer_a_of[1] && chip->mode_timer_a_enable[1]); chip->timer_a_load_old[0] = chip->timer_a_load_dlatch; chip->timer_a_load_latch[0] = (!chip->timer_a_load_old[1] && chip->timer_a_load_dlatch) || chip->timer_a_of[1]; if (!chip->timer_a_load_dlatch) chip->timer_a_cnt[0] = 0; else chip->timer_a_cnt[0] = time; chip->timer_a_of[0] = (time & 1024) != 0; subcnt = chip->timer_b_subcnt[1]; if (chip->fsm_clock_timers1) subcnt++; if (chip->input.ic) chip->timer_b_subcnt[0] = 0; else chip->timer_b_subcnt[0] = subcnt; chip->timer_b_subcnt_of[0] = (subcnt & 16) != 0; if (chip->timer_b_load_latch[1]) time = chip->mode_timer_b_reg[1]; else time = chip->timer_b_cnt[1]; if ((chip->timer_b_load_dlatch && chip->timer_b_subcnt_of[1]) || test_timers) time++; reset = chip->mode_timer_b_reset[1] || chip->input.ic; if (reset) chip->timer_b_status[0] = 0; else chip->timer_b_status[0] = chip->timer_b_status[1] || (chip->timer_b_of[1] && chip->mode_timer_b_enable[1]); chip->timer_b_load_old[0] = chip->timer_b_load_dlatch; chip->timer_b_load_latch[0] = (!chip->timer_b_load_old[1] && chip->timer_b_load_dlatch) || chip->timer_b_of[1]; if (!chip->timer_b_load_dlatch) chip->timer_b_cnt[0] = 0; else chip->timer_b_cnt[0] = time; chip->timer_b_of[0] = (time & 256) != 0; chip->timer_dlatch = chip->fsm_clock_timers; } void FMOPN2_Timers2(fmopn2_t *chip) { int read_enable = chip->input.cs && chip->input.rd && !chip->input.ic; chip->timer_a_load_latch[1] = chip->timer_a_load_latch[0]; chip->timer_a_load_old[1] = chip->timer_a_load_old[0]; chip->timer_a_cnt[1] = chip->timer_a_cnt[0] & 1023; chip->timer_a_of[1] = chip->timer_a_of[0]; chip->timer_a_status[1] = chip->timer_a_status[0]; chip->timer_b_subcnt[1] = chip->timer_b_subcnt[0] & 15; chip->timer_b_subcnt_of[1] = chip->timer_b_subcnt_of[0]; chip->timer_b_load_latch[1] = chip->timer_b_load_latch[0]; chip->timer_b_load_old[1] = chip->timer_b_load_old[0]; chip->timer_b_cnt[1] = chip->timer_b_cnt[0] & 255; chip->timer_b_of[1] = chip->timer_b_of[0]; chip->timer_b_status[1] = chip->timer_b_status[0]; if (!chip->timer_dlatch && chip->fsm_clock_timers) { chip->timer_a_load_dlatch = chip->mode_timer_a_load[1]; chip->timer_b_load_dlatch = chip->mode_timer_b_load[1]; chip->timer_csm_key_dlatch = chip->mode_ch3[1] == 2 && ((!chip->timer_a_load_old[1] && chip->timer_a_load_dlatch) || chip->timer_a_of[1]); } if (!read_enable) { chip->status_timer_a_dlatch = chip->timer_a_status[1]; chip->status_timer_b_dlatch = chip->timer_b_status[1]; } } void FMOPN2_ClockPhase1(fmopn2_t *chip) { FMOPN2_DoShiftRegisters(chip, 0); FMOPN2_HandleIO1(chip); FMOPN2_FMRegisters1(chip); FMOPN2_FSM1(chip); FMOPN2_Misc1(chip); FMOPN2_LFO1(chip); FMOPN2_PhaseGenerator1(chip); FMOPN2_EnvelopeGenerator1(chip); FMOPN2_Operator1(chip); if (!(chip->flags & fmopn2_flags_ym3438)) FMOPN2_YMF276Accumulator1(chip); else FMOPN2_YM3438Accumulator1(chip); FMOPN2_Timers1(chip); } void FMOPN2_ClockPhase2(fmopn2_t *chip) { FMOPN2_DoShiftRegisters(chip, 1); FMOPN2_HandleIO2(chip); FMOPN2_FMRegisters2(chip); FMOPN2_FSM2(chip); FMOPN2_Misc2(chip); FMOPN2_LFO2(chip); FMOPN2_PhaseGenerator2(chip); FMOPN2_EnvelopeGenerator2(chip); FMOPN2_Operator2(chip); if (!(chip->flags & fmopn2_flags_ym3438)) FMOPN2_YMF276Accumulator2(chip); else FMOPN2_YM3438Accumulator2(chip); FMOPN2_Timers2(chip); } void FMOPN2_ClockFM(fmopn2_t *chip) { FMOPN2_HandleIO(chip); if (chip->i_phi1) { FMOPN2_ClockPhase1(chip); } if (chip->i_phi2) { FMOPN2_ClockPhase2(chip); } } void FMOPN2_Clock(fmopn2_t *chip, int clk) { chip->pinput.phi = clk; if (memcmp(&chip->pinput, &chip->pinput_old, sizeof(chip->pinput)) != 0) { FMOPN2_Prescaler(chip); chip->pinput_old = chip->pinput; chip->input.i_fsm_reset = chip->fsm_reset; if (chip->phi1_latch[1]) chip->input.phi_phase = 1; if (chip->phi2_latch[1]) chip->input.phi_phase = 2; chip->i_phi1 = chip->phi1_latch[1]; chip->i_phi2 = chip->phi2_latch[1]; } if (memcmp(&chip->input, &chip->input_old, sizeof(chip->input)) != 0) { FMOPN2_ClockFM(chip); chip->input_old = chip->input; } if (!(chip->flags & fmopn2_flags_ym3438)) FMOPN2_YMF276DAC(chip); }