299 lines
7.6 KiB
C++
299 lines
7.6 KiB
C++
// Project64 - A Nintendo 64 emulator
|
|
// https://www.pj64-emu.com/
|
|
// Copyright(C) 2001-2021 Project64.
|
|
// Copyright(C) 2015 Gilles Siberlin
|
|
// Copyright(C) 2007-2009 Richard Goedeken
|
|
// Copyright(C) 2007-2008 Ebenblues
|
|
// Copyright(C) 2003 JttL
|
|
// Copyright(C) 2002 Hacktarux
|
|
// GNU/GPLv2 licensed: https://gnu.org/licenses/gpl-2.0.html
|
|
|
|
#include <Common/Util.h>
|
|
#ifdef _WIN32
|
|
#include <Project64-audio/Driver/DirectSound.h>
|
|
#else
|
|
#include <Project64-audio/Driver/OpenSLES.h>
|
|
#endif
|
|
#include <Project64-plugin-spec/Audio.h>
|
|
#include "Version.h"
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "AudioSettings.h"
|
|
#include "trace.h"
|
|
#include "AudioMain.h"
|
|
#include "ConfigUI.h"
|
|
#include "SettingsID.h"
|
|
|
|
#ifdef _WIN32
|
|
void SetTimerResolution ( void );
|
|
#endif
|
|
|
|
#define ENDIAN_SWAP_BYTE (~0 & 0x7 & 3)
|
|
#define BES(address) ((address) ^ ENDIAN_SWAP_BYTE)
|
|
|
|
// Read header for type definition
|
|
AUDIO_INFO g_AudioInfo;
|
|
|
|
bool g_PluginInit = false;
|
|
bool g_romopen = false;
|
|
uint32_t g_Dacrate = 0, hack = 0;
|
|
|
|
#ifdef _WIN32
|
|
DirectSoundDriver * g_SoundDriver = nullptr;
|
|
#else
|
|
OpenSLESDriver * g_SoundDriver = nullptr;
|
|
#endif
|
|
|
|
enum SYSTEM_TYPE
|
|
{
|
|
SYSTEM_NTSC = 0,
|
|
SYSTEM_PAL = 1,
|
|
SYSTEM_MPAL = 2,
|
|
};
|
|
|
|
void PluginInit(void)
|
|
{
|
|
if (g_PluginInit)
|
|
{
|
|
return;
|
|
}
|
|
SetupTrace();
|
|
SetupAudioSettings();
|
|
StartTrace();
|
|
//SetTimerResolution();
|
|
g_PluginInit = true;
|
|
}
|
|
|
|
EXPORT void CALL PluginLoaded(void)
|
|
{
|
|
PluginInit();
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Called");
|
|
if (g_settings != nullptr)
|
|
{
|
|
g_settings->SetSyncViaAudioEnabled(true);
|
|
}
|
|
}
|
|
|
|
EXPORT void CALL AiDacrateChanged(int SystemType)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start (SystemType: %d)", SystemType);
|
|
if (!g_PluginInit)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceNotice, "Plugin has not been initialized");
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done");
|
|
return;
|
|
}
|
|
|
|
if (g_SoundDriver && g_Dacrate != *g_AudioInfo.AI_DACRATE_REG)
|
|
{
|
|
g_Dacrate = *g_AudioInfo.AI_DACRATE_REG & 0x00003FFF;
|
|
if (g_Dacrate != *g_AudioInfo.AI_DACRATE_REG)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceNotice, "Unknown/reserved bits in AI_DACRATE_REG set. 0x%08X", *g_AudioInfo.AI_DACRATE_REG);
|
|
}
|
|
|
|
uint32_t video_clock = 0, BufferSize = 0, Frequency = 0, divider = 0;
|
|
|
|
switch (SystemType)
|
|
{
|
|
case SYSTEM_NTSC: video_clock = 48681812; break;
|
|
case SYSTEM_PAL: video_clock = 49656530; break;
|
|
case SYSTEM_MPAL: video_clock = 48628316; break;
|
|
}
|
|
|
|
Frequency = (video_clock / (g_Dacrate + 1));
|
|
|
|
if (Frequency < 8000)
|
|
{
|
|
WriteTrace(TraceAudioDriver, TraceDebug, "Not audio data!");
|
|
return;
|
|
}
|
|
|
|
switch (g_settings->GetBuffer())
|
|
{
|
|
case 1: divider = 50; break;
|
|
case 2: divider = 45; break;
|
|
case 3: divider = 40; break;
|
|
case 4: divider = 30; break;
|
|
case 5: divider = 25; break;
|
|
case 6: divider = 20; break;
|
|
case 7: divider = 15; break;
|
|
}
|
|
|
|
BufferSize = (Frequency / divider) + 3 & ~0x3;
|
|
|
|
if (hack == 0x4248 /*BH*/ && SystemType != SYSTEM_PAL) BufferSize -= 16;
|
|
|
|
g_SoundDriver->AI_SetFrequency(Frequency, BufferSize);
|
|
}
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done");
|
|
}
|
|
|
|
EXPORT void CALL AiLenChanged(void)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start (DRAM_ADDR = 0x%X Len = 0x%X)", *g_AudioInfo.AI_DRAM_ADDR_REG, *g_AudioInfo.AI_LEN_REG);
|
|
if (g_SoundDriver && g_settings->AudioEnabled())
|
|
{
|
|
uint32_t Len = *g_AudioInfo.AI_LEN_REG & 0x3FFF8;
|
|
uint8_t * Buffer = (g_AudioInfo.RDRAM + (*g_AudioInfo.AI_DRAM_ADDR_REG & 0x00FFFFF8));
|
|
|
|
g_SoundDriver->AI_LenChanged(Buffer, Len);
|
|
}
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done");
|
|
}
|
|
|
|
EXPORT uint32_t CALL AiReadLength(void)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start");
|
|
uint32_t len = 0;
|
|
if (g_SoundDriver != nullptr)
|
|
{
|
|
*g_AudioInfo.AI_LEN_REG = g_SoundDriver->AI_ReadLength();
|
|
len = *g_AudioInfo.AI_LEN_REG;
|
|
}
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done (len: 0x%X)", len);
|
|
return len;
|
|
}
|
|
|
|
EXPORT void CALL AiUpdate(int32_t Wait)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start (Wait: %s)", Wait ? "true" : "false");
|
|
if (g_SoundDriver)
|
|
{
|
|
g_SoundDriver->AI_Update(Wait != 0);
|
|
}
|
|
else
|
|
{
|
|
pjutil::Sleep(1); // TODO: Fix this: Ai update appears to be problematic
|
|
}
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done");
|
|
}
|
|
|
|
EXPORT void CALL CloseDLL(void)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Called");
|
|
if (g_SoundDriver != nullptr)
|
|
{
|
|
g_SoundDriver->AI_Shutdown();
|
|
delete g_SoundDriver;
|
|
g_SoundDriver = nullptr;
|
|
}
|
|
CleanupAudioSettings();
|
|
StopTrace();
|
|
}
|
|
|
|
EXPORT void CALL DllAbout(void * /*hParent*/)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Called");
|
|
}
|
|
|
|
EXPORT void CALL DllConfig(void * hParent)
|
|
{
|
|
#ifdef _WIN32
|
|
ConfigAudio(hParent);
|
|
if (g_SoundDriver)
|
|
{
|
|
g_SoundDriver->SetVolume(g_settings->GetVolume());
|
|
}
|
|
#endif
|
|
}
|
|
|
|
EXPORT void CALL DllTest(void * /*hParent*/)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Called");
|
|
}
|
|
|
|
EXPORT void CALL GetDllInfo(PLUGIN_INFO * PluginInfo)
|
|
{
|
|
PluginInfo->Version = 0x0101;
|
|
PluginInfo->Type = PLUGIN_TYPE_AUDIO;
|
|
#ifdef _DEBUG
|
|
sprintf(PluginInfo->Name, "Project64 audio plugin (Debug): %s", VER_FILE_VERSION_STR);
|
|
#else
|
|
sprintf(PluginInfo->Name, "Project64 audio plugin: %s", VER_FILE_VERSION_STR);
|
|
#endif
|
|
PluginInfo->Reserved1 = false;
|
|
PluginInfo->Reserved2 = true;
|
|
}
|
|
|
|
EXPORT int32_t CALL InitiateAudio(AUDIO_INFO Audio_Info)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start");
|
|
if (g_SoundDriver != nullptr)
|
|
{
|
|
g_SoundDriver->AI_Shutdown();
|
|
delete g_SoundDriver;
|
|
}
|
|
#ifdef _WIN32
|
|
SetTimerResolution();
|
|
#endif
|
|
g_AudioInfo = Audio_Info;
|
|
#ifdef _WIN32
|
|
g_SoundDriver = new DirectSoundDriver;
|
|
#else
|
|
g_SoundDriver = new OpenSLESDriver;
|
|
#endif
|
|
const uint16_t cart_ID = 0x0000
|
|
| (g_AudioInfo.HEADER[BES(0x3C)] << 8)
|
|
| (g_AudioInfo.HEADER[BES(0x3D)] << 0)
|
|
;
|
|
hack = cart_ID;
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done (res: true)");
|
|
return true;
|
|
}
|
|
|
|
EXPORT void CALL RomOpen()
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start");
|
|
g_romopen = true;
|
|
g_settings->ReadSettings();
|
|
if (g_SoundDriver)
|
|
{
|
|
g_SoundDriver->AI_Startup();
|
|
}
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done");
|
|
}
|
|
|
|
EXPORT void CALL RomClosed(void)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Start");
|
|
g_Dacrate = 0;
|
|
if (g_SoundDriver)
|
|
{
|
|
g_SoundDriver->AI_Shutdown();
|
|
}
|
|
g_romopen = false;
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Done");
|
|
}
|
|
|
|
EXPORT void CALL ProcessAList(void)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Called");
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
#include <Windows.h>
|
|
#endif
|
|
|
|
extern "C" void UseUnregisteredSetting(int /*SettingID*/)
|
|
{
|
|
WriteTrace(TraceAudioInterface, TraceDebug, "Called");
|
|
#ifdef _WIN32
|
|
DebugBreak();
|
|
#endif
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
void SetTimerResolution(void)
|
|
{
|
|
HMODULE hMod = GetModuleHandle(L"ntdll.dll");
|
|
if (hMod != nullptr)
|
|
{
|
|
typedef LONG(NTAPI* tNtSetTimerResolution)(IN ULONG DesiredResolution, IN BOOLEAN SetResolution, OUT PULONG CurrentResolution);
|
|
tNtSetTimerResolution NtSetTimerResolution = (tNtSetTimerResolution)GetProcAddress(hMod, "NtSetTimerResolution");
|
|
ULONG CurrentResolution = 0;
|
|
NtSetTimerResolution(10000, TRUE, &CurrentResolution);
|
|
}
|
|
}
|
|
#endif
|