Zelda HLE: Handle the smaller VPBs used by the NTSC IPL.

Damn you, Nintendo.
This commit is contained in:
Pierre Bourdon 2014-12-22 12:11:16 +01:00
parent 1807c113b5
commit 43882f3e6e
2 changed files with 77 additions and 16 deletions

View File

@ -27,19 +27,23 @@ enum ZeldaUCodeFlag
// If set, only consider 4 of the 6 non-Dolby mixing outputs. Early // If set, only consider 4 of the 6 non-Dolby mixing outputs. Early
// versions of the Zelda UCode only had 4. // versions of the Zelda UCode only had 4.
FOUR_MIXING_DESTS = 0x00000008, FOUR_MIXING_DESTS = 0x00000008,
// Handle smaller VPBs that are missing their 0x40-0x80 area. Very early
// versions of the Zelda UCode used 0x80 sized VPBs.
TINY_VPB = 0x00000010,
}; };
static const std::map<u32, u32> UCODE_FLAGS = { static const std::map<u32, u32> UCODE_FLAGS = {
// GameCube IPL/BIOS, NTSC.
{ 0x24B22038, LIGHT_PROTOCOL | FOUR_MIXING_DESTS | TINY_VPB },
// GameCube IPL/BIOS, PAL.
{ 0x6BA3B3EA, LIGHT_PROTOCOL | FOUR_MIXING_DESTS },
// The Legend of Zelda: The Wind Waker. // The Legend of Zelda: The Wind Waker.
{ 0x86840740, 0 }, { 0x86840740, 0 },
// The Legend of Zelda: Twilight Princess / GC. // The Legend of Zelda: Twilight Princess / GC.
{ 0x6CA33A6D, MAKE_DOLBY_LOUDER }, { 0x6CA33A6D, MAKE_DOLBY_LOUDER },
// Super Mario Galaxy. // Super Mario Galaxy.
{ 0xD643001F, NO_ARAM | MAKE_DOLBY_LOUDER }, { 0xD643001F, NO_ARAM | MAKE_DOLBY_LOUDER },
// GameCube IPL/BIOS, PAL.
{ 0x6BA3B3EA, LIGHT_PROTOCOL | FOUR_MIXING_DESTS },
// GameCube IPL/BIOS, NTSC.
{ 0x24B22038, LIGHT_PROTOCOL | FOUR_MIXING_DESTS },
// TODO: Other games that use this UCode (exhaustive list): // TODO: Other games that use this UCode (exhaustive list):
// * Animal Crossing (type ????, CRC ????) // * Animal Crossing (type ????, CRC ????)
@ -494,9 +498,6 @@ void ZeldaUCode::RenderAudio()
#pragma pack(push, 1) #pragma pack(push, 1)
struct ZeldaAudioRenderer::VPB struct ZeldaAudioRenderer::VPB
{ {
static const u16 SIZE_IN_WORDS = 0xC0;
static const u16 RW_SIZE_IN_WORDS = 0x80;
// If zero, skip processing this voice. // If zero, skip processing this voice.
u16 enabled; u16 enabled;
@ -700,7 +701,55 @@ struct ZeldaAudioRenderer::VPB
u16 base_address_l; u16 base_address_l;
DEFINE_32BIT_ACCESSOR(base_address, BaseAddress) DEFINE_32BIT_ACCESSOR(base_address, BaseAddress)
u16 padding[SIZE_IN_WORDS]; u16 padding[0xC0];
// These next two functions are terrible hacks used in order to support two
// different VPB sizes.
// Transforms from an NTSC-IPL type 0x80-sized VPB to a full size VPB.
void Uncompress()
{
u16* words = (u16*)this;
// RO part of the VPB is from 0x40-0x80 instead of 0x80-0xC0.
for (int i = 0; i < 0x40; ++i)
{
words[0x80 + i] = words[0x40 + i];
words[0x40 + i] = 0;
}
// AFC decoded samples are offset by 0x28.
for (int i = 0; i < 0x10; ++i)
{
words[0x58 + i] = words[0x30 + i];
words[0x30 + i] = 0;
}
// Most things are offset by 0x18 because no Dolby mixing.
for (int i = 0; i < 0x18; ++i)
{
words[0x30 + i] = words[0x18 + i];
words[0x18 + i] = 0;
}
}
// Transforms from a full size VPB to an NTSC-IPL 0x80-sized VPB.
void Compress()
{
u16* words = (u16*)this;
for (int i = 0; i < 0x18; ++i)
{
words[0x18 + i] = words[0x30 + i];
words[0x30 + i] = 0;
}
for (int i = 0; i < 0x10; ++i)
{
words[0x30 + i] = words[0x58 + i];
words[0x58 + i] = 0;
}
for (int i = 0; i < 0x40; ++i)
{
words[0x40 + i] = words[0x80 + i];
words[0x80 + i] = 0;
}
}
}; };
struct ReverbPB struct ReverbPB
@ -1026,7 +1075,7 @@ void ZeldaAudioRenderer::AddVoice(u16 voice_id)
if (!vpb.use_constant_sample) if (!vpb.use_constant_sample)
vpb.reset_vpb = false; vpb.reset_vpb = false;
StoreVPB(voice_id, vpb); StoreVPB(voice_id, &vpb);
} }
void ZeldaAudioRenderer::FinalizeFrame() void ZeldaAudioRenderer::FinalizeFrame()
@ -1058,20 +1107,32 @@ void ZeldaAudioRenderer::FetchVPB(u16 voice_id, VPB* vpb)
u16* vpb_words = (u16*)vpb; u16* vpb_words = (u16*)vpb;
u16* ram_vpbs = (u16*)HLEMemory_Get_Pointer(m_vpb_base_addr); u16* ram_vpbs = (u16*)HLEMemory_Get_Pointer(m_vpb_base_addr);
size_t base_idx = voice_id * VPB::SIZE_IN_WORDS; // A few versions of the UCode have VPB of size 0x80 (vs. the standard
for (size_t i = 0; i < VPB::SIZE_IN_WORDS; ++i) // 0xC0). The whole 0x40-0x80 part is gone. Handle that by moving things
// around.
size_t vpb_size = (m_flags & TINY_VPB) ? 0x80 : 0xC0;
size_t base_idx = voice_id * vpb_size;
for (size_t i = 0; i < vpb_size; ++i)
vpb_words[i] = Common::swap16(ram_vpbs[base_idx + i]); vpb_words[i] = Common::swap16(ram_vpbs[base_idx + i]);
if (m_flags & TINY_VPB)
vpb->Uncompress();
} }
void ZeldaAudioRenderer::StoreVPB(u16 voice_id, const VPB& vpb) void ZeldaAudioRenderer::StoreVPB(u16 voice_id, VPB* vpb)
{ {
const u16* vpb_words = (const u16*)&vpb; u16* vpb_words = (u16*)vpb;
u16* ram_vpbs = (u16*)HLEMemory_Get_Pointer(m_vpb_base_addr); u16* ram_vpbs = (u16*)HLEMemory_Get_Pointer(m_vpb_base_addr);
size_t base_idx = voice_id * VPB::SIZE_IN_WORDS; size_t vpb_size = (m_flags & TINY_VPB) ? 0x80 : 0xC0;
size_t base_idx = voice_id * vpb_size;
if (m_flags & TINY_VPB)
vpb->Compress();
// Only the first 0x80 words are transferred back - the rest is read-only. // Only the first 0x80 words are transferred back - the rest is read-only.
for (size_t i = 0; i < VPB::RW_SIZE_IN_WORDS; ++i) for (size_t i = 0; i < vpb_size - 0x40; ++i)
ram_vpbs[base_idx + i] = Common::swap16(vpb_words[i]); ram_vpbs[base_idx + i] = Common::swap16(vpb_words[i]);
} }

View File

@ -127,7 +127,7 @@ private:
// Base address where VPBs are stored linearly in RAM. // Base address where VPBs are stored linearly in RAM.
u32 m_vpb_base_addr; u32 m_vpb_base_addr;
void FetchVPB(u16 voice_id, VPB* vpb); void FetchVPB(u16 voice_id, VPB* vpb);
void StoreVPB(u16 voice_id, const VPB& vpb); void StoreVPB(u16 voice_id, VPB* vpb);
// Sine table transferred from MRAM. Contains sin(x) values for x in // Sine table transferred from MRAM. Contains sin(x) values for x in
// [0.0;pi/4] (sin(x) in [1.0;0.0]), in 1.15 fixed format. // [0.0;pi/4] (sin(x) in [1.0;0.0]), in 1.15 fixed format.