Zelda HLE: Introduce behavior flags to handle UCode version differences.
MAKE_DOLBY_LOUDER solves some of the volume issues that were happening in Zelda Twilight Princess and SMG1.
This commit is contained in:
parent
8b9b9f033a
commit
c033395e28
|
@ -2,7 +2,34 @@
|
||||||
// Licensed under GPLv2+
|
// Licensed under GPLv2+
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
// Games that use this UCode (exhaustive list):
|
#include "Core/ConfigManager.h"
|
||||||
|
#include "Core/HW/DSPHLE/MailHandler.h"
|
||||||
|
#include "Core/HW/DSPHLE/UCodes/UCodes.h"
|
||||||
|
#include "Core/HW/DSPHLE/UCodes/Zelda.h"
|
||||||
|
|
||||||
|
// These flags modify the behavior of the HLE implementation based on the UCode
|
||||||
|
// version. When introducing a new flag, please recheck the behavior of each
|
||||||
|
// UCode version.
|
||||||
|
enum ZeldaUCodeFlag
|
||||||
|
{
|
||||||
|
// UCode for Wii where no ARAM is present. Instead of using ARAM, DMAs from
|
||||||
|
// MRAM are used to transfer sound data.
|
||||||
|
NO_ARAM = 0x00000001,
|
||||||
|
|
||||||
|
// Multiply by two the computed Dolby positional volumes. Some UCodes do
|
||||||
|
// not do that (Zelda TWW for example), others do (Zelda TP, SMG).
|
||||||
|
MAKE_DOLBY_LOUDER = 0x00000002,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const std::map<u32, u32> UCODE_FLAGS = {
|
||||||
|
// The Legend of Zelda: The Wind Waker.
|
||||||
|
{ 0x86840740, 0 },
|
||||||
|
// The Legend of Zelda: Twilight Princess / GC.
|
||||||
|
{ 0x6CA33A6D, MAKE_DOLBY_LOUDER },
|
||||||
|
// Super Mario Galaxy.
|
||||||
|
{ 0xD643001F, NO_ARAM | MAKE_DOLBY_LOUDER },
|
||||||
|
|
||||||
|
// TODO: Other games that use this UCode (exhaustive list):
|
||||||
// * Animal Crossing (type ????, CRC ????)
|
// * Animal Crossing (type ????, CRC ????)
|
||||||
// * Donkey Kong Jungle Beat (type ????, CRC ????)
|
// * Donkey Kong Jungle Beat (type ????, CRC ????)
|
||||||
// * IPL (type ????, CRC ????)
|
// * IPL (type ????, CRC ????)
|
||||||
|
@ -10,24 +37,24 @@
|
||||||
// * Mario Kart: Double Dash!! (type ????, CRC ????)
|
// * Mario Kart: Double Dash!! (type ????, CRC ????)
|
||||||
// * Pikmin (type ????, CRC ????)
|
// * Pikmin (type ????, CRC ????)
|
||||||
// * Pikmin 2 (type ????, CRC ????)
|
// * Pikmin 2 (type ????, CRC ????)
|
||||||
// * Super Mario Galaxy (type Wii-DAC, CRC D643001F)
|
|
||||||
// * Super Mario Galaxy 2 (type ????, CRC ????)
|
// * Super Mario Galaxy 2 (type ????, CRC ????)
|
||||||
// * Super Mario Sunshine (type ????, CRC ????)
|
// * Super Mario Sunshine (type ????, CRC ????)
|
||||||
// * The Legend of Zelda: Four Swords Adventures (type ????, CRC ????)
|
// * The Legend of Zelda: Four Swords Adventures (type ????, CRC ????)
|
||||||
// * The Legend of Zelda: The Wind Waker (type DAC, CRC 86840740)
|
|
||||||
// * The Legend of Zelda: Twilight Princess / GC (type DAC, CRC 6CA33A6D)
|
|
||||||
// * The Legend of Zelda: Twilight Princess / Wii (type ????, CRC ????)
|
// * The Legend of Zelda: Twilight Princess / Wii (type ????, CRC ????)
|
||||||
|
};
|
||||||
#include "Core/ConfigManager.h"
|
|
||||||
#include "Core/HW/DSPHLE/MailHandler.h"
|
|
||||||
#include "Core/HW/DSPHLE/UCodes/UCodes.h"
|
|
||||||
#include "Core/HW/DSPHLE/UCodes/Zelda.h"
|
|
||||||
|
|
||||||
ZeldaUCode::ZeldaUCode(DSPHLE *dsphle, u32 crc)
|
ZeldaUCode::ZeldaUCode(DSPHLE *dsphle, u32 crc)
|
||||||
: UCodeInterface(dsphle, crc)
|
: UCodeInterface(dsphle, crc)
|
||||||
{
|
{
|
||||||
m_mail_handler.PushMail(DSP_INIT, true);
|
m_mail_handler.PushMail(DSP_INIT, true);
|
||||||
m_mail_handler.PushMail(0xF3551111); // handshake
|
m_mail_handler.PushMail(0xF3551111); // handshake
|
||||||
|
|
||||||
|
auto it = UCODE_FLAGS.find(crc);
|
||||||
|
if (it == UCODE_FLAGS.end())
|
||||||
|
PanicAlert("No flags definition found for Zelda CRC %08x", crc);
|
||||||
|
|
||||||
|
m_flags = it->second;
|
||||||
|
m_renderer.SetFlags(m_flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
ZeldaUCode::~ZeldaUCode()
|
ZeldaUCode::~ZeldaUCode()
|
||||||
|
@ -50,6 +77,7 @@ u32 ZeldaUCode::GetUpdateMs()
|
||||||
|
|
||||||
void ZeldaUCode::DoState(PointerWrap &p)
|
void ZeldaUCode::DoState(PointerWrap &p)
|
||||||
{
|
{
|
||||||
|
p.Do(m_flags);
|
||||||
p.Do(m_mail_current_state);
|
p.Do(m_mail_current_state);
|
||||||
p.Do(m_mail_expected_cmd_mails);
|
p.Do(m_mail_expected_cmd_mails);
|
||||||
|
|
||||||
|
@ -271,7 +299,7 @@ void ZeldaUCode::RunPendingCommands()
|
||||||
// because the Wii does not have an ARAM, so it simulates it with MRAM
|
// because the Wii does not have an ARAM, so it simulates it with MRAM
|
||||||
// and DMAs.
|
// and DMAs.
|
||||||
case 0x0E:
|
case 0x0E:
|
||||||
if (!IsWiiDAC())
|
if (!(m_flags & NO_ARAM))
|
||||||
PanicAlert("Setting base ARAM addr on non Wii DAC.");
|
PanicAlert("Setting base ARAM addr on non Wii DAC.");
|
||||||
m_renderer.SetARAMBaseAddr(Read32());
|
m_renderer.SetARAMBaseAddr(Read32());
|
||||||
SendCommandAck(CommandAck::STANDARD, sync);
|
SendCommandAck(CommandAck::STANDARD, sync);
|
||||||
|
@ -779,11 +807,12 @@ void ZeldaAudioRenderer::AddVoice(u16 voice_id)
|
||||||
s16 front_volume = m_sine_table[vpb.GetDolbyVoiceY() ^ 0x7F];
|
s16 front_volume = m_sine_table[vpb.GetDolbyVoiceY() ^ 0x7F];
|
||||||
|
|
||||||
// Compute volume for each quadrant.
|
// Compute volume for each quadrant.
|
||||||
|
u16 shift_factor = (m_flags & MAKE_DOLBY_LOUDER) ? 15 : 16;
|
||||||
s16 quadrant_volumes[4] = {
|
s16 quadrant_volumes[4] = {
|
||||||
(s16)((left_volume * front_volume) >> 16),
|
(s16)((left_volume * front_volume) >> shift_factor),
|
||||||
(s16)((left_volume * back_volume) >> 16),
|
(s16)((left_volume * back_volume) >> shift_factor),
|
||||||
(s16)((right_volume * front_volume) >> 16),
|
(s16)((right_volume * front_volume) >> shift_factor),
|
||||||
(s16)((right_volume * back_volume) >> 16),
|
(s16)((right_volume * back_volume) >> shift_factor),
|
||||||
};
|
};
|
||||||
|
|
||||||
// Compute the volume delta for each sample to match the difference
|
// Compute the volume delta for each sample to match the difference
|
||||||
|
@ -791,19 +820,19 @@ void ZeldaAudioRenderer::AddVoice(u16 voice_id)
|
||||||
s16 delta = vpb.dolby_volume_target - vpb.dolby_volume_current;
|
s16 delta = vpb.dolby_volume_target - vpb.dolby_volume_current;
|
||||||
s16 volume_deltas[4];
|
s16 volume_deltas[4];
|
||||||
for (size_t i = 0; i < 4; ++i)
|
for (size_t i = 0; i < 4; ++i)
|
||||||
volume_deltas[i] = ((u16)quadrant_volumes[i] * delta) >> 16;
|
volume_deltas[i] = ((u16)quadrant_volumes[i] * delta) >> shift_factor;
|
||||||
|
|
||||||
// Apply master volume to each quadrant.
|
// Apply master volume to each quadrant.
|
||||||
for (size_t i = 0; i < 4; ++i)
|
for (size_t i = 0; i < 4; ++i)
|
||||||
quadrant_volumes[i] = (quadrant_volumes[i] * vpb.dolby_volume_current) >> 16;
|
quadrant_volumes[i] = (quadrant_volumes[i] * vpb.dolby_volume_current) >> shift_factor;
|
||||||
|
|
||||||
// Compute reverb volume and ramp deltas.
|
// Compute reverb volume and ramp deltas.
|
||||||
s16 reverb_volumes[4], reverb_volume_deltas[4];
|
s16 reverb_volumes[4], reverb_volume_deltas[4];
|
||||||
s16 reverb_volume_factor = (vpb.dolby_volume_current * vpb.dolby_reverb_factor) >> 15;
|
s16 reverb_volume_factor = (vpb.dolby_volume_current * vpb.dolby_reverb_factor) >> (shift_factor - 1);
|
||||||
for (size_t i = 0; i < 4; ++i)
|
for (size_t i = 0; i < 4; ++i)
|
||||||
{
|
{
|
||||||
reverb_volumes[i] = (quadrant_volumes[i] * reverb_volume_factor) >> 15;
|
reverb_volumes[i] = (quadrant_volumes[i] * reverb_volume_factor) >> shift_factor;
|
||||||
reverb_volume_deltas[i] = (volume_deltas[i] * vpb.dolby_reverb_factor) >> 16;
|
reverb_volume_deltas[i] = (volume_deltas[i] * vpb.dolby_reverb_factor) >> shift_factor;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
|
@ -1351,6 +1380,7 @@ void ZeldaAudioRenderer::DownloadRawSamplesFromMRAM(
|
||||||
|
|
||||||
void ZeldaAudioRenderer::DoState(PointerWrap& p)
|
void ZeldaAudioRenderer::DoState(PointerWrap& p)
|
||||||
{
|
{
|
||||||
|
p.Do(m_flags);
|
||||||
p.Do(m_prepared);
|
p.Do(m_prepared);
|
||||||
|
|
||||||
p.Do(m_output_lbuf_addr);
|
p.Do(m_output_lbuf_addr);
|
||||||
|
@ -1373,9 +1403,10 @@ void ZeldaAudioRenderer::DoState(PointerWrap& p)
|
||||||
p.Do(m_sine_table);
|
p.Do(m_sine_table);
|
||||||
p.Do(m_afc_coeffs);
|
p.Do(m_afc_coeffs);
|
||||||
|
|
||||||
|
p.Do(m_aram_base_addr);
|
||||||
p.Do(m_vpb_base_addr);
|
p.Do(m_vpb_base_addr);
|
||||||
|
|
||||||
p.Do(m_reverb_pb_base_addr);
|
p.Do(m_reverb_pb_base_addr);
|
||||||
|
|
||||||
p.Do(m_reverb_pb_frames_count);
|
p.Do(m_reverb_pb_frames_count);
|
||||||
p.Do(m_buf_unk0_reverb_last8);
|
p.Do(m_buf_unk0_reverb_last8);
|
||||||
p.Do(m_buf_unk1_reverb_last8);
|
p.Do(m_buf_unk1_reverb_last8);
|
||||||
|
|
|
@ -15,6 +15,7 @@ public:
|
||||||
void AddVoice(u16 voice_id);
|
void AddVoice(u16 voice_id);
|
||||||
void FinalizeFrame();
|
void FinalizeFrame();
|
||||||
|
|
||||||
|
void SetFlags(u32 flags) { m_flags = flags; }
|
||||||
void SetSineTable(std::array<s16, 0x80>&& sine_table) { m_sine_table = sine_table; }
|
void SetSineTable(std::array<s16, 0x80>&& sine_table) { m_sine_table = sine_table; }
|
||||||
void SetConstPatterns(std::array<s16, 0x100>&& patterns) { m_const_patterns = patterns; }
|
void SetConstPatterns(std::array<s16, 0x100>&& patterns) { m_const_patterns = patterns; }
|
||||||
void SetResamplingCoeffs(std::array<s16, 0x100>&& coeffs) { m_resampling_coeffs = coeffs; }
|
void SetResamplingCoeffs(std::array<s16, 0x100>&& coeffs) { m_resampling_coeffs = coeffs; }
|
||||||
|
@ -31,6 +32,9 @@ public:
|
||||||
private:
|
private:
|
||||||
struct VPB;
|
struct VPB;
|
||||||
|
|
||||||
|
// See Zelda.cpp for the list of possible flags.
|
||||||
|
u32 m_flags;
|
||||||
|
|
||||||
// Utility functions for audio operations.
|
// Utility functions for audio operations.
|
||||||
|
|
||||||
// Apply volume to a buffer. The volume is a fixed point integer, usually
|
// Apply volume to a buffer. The volume is a fixed point integer, usually
|
||||||
|
@ -189,12 +193,11 @@ public:
|
||||||
|
|
||||||
void DoState(PointerWrap &p) override;
|
void DoState(PointerWrap &p) override;
|
||||||
|
|
||||||
bool IsWiiDAC() const
|
|
||||||
{
|
|
||||||
return m_crc == 0xd643001f;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
// Flags that alter the behavior of the UCode. See Zelda.cpp for complete
|
||||||
|
// list and explanation.
|
||||||
|
u32 m_flags;
|
||||||
|
|
||||||
// UCode state machine. The control flow in the Zelda UCode family is quite
|
// UCode state machine. The control flow in the Zelda UCode family is quite
|
||||||
// complex, using interrupt handlers heavily to handle incoming messages
|
// complex, using interrupt handlers heavily to handle incoming messages
|
||||||
// which, depending on the type, get handled immediately or are queued in a
|
// which, depending on the type, get handled immediately or are queued in a
|
||||||
|
|
Loading…
Reference in New Issue