project64/Source/Android/PluginRSP/alist_audio.cpp

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);
}