2016-06-26 00:34:39 +02:00
|
|
|
/*PCem v0.8 by Tom Walker
|
|
|
|
|
|
|
|
|
|
AD1848 CODEC emulation (Windows Sound System compatible)*/
|
|
|
|
|
|
2017-09-25 04:31:20 -04:00
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <wchar.h>
|
2017-05-05 01:49:42 +02:00
|
|
|
#include <math.h>
|
2017-05-06 17:48:33 +02:00
|
|
|
#include "../ibm.h"
|
|
|
|
|
#include "../dma.h"
|
|
|
|
|
#include "../pic.h"
|
|
|
|
|
#include "../timer.h"
|
2016-06-26 00:34:39 +02:00
|
|
|
#include "sound.h"
|
2017-05-06 17:48:33 +02:00
|
|
|
#include "snd_ad1848.h"
|
|
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
|
|
|
|
|
static int ad1848_vols[64];
|
|
|
|
|
|
2017-05-06 17:48:33 +02:00
|
|
|
|
2016-06-26 00:34:39 +02:00
|
|
|
void ad1848_setirq(ad1848_t *ad1848, int irq)
|
|
|
|
|
{
|
|
|
|
|
ad1848->irq = irq;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ad1848_setdma(ad1848_t *ad1848, int dma)
|
|
|
|
|
{
|
|
|
|
|
ad1848->dma = dma;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t ad1848_read(uint16_t addr, void *p)
|
|
|
|
|
{
|
|
|
|
|
ad1848_t *ad1848 = (ad1848_t *)p;
|
|
|
|
|
uint8_t temp = 0xff;
|
|
|
|
|
switch (addr & 3)
|
|
|
|
|
{
|
|
|
|
|
case 0: /*Index*/
|
|
|
|
|
temp = ad1848->index | ad1848->trd | ad1848->mce;
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
temp = ad1848->regs[ad1848->index];
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
temp = ad1848->status;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return temp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ad1848_write(uint16_t addr, uint8_t val, void *p)
|
|
|
|
|
{
|
|
|
|
|
ad1848_t *ad1848 = (ad1848_t *)p;
|
|
|
|
|
double freq;
|
|
|
|
|
switch (addr & 3)
|
|
|
|
|
{
|
|
|
|
|
case 0: /*Index*/
|
|
|
|
|
ad1848->index = val & 0xf;
|
|
|
|
|
ad1848->trd = val & 0x20;
|
|
|
|
|
ad1848->mce = val & 0x40;
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
switch (ad1848->index)
|
|
|
|
|
{
|
|
|
|
|
case 8:
|
2017-10-09 01:48:36 +02:00
|
|
|
freq = (val & 1) ? 16934400LL : 24576000LL;
|
2016-06-26 00:34:39 +02:00
|
|
|
switch ((val >> 1) & 7)
|
|
|
|
|
{
|
|
|
|
|
case 0: freq /= 3072; break;
|
|
|
|
|
case 1: freq /= 1536; break;
|
|
|
|
|
case 2: freq /= 896; break;
|
|
|
|
|
case 3: freq /= 768; break;
|
|
|
|
|
case 4: freq /= 448; break;
|
|
|
|
|
case 5: freq /= 384; break;
|
|
|
|
|
case 6: freq /= 512; break;
|
|
|
|
|
case 7: freq /= 2560; break;
|
|
|
|
|
}
|
|
|
|
|
ad1848->freq = freq;
|
2017-10-09 01:48:36 +02:00
|
|
|
ad1848->timer_latch = (int64_t)((double)TIMER_USEC * (1000000.0 / (double)ad1848->freq));
|
2016-06-26 00:34:39 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 9:
|
|
|
|
|
ad1848->enable = ((val & 0x41) == 0x01);
|
|
|
|
|
if (!ad1848->enable)
|
|
|
|
|
ad1848->out_l = ad1848->out_r = 0;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 12:
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
case 14:
|
|
|
|
|
ad1848->count = ad1848->regs[15] | (val << 8);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
ad1848->regs[ad1848->index] = val;
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
ad1848->status &= 0xfe;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ad1848_speed_changed(ad1848_t *ad1848)
|
|
|
|
|
{
|
2017-10-09 01:48:36 +02:00
|
|
|
ad1848->timer_latch = (int64_t)((double)TIMER_USEC * (1000000.0 / (double)ad1848->freq));
|
2016-06-26 00:34:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ad1848_update(ad1848_t *ad1848)
|
|
|
|
|
{
|
|
|
|
|
for (; ad1848->pos < sound_pos_global; ad1848->pos++)
|
|
|
|
|
{
|
|
|
|
|
ad1848->buffer[ad1848->pos*2] = ad1848->out_l;
|
|
|
|
|
ad1848->buffer[ad1848->pos*2 + 1] = ad1848->out_r;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void ad1848_poll(void *p)
|
|
|
|
|
{
|
|
|
|
|
ad1848_t *ad1848 = (ad1848_t *)p;
|
|
|
|
|
|
|
|
|
|
if (ad1848->timer_latch)
|
|
|
|
|
ad1848->timer_count += ad1848->timer_latch;
|
|
|
|
|
else
|
|
|
|
|
ad1848->timer_count = TIMER_USEC;
|
|
|
|
|
|
|
|
|
|
ad1848_update(ad1848);
|
|
|
|
|
|
|
|
|
|
if (ad1848->enable)
|
|
|
|
|
{
|
|
|
|
|
int32_t temp;
|
|
|
|
|
|
|
|
|
|
switch (ad1848->regs[8] & 0x70)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: /*Mono, 8-bit PCM*/
|
|
|
|
|
ad1848->out_l = ad1848->out_r = (dma_channel_read(ad1848->dma) ^ 0x80) * 256;
|
|
|
|
|
break;
|
|
|
|
|
case 0x10: /*Stereo, 8-bit PCM*/
|
|
|
|
|
ad1848->out_l = (dma_channel_read(ad1848->dma) ^ 0x80) * 256;
|
|
|
|
|
ad1848->out_r = (dma_channel_read(ad1848->dma) ^ 0x80) * 256;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 0x40: /*Mono, 16-bit PCM*/
|
|
|
|
|
temp = dma_channel_read(ad1848->dma);
|
|
|
|
|
ad1848->out_l = ad1848->out_r = (dma_channel_read(ad1848->dma) << 8) | temp;
|
|
|
|
|
break;
|
|
|
|
|
case 0x50: /*Stereo, 16-bit PCM*/
|
|
|
|
|
temp = dma_channel_read(ad1848->dma);
|
|
|
|
|
ad1848->out_l = (dma_channel_read(ad1848->dma) << 8) | temp;
|
|
|
|
|
temp = dma_channel_read(ad1848->dma);
|
|
|
|
|
ad1848->out_r = (dma_channel_read(ad1848->dma) << 8) | temp;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ad1848->regs[6] & 0x80)
|
|
|
|
|
ad1848->out_l = 0;
|
|
|
|
|
else
|
|
|
|
|
ad1848->out_l = (ad1848->out_l * ad1848_vols[ad1848->regs[6] & 0x3f]) >> 16;
|
|
|
|
|
|
|
|
|
|
if (ad1848->regs[7] & 0x80)
|
|
|
|
|
ad1848->out_r = 0;
|
|
|
|
|
else
|
|
|
|
|
ad1848->out_r = (ad1848->out_r * ad1848_vols[ad1848->regs[7] & 0x3f]) >> 16;
|
|
|
|
|
|
|
|
|
|
if (ad1848->count < 0)
|
|
|
|
|
{
|
|
|
|
|
ad1848->count = ad1848->regs[15] | (ad1848->regs[14] << 8);
|
|
|
|
|
if (!(ad1848->status & 0x01))
|
|
|
|
|
{
|
|
|
|
|
ad1848->status |= 0x01;
|
|
|
|
|
if (ad1848->regs[0xa] & 2)
|
|
|
|
|
picint(1 << ad1848->irq);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ad1848->count--;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
ad1848->out_l = ad1848->out_r = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ad1848_init(ad1848_t *ad1848)
|
|
|
|
|
{
|
|
|
|
|
int c;
|
|
|
|
|
double attenuation;
|
|
|
|
|
|
|
|
|
|
ad1848->enable = 0;
|
|
|
|
|
|
|
|
|
|
ad1848->status = 0xcc;
|
|
|
|
|
ad1848->index = ad1848->trd = 0;
|
|
|
|
|
ad1848->mce = 0x40;
|
|
|
|
|
|
|
|
|
|
ad1848->regs[0] = ad1848->regs[1] = 0;
|
|
|
|
|
ad1848->regs[2] = ad1848->regs[3] = 0x80;
|
|
|
|
|
ad1848->regs[4] = ad1848->regs[5] = 0x80;
|
|
|
|
|
ad1848->regs[6] = ad1848->regs[7] = 0x80;
|
|
|
|
|
ad1848->regs[8] = 0;
|
|
|
|
|
ad1848->regs[9] = 0x08;
|
|
|
|
|
ad1848->regs[10] = ad1848->regs[11] = 0;
|
|
|
|
|
ad1848->regs[12] = 0xa;
|
|
|
|
|
ad1848->regs[13] = 0;
|
|
|
|
|
ad1848->regs[14] = ad1848->regs[15] = 0;
|
|
|
|
|
|
|
|
|
|
ad1848->out_l = 0;
|
|
|
|
|
ad1848->out_r = 0;
|
|
|
|
|
|
|
|
|
|
for (c = 0; c < 64; c++)
|
|
|
|
|
{
|
|
|
|
|
attenuation = 0.0;
|
|
|
|
|
if (c & 0x01) attenuation -= 1.5;
|
|
|
|
|
if (c & 0x02) attenuation -= 3.0;
|
|
|
|
|
if (c & 0x04) attenuation -= 6.0;
|
|
|
|
|
if (c & 0x08) attenuation -= 12.0;
|
|
|
|
|
if (c & 0x10) attenuation -= 24.0;
|
|
|
|
|
if (c & 0x20) attenuation -= 48.0;
|
|
|
|
|
|
|
|
|
|
attenuation = pow(10, attenuation / 10);
|
|
|
|
|
|
|
|
|
|
ad1848_vols[c] = (int)(attenuation * 65536);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
timer_add(ad1848_poll, &ad1848->timer_count, &ad1848->enable, ad1848);
|
|
|
|
|
}
|