mirror of
https://github.com/tildearrow/furnace.git
synced 2024-11-22 12:35:11 +00:00
450 lines
11 KiB
C
450 lines
11 KiB
C
// Copyright (C) 2021 Nuke.YKT
|
|
// License: GPLv2+
|
|
// Version 1.0.1
|
|
#include <string.h>
|
|
#include "ympsg.h"
|
|
|
|
const float ympsg_vol[17] = {
|
|
1.0, 0.772, 0.622, 0.485, 0.382, 0.29, 0.229, 0.174, 0.132, 0.096, 0.072, 0.051, 0.034, 0.019, 0.009, 0.0, -1.059
|
|
};
|
|
|
|
const float tipsg_vol[17] = {
|
|
1.0, 0.794, 0.631, 0.501, 0.398, 0.316, 0.251, 0.2, 0.158, 0.126, 0.1, 0.079, 0.063, 0.05, 0.04, 0.0, -1.059
|
|
};
|
|
|
|
static void YMPSG_WriteLatch(ympsg_t *chip)
|
|
{
|
|
uint8_t data = chip->data;
|
|
if (chip->data_mask)
|
|
{
|
|
data = 0;
|
|
}
|
|
if (data & 128)
|
|
{
|
|
chip->latch = (data >> 4) & 7;
|
|
}
|
|
}
|
|
|
|
static void YMPSG_UpdateRegisters(ympsg_t* chip)
|
|
{
|
|
uint8_t data = chip->data;
|
|
if (chip->data_mask)
|
|
{
|
|
data = 0;
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 1))
|
|
{
|
|
chip->volume[0] = data & 15;
|
|
if (chip->data_mask)
|
|
{
|
|
chip->volume[0] = 15;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 3))
|
|
{
|
|
chip->volume[1] = data & 15;
|
|
if (chip->data_mask)
|
|
{
|
|
chip->volume[1] = 15;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 4))
|
|
{
|
|
if ((data & 128) || chip->reg_reset)
|
|
{
|
|
chip->freq[2] &= 1008;
|
|
chip->freq[2] |= data & 15;
|
|
}
|
|
if (!(data & 128))
|
|
{
|
|
chip->freq[2] &= 15;
|
|
chip->freq[2] |= (data << 4) & 1008;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 2))
|
|
{
|
|
if ((data & 128) || chip->reg_reset)
|
|
{
|
|
chip->freq[1] &= 1008;
|
|
chip->freq[1] |= data & 15;
|
|
}
|
|
if (!(data & 128))
|
|
{
|
|
chip->freq[1] &= 15;
|
|
chip->freq[1] |= (data << 4) & 1008;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 5))
|
|
{
|
|
chip->volume[2] = data & 15;
|
|
if (chip->data_mask)
|
|
{
|
|
chip->volume[2] = 15;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 0))
|
|
{
|
|
if ((data & 128) || chip->reg_reset)
|
|
{
|
|
chip->freq[0] &= 1008;
|
|
chip->freq[0] |= data & 15;
|
|
}
|
|
if (!(data & 128))
|
|
{
|
|
chip->freq[0] &= 15;
|
|
chip->freq[0] |= (data << 4) & 1008;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 7))
|
|
{
|
|
chip->volume[3] = data & 15;
|
|
if (chip->data_mask)
|
|
{
|
|
chip->volume[3] = 15;
|
|
}
|
|
}
|
|
if (chip->reg_reset || (chip->write_flag_l && chip->latch == 6))
|
|
{
|
|
chip->noise_data = data & 7;
|
|
chip->noise_trig = 1;
|
|
}
|
|
}
|
|
|
|
static void YMPSG_ClockInternal1(ympsg_t *chip)
|
|
{
|
|
uint16_t freq = 0;
|
|
uint8_t chan_sel = chip->chan_sel;
|
|
uint8_t noise_of, noise_bit1, noise_bit2, noise_next;
|
|
if ((chip->noise_data & 3) == 3)
|
|
{
|
|
noise_of = (chip->sign >> 1) & 1;
|
|
}
|
|
else
|
|
{
|
|
noise_of = chip->sign & 1;
|
|
}
|
|
if (chip->noise_trig_l || (chip->ic_latch2 & 1))
|
|
{
|
|
chip->noise = 0;
|
|
}
|
|
else if (noise_of && !chip->noise_of)
|
|
{
|
|
noise_bit1 = (chip->noise >> chip->noise_tap2) & 1;
|
|
noise_bit2 = (chip->noise >> chip->noise_tap1) & 1;
|
|
noise_bit1 ^= noise_bit2;
|
|
noise_next = ((noise_bit1 && ((chip->noise_data >> 2) & 1)) || ((chip->noise & chip->noise_size) == 0));
|
|
chip->noise <<= 1;
|
|
chip->noise |= noise_next;
|
|
}
|
|
chip->noise_of = noise_of;
|
|
if (chip->ic_latch2 & 2)
|
|
{
|
|
chan_sel = 0;
|
|
}
|
|
if (chip->chan_sel & 1)
|
|
{
|
|
freq |= chip->freq[0];
|
|
}
|
|
if (chip->chan_sel & 2)
|
|
{
|
|
freq |= chip->freq[1];
|
|
}
|
|
if (chip->chan_sel & 4)
|
|
{
|
|
freq |= chip->freq[2];
|
|
}
|
|
if (chip->chan_sel & 8)
|
|
{
|
|
if ((chip->noise_data & 3) == 0)
|
|
{
|
|
freq |= 16;
|
|
}
|
|
if ((chip->noise_data & 3) == 1)
|
|
{
|
|
freq |= 32;
|
|
}
|
|
if ((chip->noise_data & 3) == 2)
|
|
{
|
|
freq |= 64;
|
|
}
|
|
}
|
|
if (chip->chan_sel & 1)
|
|
{
|
|
chip->sign ^= chip->counter_of & 15;
|
|
}
|
|
if (chip->ic_latch2 & 2)
|
|
{
|
|
chip->sign = 0;
|
|
}
|
|
chip->counter_of <<= 1;
|
|
if (chip->counter[chip->rot] >= freq)
|
|
{
|
|
chip->counter_of |= 1;
|
|
chip->counter[chip->rot] = 0;
|
|
}
|
|
if (chip->ic_latch2 & 2)
|
|
{
|
|
chip->counter[chip->rot] = 0;
|
|
}
|
|
chip->counter[chip->rot]++;
|
|
chip->counter[chip->rot] &= 1023;
|
|
if ((chip->ic_latch2 & 1) || (chip->chan_sel & 7) != 0)
|
|
{
|
|
chip->chan_sel <<= 1;
|
|
}
|
|
else
|
|
{
|
|
chip->chan_sel <<= 1;
|
|
chip->chan_sel |= 1;
|
|
}
|
|
chip->ic_latch2 <<= 1;
|
|
chip->ic_latch2 |= chip->ic & 1;
|
|
chip->noise_trig_l = chip->noise_trig;
|
|
}
|
|
|
|
static void YMPSG_ClockInternal2(ympsg_t *chip)
|
|
{
|
|
chip->data_mask = (chip->ic_latch2 >> 1) & 1;
|
|
chip->reg_reset = (chip->ic_latch2 >> 0) & 1;
|
|
if (chip->noise_trig_l)
|
|
{
|
|
chip->noise_trig = 0;
|
|
}
|
|
YMPSG_UpdateRegisters(chip);
|
|
|
|
chip->rot = (chip->rot + 1) & 3;
|
|
chip->sign_l = chip->sign;
|
|
chip->noise_sign_l = (chip->noise >> 14) & 1;
|
|
}
|
|
|
|
static void YMPSG_UpdateSample(ympsg_t *chip)
|
|
{
|
|
uint32_t i;
|
|
uint8_t sign = chip->sign & 14;
|
|
sign |= chip->noise_sign_l;
|
|
if (chip->test & 1)
|
|
{
|
|
sign |= 15;
|
|
}
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
if ((sign >> (3 - i)) & 1)
|
|
{
|
|
chip->volume_out[i] = chip->volume[i];
|
|
}
|
|
else
|
|
{
|
|
chip->volume_out[i] = 15;
|
|
}
|
|
}
|
|
}
|
|
|
|
void YMPSG_Write(ympsg_t *chip, uint8_t data)
|
|
{
|
|
chip->data = data;
|
|
chip->write_flag = 1;
|
|
}
|
|
|
|
void YMPSG_WriteStereo(ympsg_t *chip, uint8_t data)
|
|
{
|
|
chip->stereo = data;
|
|
}
|
|
|
|
uint16_t YMPSG_Read(ympsg_t *chip)
|
|
{
|
|
uint16_t data = 0;
|
|
uint32_t i;
|
|
YMPSG_UpdateSample(chip);
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
data |= chip->volume_out[i] << ((3 - i) * 4);
|
|
}
|
|
return data;
|
|
}
|
|
|
|
void YMPSG_Init(ympsg_t *chip, uint8_t real_sn, uint8_t noise_tap1, uint8_t noise_tap2, uint32_t noise_size)
|
|
{
|
|
uint32_t i;
|
|
memset(chip, 0, sizeof(ympsg_t));
|
|
YMPSG_SetIC(chip, 1);
|
|
chip->noise_tap1 = noise_tap1;
|
|
chip->noise_tap2 = noise_tap2;
|
|
chip->noise_size = noise_size;
|
|
chip->stereo = 0xff;
|
|
for (i = 0; i < 17; i++)
|
|
{
|
|
chip->vol_table[i]=(real_sn?tipsg_vol[i]:ympsg_vol[i]) * 8192.0f;
|
|
}
|
|
for (i = 0; i < 16; i++)
|
|
{
|
|
YMPSG_Clock(chip);
|
|
}
|
|
YMPSG_SetIC(chip, 0);
|
|
}
|
|
|
|
void YMPSG_SetIC(ympsg_t *chip, uint32_t ic)
|
|
{
|
|
chip->ic = (uint8_t)ic;
|
|
}
|
|
|
|
void YMPSG_Clock(ympsg_t *chip)
|
|
{
|
|
uint8_t prescaler2_latch;
|
|
prescaler2_latch = chip->prescaler_2;
|
|
chip->prescaler_2 += chip->prescaler_1;
|
|
chip->prescaler_2 &= 1;
|
|
chip->prescaler_1 ^= 1;
|
|
if ((chip->ic_latch1 & 3) == 2)
|
|
{
|
|
chip->prescaler_1 = 0;
|
|
chip->prescaler_2 = 0;
|
|
}
|
|
chip->ic_latch1 <<= 1;
|
|
chip->ic_latch1 |= chip->ic & 1;
|
|
YMPSG_UpdateRegisters(chip);
|
|
chip->write_flag_l = 0;
|
|
if (chip->write_flag)
|
|
{
|
|
YMPSG_WriteLatch(chip);
|
|
chip->write_flag = 0;
|
|
chip->write_flag_l = 1;
|
|
}
|
|
if (chip->prescaler_1)
|
|
{
|
|
if (!prescaler2_latch)
|
|
{
|
|
YMPSG_ClockInternal1(chip);
|
|
}
|
|
else
|
|
{
|
|
YMPSG_ClockInternal2(chip);
|
|
}
|
|
}
|
|
}
|
|
|
|
void YMPSG_GetOutput(ympsg_t *chip, int* left, int* right)
|
|
{
|
|
int sample_left = 0;
|
|
int sample_right = 0;
|
|
uint32_t i;
|
|
YMPSG_UpdateSample(chip);
|
|
if (chip->test & 1)
|
|
{
|
|
sample_left += chip->vol_table[chip->volume_out[chip->test >> 1]];
|
|
sample_left += chip->vol_table[16] * 3;
|
|
sample_right += chip->vol_table[chip->volume_out[chip->test >> 1]];
|
|
sample_right += chip->vol_table[16] * 3;
|
|
}
|
|
else if (!chip->mute)
|
|
{
|
|
if (chip->stereo&(0x10)) {
|
|
sample_left += chip->vol_table[chip->volume_out[0]];
|
|
}
|
|
if (chip->stereo&(0x01)) {
|
|
sample_right += chip->vol_table[chip->volume_out[0]];
|
|
}
|
|
if (chip->stereo&(0x20)) {
|
|
sample_left += chip->vol_table[chip->volume_out[1]];
|
|
}
|
|
if (chip->stereo&(0x02)) {
|
|
sample_right += chip->vol_table[chip->volume_out[1]];
|
|
}
|
|
if (chip->stereo&(0x40)) {
|
|
sample_left += chip->vol_table[chip->volume_out[2]];
|
|
}
|
|
if (chip->stereo&(0x04)) {
|
|
sample_right += chip->vol_table[chip->volume_out[2]];
|
|
}
|
|
if (chip->stereo&(0x80)) {
|
|
sample_left += chip->vol_table[chip->volume_out[3]];
|
|
}
|
|
if (chip->stereo&(0x08)) {
|
|
sample_right += chip->vol_table[chip->volume_out[3]];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
if (!((chip->mute>>i) & 1)) {
|
|
if (chip->stereo&(0x10<<i)) {
|
|
sample_left += chip->vol_table[chip->volume_out[i]];
|
|
}
|
|
if (chip->stereo&(0x01<<i)) {
|
|
sample_right += chip->vol_table[chip->volume_out[i]];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
*left=sample_left;
|
|
*right=sample_right;
|
|
}
|
|
|
|
void YMPSG_Test(ympsg_t *chip, uint16_t test)
|
|
{
|
|
chip->test = (test >> 9) & 7;
|
|
}
|
|
|
|
/*
|
|
void YMPSG_Generate(ympsg_t *chip, int32_t *buf)
|
|
{
|
|
uint32_t i;
|
|
float out;
|
|
|
|
for (i = 0; i < 16; i++)
|
|
{
|
|
YMPSG_Clock(chip);
|
|
|
|
while (chip->writebuf[chip->writebuf_cur].time <= chip->writebuf_samplecnt)
|
|
{
|
|
if (!chip->writebuf[chip->writebuf_cur].stat)
|
|
{
|
|
break;
|
|
}
|
|
chip->writebuf[chip->writebuf_cur].stat = 0;
|
|
YMPSG_Write(chip, chip->writebuf[chip->writebuf_cur].data);
|
|
chip->writebuf_cur = (chip->writebuf_cur + 1) % YMPSG_WRITEBUF_SIZE;
|
|
}
|
|
chip->writebuf_samplecnt++;
|
|
}
|
|
out = YMPSG_GetOutput(chip);
|
|
*buf = (int32_t)(out * 8192.f);
|
|
}*/
|
|
|
|
void YMPSG_WriteBuffered(ympsg_t *chip, uint8_t data)
|
|
{
|
|
uint64_t time1, time2;
|
|
uint64_t skip;
|
|
|
|
if (chip->writebuf[chip->writebuf_last].stat)
|
|
{
|
|
YMPSG_Write(chip, chip->writebuf[chip->writebuf_last].data);
|
|
|
|
chip->writebuf_cur = (chip->writebuf_last + 1) % YMPSG_WRITEBUF_SIZE;
|
|
skip = chip->writebuf[chip->writebuf_last].time - chip->writebuf_samplecnt;
|
|
chip->writebuf_samplecnt = chip->writebuf[chip->writebuf_last].time;
|
|
while (skip--)
|
|
{
|
|
YMPSG_Clock(chip);
|
|
}
|
|
}
|
|
|
|
chip->writebuf[chip->writebuf_last].stat = 1;
|
|
chip->writebuf[chip->writebuf_last].data = data;
|
|
time1 = chip->writebuf_lasttime + YMPSG_WRITEBUF_DELAY;
|
|
time2 = chip->writebuf_samplecnt;
|
|
|
|
if (time1 < time2)
|
|
{
|
|
time1 = time2;
|
|
}
|
|
|
|
chip->writebuf[chip->writebuf_last].time = time1;
|
|
chip->writebuf_lasttime = time1;
|
|
chip->writebuf_last = (chip->writebuf_last + 1) % YMPSG_WRITEBUF_SIZE;
|
|
}
|
|
|
|
void YMPSG_SetMute(ympsg_t *chip, uint8_t mute)
|
|
{
|
|
chip->mute = mute;
|
|
}
|