furnace/extern/YMF276-LLE/fmopn2.c
2023-11-25 17:46:46 -05:00

2342 lines
79 KiB
C

/*
* 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 <stdio.h>
#include <string.h>
#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);
}