306 lines
8.1 KiB
C++
306 lines
8.1 KiB
C++
/****************************************************************************
|
|
* *
|
|
* Project64 - A Nintendo 64 emulator. *
|
|
* http://www.pj64-emu.com/ *
|
|
* Copyright (C) 2016 Project64. All rights reserved. *
|
|
* *
|
|
* License: *
|
|
* GNU/GPLv2 http://www.gnu.org/licenses/gpl-2.0.html *
|
|
* *
|
|
****************************************************************************/
|
|
#include "stdafx.h"
|
|
#include <string.h>
|
|
|
|
#include "alist.h"
|
|
#include "common.h"
|
|
#include "hle.h"
|
|
#include "mem.h"
|
|
#include "ucodes.h"
|
|
|
|
enum { DMEM_BASE = 0x5c0 };
|
|
|
|
/* helper functions */
|
|
static uint32_t get_address(CHle * hle, uint32_t so)
|
|
{
|
|
return alist_get_address(hle, so, hle->alist_audio().segments, N_SEGMENTS);
|
|
}
|
|
|
|
static void set_address(CHle * hle, uint32_t so)
|
|
{
|
|
alist_set_address(hle, so, hle->alist_audio().segments, N_SEGMENTS);
|
|
}
|
|
|
|
static void clear_segments(CHle * hle)
|
|
{
|
|
memset(hle->alist_audio().segments, 0, N_SEGMENTS*sizeof(hle->alist_audio().segments[0]));
|
|
}
|
|
|
|
/* audio commands definition */
|
|
static void SPNOOP(CHle * UNUSED(hle), uint32_t UNUSED(w1), uint32_t UNUSED(w2))
|
|
{
|
|
}
|
|
|
|
static void CLEARBUFF(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint16_t dmem = (w1 + DMEM_BASE) & 0xFFFF;
|
|
uint16_t count = w2;
|
|
|
|
if (count == 0)
|
|
return;
|
|
|
|
alist_clear(hle, dmem, align(count, 16));
|
|
}
|
|
|
|
static void ENVMIXER(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16) & 0xFF;
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
alist_envmix_exp(
|
|
hle,
|
|
flags & A_INIT,
|
|
flags & A_AUX,
|
|
hle->alist_audio().out, hle->alist_audio().dry_right,
|
|
hle->alist_audio().wet_left, hle->alist_audio().wet_right,
|
|
hle->alist_audio().in, hle->alist_audio().count,
|
|
hle->alist_audio().dry, hle->alist_audio().wet,
|
|
hle->alist_audio().vol,
|
|
hle->alist_audio().target,
|
|
hle->alist_audio().rate,
|
|
address);
|
|
}
|
|
|
|
static void ENVMIXER_GE(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16);
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
alist_envmix_ge(
|
|
hle,
|
|
flags & A_INIT,
|
|
flags & A_AUX,
|
|
hle->alist_audio().out, hle->alist_audio().dry_right,
|
|
hle->alist_audio().wet_left, hle->alist_audio().wet_right,
|
|
hle->alist_audio().in, hle->alist_audio().count,
|
|
hle->alist_audio().dry, hle->alist_audio().wet,
|
|
hle->alist_audio().vol,
|
|
hle->alist_audio().target,
|
|
hle->alist_audio().rate,
|
|
address);
|
|
}
|
|
|
|
static void RESAMPLE(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16) & 0xFF;
|
|
uint16_t pitch = w1 & 0xFFFF;
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
alist_resample(
|
|
hle,
|
|
flags & 0x1,
|
|
flags & 0x2,
|
|
hle->alist_audio().out,
|
|
hle->alist_audio().in,
|
|
align(hle->alist_audio().count, 16),
|
|
pitch << 1,
|
|
address);
|
|
}
|
|
|
|
static void SETVOL(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16) & 0xFF;
|
|
|
|
if (flags & A_AUX)
|
|
{
|
|
hle->alist_audio().dry = w1 & 0xFFFF;
|
|
hle->alist_audio().wet = w2 & 0xFFFF;
|
|
}
|
|
else
|
|
{
|
|
unsigned lr = (flags & A_LEFT) ? 0 : 1;
|
|
|
|
if (flags & A_VOL)
|
|
{
|
|
hle->alist_audio().vol[lr] = w1 & 0xFFFF;
|
|
}
|
|
else
|
|
{
|
|
hle->alist_audio().target[lr] = w1 & 0xFFFF;
|
|
hle->alist_audio().rate[lr] = w2;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void SETLOOP(CHle * hle, uint32_t UNUSED(w1), uint32_t w2)
|
|
{
|
|
hle->alist_audio().loop = get_address(hle, w2);
|
|
}
|
|
|
|
static void ADPCM(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16) & 0xFF;
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
alist_adpcm(
|
|
hle,
|
|
flags & 0x1,
|
|
flags & 0x2,
|
|
false, /* unsupported in this ucode */
|
|
hle->alist_audio().out,
|
|
hle->alist_audio().in,
|
|
align(hle->alist_audio().count, 32),
|
|
hle->alist_audio().table,
|
|
hle->alist_audio().loop,
|
|
address);
|
|
}
|
|
|
|
static void LOADBUFF(CHle * hle, uint32_t UNUSED(w1), uint32_t w2)
|
|
{
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
if (hle->alist_audio().count == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
alist_load(hle, hle->alist_audio().in, address, hle->alist_audio().count);
|
|
}
|
|
|
|
static void SAVEBUFF(CHle * hle, uint32_t UNUSED(w1), uint32_t w2)
|
|
{
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
if (hle->alist_audio().count == 0)
|
|
{
|
|
return;
|
|
}
|
|
alist_save(hle, hle->alist_audio().out, address, hle->alist_audio().count);
|
|
}
|
|
|
|
static void SETBUFF(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16) & 0xFF;
|
|
|
|
if (flags & A_AUX)
|
|
{
|
|
hle->alist_audio().dry_right = (w1 + DMEM_BASE) & 0xFFFF;
|
|
hle->alist_audio().wet_left = (w2 >> 16) + DMEM_BASE;
|
|
hle->alist_audio().wet_right = (w2 + DMEM_BASE) & 0xFFFF;
|
|
}
|
|
else
|
|
{
|
|
hle->alist_audio().in = (w1 + DMEM_BASE) & 0xFFFF;
|
|
hle->alist_audio().out = ((w2 >> 16) + DMEM_BASE) & 0xFFFF;
|
|
hle->alist_audio().count = w2 & 0xFFFF;
|
|
}
|
|
}
|
|
|
|
static void DMEMMOVE(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint16_t dmemi = (w1 + DMEM_BASE) & 0xFFFF;
|
|
uint16_t dmemo = (w2 >> 16) + DMEM_BASE;
|
|
uint16_t count = (w2)& 0xFFFF;
|
|
|
|
if (count == 0)
|
|
return;
|
|
|
|
alist_move(hle, dmemo, dmemi, align(count, 16));
|
|
}
|
|
|
|
static void LOADADPCM(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint16_t count = (w1 & 0xFFFF);
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
dram_load_u16(hle, (uint16_t*)hle->alist_audio().table, address, align(count, 8) >> 1);
|
|
}
|
|
|
|
static void INTERLEAVE(CHle * hle, uint32_t UNUSED(w1), uint32_t w2)
|
|
{
|
|
uint16_t left = (w2 >> 16) + DMEM_BASE;
|
|
uint16_t right = (w2 + DMEM_BASE) & 0xFFFF;
|
|
|
|
if (hle->alist_audio().count == 0)
|
|
return;
|
|
|
|
alist_interleave(hle, hle->alist_audio().out, left, right, align(hle->alist_audio().count, 16));
|
|
}
|
|
|
|
static void MIXER(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
int16_t gain = (w1)& 0xFFFF;
|
|
uint16_t dmemi = ((w2 >> 16) + DMEM_BASE) & 0xFFFF;
|
|
uint16_t dmemo = (w2 + DMEM_BASE) & 0xFFFF;
|
|
|
|
if (hle->alist_audio().count == 0)
|
|
return;
|
|
|
|
alist_mix(hle, dmemo, dmemi, align(hle->alist_audio().count, 32), gain);
|
|
}
|
|
|
|
static void SEGMENT(CHle * hle, uint32_t UNUSED(w1), uint32_t w2)
|
|
{
|
|
set_address(hle, w2);
|
|
}
|
|
|
|
static void POLEF(CHle * hle, uint32_t w1, uint32_t w2)
|
|
{
|
|
uint8_t flags = (w1 >> 16);
|
|
uint16_t gain = w1;
|
|
uint32_t address = get_address(hle, w2);
|
|
|
|
if (hle->alist_audio().count == 0)
|
|
return;
|
|
|
|
alist_polef(
|
|
hle,
|
|
flags & A_INIT,
|
|
hle->alist_audio().out,
|
|
hle->alist_audio().in,
|
|
align(hle->alist_audio().count, 16),
|
|
gain,
|
|
hle->alist_audio().table,
|
|
address);
|
|
}
|
|
|
|
/* global functions */
|
|
void alist_process_audio(CHle * hle)
|
|
{
|
|
static const acmd_callback_t ABI[0x10] = {
|
|
SPNOOP, ADPCM, CLEARBUFF, ENVMIXER,
|
|
LOADBUFF, RESAMPLE, SAVEBUFF, SEGMENT,
|
|
SETBUFF, SETVOL, DMEMMOVE, LOADADPCM,
|
|
MIXER, INTERLEAVE, POLEF, SETLOOP
|
|
};
|
|
|
|
clear_segments(hle);
|
|
alist_process(hle, ABI, 0x10);
|
|
}
|
|
|
|
void alist_process_audio_ge(CHle * hle)
|
|
{
|
|
static const acmd_callback_t ABI[0x10] =
|
|
{
|
|
SPNOOP, ADPCM, CLEARBUFF, ENVMIXER_GE,
|
|
LOADBUFF, RESAMPLE, SAVEBUFF, SEGMENT,
|
|
SETBUFF, SETVOL, DMEMMOVE, LOADADPCM,
|
|
MIXER, INTERLEAVE, POLEF, SETLOOP
|
|
};
|
|
|
|
clear_segments(hle);
|
|
alist_process(hle, ABI, 0x10);
|
|
}
|
|
|
|
void alist_process_audio_bc(CHle * hle)
|
|
{
|
|
static const acmd_callback_t ABI[0x10] =
|
|
{
|
|
SPNOOP, ADPCM, CLEARBUFF, ENVMIXER_GE,
|
|
LOADBUFF, RESAMPLE, SAVEBUFF, SEGMENT,
|
|
SETBUFF, SETVOL, DMEMMOVE, LOADADPCM,
|
|
MIXER, INTERLEAVE, POLEF, SETLOOP
|
|
};
|
|
|
|
clear_segments(hle);
|
|
alist_process(hle, ABI, 0x10);
|
|
} |