SB16: fixed output volume calculation.

ES1370: fixed and improved output volume calculation similar to SB16.
TODO: apply volume control to the FM output.
This commit is contained in:
Volker Ruppert 2015-03-31 18:01:52 +00:00
parent d93eeb98be
commit 10d79f2e9a
3 changed files with 34 additions and 22 deletions

View File

@ -504,10 +504,9 @@ void bx_es1370_c::write(Bit32u address, Bit32u value, unsigned io_len)
#endif // !BX_USE_ES1370_SMF #endif // !BX_USE_ES1370_SMF
Bit16u offset; Bit16u offset;
Bit32u shift, mask; Bit32u shift, mask;
Bit8u index, master_vol, dac_vol; Bit8u index;
bx_bool set_wave_vol = 0; bx_bool set_wave_vol = 0;
unsigned i; unsigned i;
float tmp_vol;
BX_DEBUG(("register write to address 0x%04x - value = 0x%08x", address, value)); BX_DEBUG(("register write to address 0x%04x - value = 0x%08x", address, value));
@ -588,17 +587,26 @@ void bx_es1370_c::write(Bit32u address, Bit32u value, unsigned io_len)
} }
if (set_wave_vol) { if (set_wave_vol) {
master_vol = (0x1f - (BX_ES1370_THIS s.codec_reg[0] & 0x1f)); BX_ES1370_THIS s.wave_vol = calc_output_volume(0x00, 0x02, 0);
dac_vol = (0x1f - (BX_ES1370_THIS s.codec_reg[2] & 0x1f)); BX_ES1370_THIS s.wave_vol |= calc_output_volume(0x01, 0x03, 1);
tmp_vol = (float)master_vol/31.0f*pow(10.0f, (float)(31-dac_vol)*-0.065f);
BX_ES1370_THIS s.wave_vol = (Bit8u)(255 * tmp_vol);
master_vol = (0x1f - (BX_ES1370_THIS s.codec_reg[1] & 0x1f));
dac_vol = (0x1f - (BX_ES1370_THIS s.codec_reg[3] & 0x1f));
tmp_vol = (float)master_vol/31.0f*pow(10.0f, (float)(31-dac_vol)*-0.065f);
BX_ES1370_THIS s.wave_vol |= ((Bit8u)(255 * tmp_vol) << 8);
} }
} }
Bit16u bx_es1370_c::calc_output_volume(Bit8u reg1, Bit8u reg2, bx_bool shift)
{
Bit8u vol1, vol2;
float fvol1, fvol2;
Bit16u result;
vol1 = (0x1f - (BX_ES1370_THIS s.codec_reg[reg1] & 0x1f));
vol2 = (0x1f - (BX_ES1370_THIS s.codec_reg[reg2] & 0x1f));
fvol1 = pow(10.0f, (float)(31-vol1)*-0.065f);
fvol2 = pow(10.0f, (float)(31-vol2)*-0.065f);
result = (Bit8u)(255 * fvol1 * fvol2);
if (shift) result <<= 8;
return result;
}
void bx_es1370_c::es1370_timer_handler(void *this_ptr) void bx_es1370_c::es1370_timer_handler(void *this_ptr)
{ {
bx_es1370_c *class_ptr = (bx_es1370_c *) this_ptr; bx_es1370_c *class_ptr = (bx_es1370_c *) this_ptr;

View File

@ -104,6 +104,7 @@ private:
BX_ES1370_SMF void run_channel(unsigned channel, int timer_id, Bit32u buflen); BX_ES1370_SMF void run_channel(unsigned channel, int timer_id, Bit32u buflen);
BX_ES1370_SMF void sendwavepacket(unsigned channel, Bit32u buflen, Bit8u *buffer); BX_ES1370_SMF void sendwavepacket(unsigned channel, Bit32u buflen, Bit8u *buffer);
BX_ES1370_SMF void closewaveoutput(); BX_ES1370_SMF void closewaveoutput();
BX_ES1370_SMF Bit16u calc_output_volume(Bit8u reg1, Bit8u reg2, bx_bool shift);
static void es1370_timer_handler(void *); static void es1370_timer_handler(void *);
void es1370_timer(void); void es1370_timer(void);

View File

@ -1589,13 +1589,14 @@ Bit16u bx_sb16_c::dma_write16(Bit16u *buffer, Bit16u maxlen)
Bit16u bx_sb16_c::calc_output_volume(Bit8u reg1, Bit8u reg2, bx_bool shift) Bit16u bx_sb16_c::calc_output_volume(Bit8u reg1, Bit8u reg2, bx_bool shift)
{ {
Bit8u vol1, vol2; Bit8u vol1, vol2;
float tmp_vol; float fvol1, fvol2;
Bit16u result; Bit16u result;
vol1 = (MIXER.reg[reg1] >> 3); vol1 = (MIXER.reg[reg1] >> 3);
vol2 = (MIXER.reg[reg2] >> 3); vol2 = (MIXER.reg[reg2] >> 3);
tmp_vol = (float)vol1/31.0f*pow(10.0f, (float)(31-vol2)*-0.065f); fvol1 = pow(10.0f, (float)(31-vol1)*-0.065f);
result = (Bit8u)(255 * tmp_vol); fvol2 = pow(10.0f, (float)(31-vol2)*-0.065f);
result = (Bit8u)(255 * fvol1 * fvol2);
if (shift) result <<= 8; if (shift) result <<= 8;
return result; return result;
} }
@ -1604,7 +1605,7 @@ Bit16u bx_sb16_c::calc_output_volume(Bit8u reg1, Bit8u reg2, bx_bool shift)
void bx_sb16_c::mixer_writedata(Bit32u value) void bx_sb16_c::mixer_writedata(Bit32u value)
{ {
int i; int i;
bx_bool set_output_vol = 0; Bit8u set_output_vol = 0;
// do some action depending on what register was written // do some action depending on what register was written
switch (MIXER.regindex) switch (MIXER.regindex)
@ -1630,7 +1631,7 @@ void bx_sb16_c::mixer_writedata(Bit32u value)
MIXER.reg[i] = 0x80; MIXER.reg[i] = 0x80;
MIXER.regindex = 0; // next mixer register read is register 0 MIXER.regindex = 0; // next mixer register read is register 0
set_output_vol = 1; set_output_vol = 3;
return; return;
case 0x04: // DAC level case 0x04: // DAC level
@ -1646,13 +1647,13 @@ void bx_sb16_c::mixer_writedata(Bit32u value)
case 0x22: // master volume case 0x22: // master volume
MIXER.reg[0x30] = (value & 0xf0) | 0x08; MIXER.reg[0x30] = (value & 0xf0) | 0x08;
MIXER.reg[0x31] = ((value & 0x0f) << 4) | 0x08; MIXER.reg[0x31] = ((value & 0x0f) << 4) | 0x08;
set_output_vol = 1; set_output_vol = 3;
break; break;
case 0x26: // FM level case 0x26: // FM level
MIXER.reg[0x34] = (value & 0xf0) | 0x08; MIXER.reg[0x34] = (value & 0xf0) | 0x08;
MIXER.reg[0x35] = ((value & 0x0f) << 4) | 0x08; MIXER.reg[0x35] = ((value & 0x0f) << 4) | 0x08;
set_output_vol = 1; set_output_vol = 2;
break; break;
case 0x28: // CD audio level case 0x28: // CD audio level
@ -1668,13 +1669,13 @@ void bx_sb16_c::mixer_writedata(Bit32u value)
case 0x30: // master volume left case 0x30: // master volume left
MIXER.reg[0x22] &= 0x0f; MIXER.reg[0x22] &= 0x0f;
MIXER.reg[0x22] |= (value & 0xf0); MIXER.reg[0x22] |= (value & 0xf0);
set_output_vol = 1; set_output_vol = 3;
break; break;
case 0x31: // master volume right case 0x31: // master volume right
MIXER.reg[0x22] &= 0xf0; MIXER.reg[0x22] &= 0xf0;
MIXER.reg[0x22] |= (value >> 4); MIXER.reg[0x22] |= (value >> 4);
set_output_vol = 1; set_output_vol = 3;
break; break;
case 0x32: // DAC level left case 0x32: // DAC level left
@ -1692,13 +1693,13 @@ void bx_sb16_c::mixer_writedata(Bit32u value)
case 0x34: // FM level left case 0x34: // FM level left
MIXER.reg[0x26] &= 0x0f; MIXER.reg[0x26] &= 0x0f;
MIXER.reg[0x26] |= (value & 0xf0); MIXER.reg[0x26] |= (value & 0xf0);
set_output_vol = 1; set_output_vol = 2;
break; break;
case 0x35: // FM level right case 0x35: // FM level right
MIXER.reg[0x26] &= 0xf0; MIXER.reg[0x26] &= 0xf0;
MIXER.reg[0x26] |= (value >> 4); MIXER.reg[0x26] |= (value >> 4);
set_output_vol = 1; set_output_vol = 2;
break; break;
case 0x36: // CD audio level left case 0x36: // CD audio level left
@ -1753,9 +1754,11 @@ void bx_sb16_c::mixer_writedata(Bit32u value)
// store the value // store the value
MIXER.reg[MIXER.regindex] = value; MIXER.reg[MIXER.regindex] = value;
if (set_output_vol) { if (set_output_vol & 1) {
DSP.dma.param.volume = calc_output_volume(0x30, 0x32, 0); DSP.dma.param.volume = calc_output_volume(0x30, 0x32, 0);
DSP.dma.param.volume |= calc_output_volume(0x31, 0x33, 1); DSP.dma.param.volume |= calc_output_volume(0x31, 0x33, 1);
}
if (set_output_vol & 2) {
BX_SB16_THIS fm_volume = calc_output_volume(0x30, 0x34, 0); BX_SB16_THIS fm_volume = calc_output_volume(0x30, 0x34, 0);
BX_SB16_THIS fm_volume |= calc_output_volume(0x31, 0x35, 1); BX_SB16_THIS fm_volume |= calc_output_volume(0x31, 0x35, 1);
} }