mirror of
https://github.com/hedge-dev/UnleashedRecomp.git
synced 2026-04-28 13:21:42 +00:00
SDL2 audio (again). (#52)
* Implement SDL2 audio (again). * Call timeBeginPeriod/timeEndPeriod. * Replace miniaudio with SDL mixer. * Queue audio samples in a separate thread. * Enable CMake option override policy & fix compilation error.
This commit is contained in:
parent
14d5a33a13
commit
18097752d9
18 changed files with 268 additions and 370 deletions
12
.gitmodules
vendored
12
.gitmodules
vendored
|
|
@ -16,9 +16,6 @@
|
||||||
[submodule "thirdparty/msdf-atlas-gen"]
|
[submodule "thirdparty/msdf-atlas-gen"]
|
||||||
path = thirdparty/msdf-atlas-gen
|
path = thirdparty/msdf-atlas-gen
|
||||||
url = https://github.com/Chlumsky/msdf-atlas-gen.git
|
url = https://github.com/Chlumsky/msdf-atlas-gen.git
|
||||||
[submodule "thirdparty/miniaudio"]
|
|
||||||
path = thirdparty/miniaudio
|
|
||||||
url = https://github.com/mackron/miniaudio
|
|
||||||
[submodule "thirdparty/vcpkg"]
|
[submodule "thirdparty/vcpkg"]
|
||||||
path = thirdparty/vcpkg
|
path = thirdparty/vcpkg
|
||||||
url = https://github.com/microsoft/vcpkg
|
url = https://github.com/microsoft/vcpkg
|
||||||
|
|
@ -58,9 +55,6 @@
|
||||||
[submodule "thirdparty/unordered_dense"]
|
[submodule "thirdparty/unordered_dense"]
|
||||||
path = thirdparty/unordered_dense
|
path = thirdparty/unordered_dense
|
||||||
url = https://github.com/martinus/unordered_dense.git
|
url = https://github.com/martinus/unordered_dense.git
|
||||||
[submodule "thirdparty/vorbis"]
|
[submodule "thirdparty/SDL_mixer"]
|
||||||
path = thirdparty/vorbis
|
path = thirdparty/SDL_mixer
|
||||||
url = https://gitlab.xiph.org/xiph/vorbis.git
|
url = https://github.com/libsdl-org/SDL_mixer
|
||||||
[submodule "thirdparty/ogg"]
|
|
||||||
path = thirdparty/ogg
|
|
||||||
url = https://gitlab.xiph.org/xiph/ogg.git
|
|
||||||
|
|
@ -137,7 +137,7 @@ set(SWA_APU_CXX_SOURCES
|
||||||
if (SWA_XAUDIO2)
|
if (SWA_XAUDIO2)
|
||||||
list(APPEND SWA_APU_CXX_SOURCES "apu/driver/xaudio_driver.cpp")
|
list(APPEND SWA_APU_CXX_SOURCES "apu/driver/xaudio_driver.cpp")
|
||||||
else()
|
else()
|
||||||
list(APPEND SWA_APU_CXX_SOURCES "apu/driver/miniaudio_driver.cpp")
|
list(APPEND SWA_APU_CXX_SOURCES "apu/driver/sdl2_driver.cpp")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(SWA_HID_CXX_SOURCES
|
set(SWA_HID_CXX_SOURCES
|
||||||
|
|
@ -212,7 +212,6 @@ set(SWA_THIRDPARTY_INCLUDES
|
||||||
"${SWA_THIRDPARTY_ROOT}/imgui"
|
"${SWA_THIRDPARTY_ROOT}/imgui"
|
||||||
"${SWA_THIRDPARTY_ROOT}/libmspack/libmspack/mspack"
|
"${SWA_THIRDPARTY_ROOT}/libmspack/libmspack/mspack"
|
||||||
"${SWA_THIRDPARTY_ROOT}/magic_enum/include"
|
"${SWA_THIRDPARTY_ROOT}/magic_enum/include"
|
||||||
"${SWA_THIRDPARTY_ROOT}/miniaudio"
|
|
||||||
"${SWA_THIRDPARTY_ROOT}/stb"
|
"${SWA_THIRDPARTY_ROOT}/stb"
|
||||||
"${SWA_THIRDPARTY_ROOT}/tiny-AES-c"
|
"${SWA_THIRDPARTY_ROOT}/tiny-AES-c"
|
||||||
"${SWA_THIRDPARTY_ROOT}/TinySHA1"
|
"${SWA_THIRDPARTY_ROOT}/TinySHA1"
|
||||||
|
|
@ -314,10 +313,11 @@ target_link_libraries(UnleashedRecomp PRIVATE
|
||||||
o1heap
|
o1heap
|
||||||
PowerUtils
|
PowerUtils
|
||||||
SDL2::SDL2-static
|
SDL2::SDL2-static
|
||||||
|
SDL2_mixer
|
||||||
tomlplusplus::tomlplusplus
|
tomlplusplus::tomlplusplus
|
||||||
UnleashedRecompLib
|
UnleashedRecompLib
|
||||||
Vorbis::vorbisfile
|
xxHash::xxhash
|
||||||
xxHash::xxhash
)
|
)
|
||||||
|
|
||||||
target_include_directories(UnleashedRecomp PRIVATE
|
target_include_directories(UnleashedRecomp PRIVATE
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}
|
${CMAKE_CURRENT_SOURCE_DIR}
|
||||||
|
|
|
||||||
|
|
@ -15,6 +15,11 @@ void App::Restart(std::vector<std::string> restartArgs)
|
||||||
void App::Exit()
|
void App::Exit()
|
||||||
{
|
{
|
||||||
Config::Save();
|
Config::Save();
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
timeEndPeriod(1);
|
||||||
|
#endif
|
||||||
|
|
||||||
std::_Exit(0);
|
std::_Exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,53 +0,0 @@
|
||||||
#include "miniaudio_driver.h"
|
|
||||||
#include <cpu/code_cache.h>
|
|
||||||
#include <cpu/guest_thread.h>
|
|
||||||
#include <cpu/guest_code.h>
|
|
||||||
#include <kernel/heap.h>
|
|
||||||
|
|
||||||
static PPCFunc* g_clientCallback{};
|
|
||||||
static uint32_t g_clientCallbackParam{}; // pointer in guest memory
|
|
||||||
static ma_device g_audioDevice{};
|
|
||||||
static std::unique_ptr<GuestThreadContext> g_audioCtx;
|
|
||||||
static uint32_t* g_audioOutput;
|
|
||||||
|
|
||||||
static void AudioCallback(ma_device* pDevice, void* pOutput, const void* pInput, ma_uint32 frameCount)
|
|
||||||
{
|
|
||||||
if (g_audioCtx == nullptr)
|
|
||||||
g_audioCtx = std::make_unique<GuestThreadContext>(0);
|
|
||||||
|
|
||||||
g_audioCtx->ppcContext.r3.u64 = g_clientCallbackParam;
|
|
||||||
g_audioOutput = reinterpret_cast<uint32_t*>(pOutput);
|
|
||||||
(*g_clientCallback)(g_audioCtx->ppcContext, reinterpret_cast<uint8_t*>(g_memory.base));
|
|
||||||
}
|
|
||||||
|
|
||||||
void XAudioInitializeSystem()
|
|
||||||
{
|
|
||||||
ma_device_config deviceConfig = ma_device_config_init(ma_device_type_playback);
|
|
||||||
deviceConfig.sampleRate = XAUDIO_SAMPLES_HZ;
|
|
||||||
deviceConfig.periodSizeInFrames = XAUDIO_NUM_SAMPLES;
|
|
||||||
deviceConfig.noPreSilencedOutputBuffer = true;
|
|
||||||
deviceConfig.dataCallback = AudioCallback;
|
|
||||||
deviceConfig.playback.format = ma_format_f32;
|
|
||||||
deviceConfig.playback.channels = XAUDIO_NUM_CHANNELS;
|
|
||||||
ma_device_init(nullptr, &deviceConfig, &g_audioDevice);
|
|
||||||
}
|
|
||||||
|
|
||||||
void XAudioRegisterClient(PPCFunc* callback, uint32_t param)
|
|
||||||
{
|
|
||||||
auto* pClientParam = static_cast<uint32_t*>(g_userHeap.Alloc(sizeof(param)));
|
|
||||||
ByteSwapInplace(param);
|
|
||||||
*pClientParam = param;
|
|
||||||
g_clientCallbackParam = g_memory.MapVirtual(pClientParam);
|
|
||||||
g_clientCallback = callback;
|
|
||||||
|
|
||||||
ma_device_start(&g_audioDevice);
|
|
||||||
}
|
|
||||||
|
|
||||||
void XAudioSubmitFrame(void* samples)
|
|
||||||
{
|
|
||||||
for (size_t i = 0; i < XAUDIO_NUM_SAMPLES; i++)
|
|
||||||
{
|
|
||||||
for (size_t j = 0; j < XAUDIO_NUM_CHANNELS; j++)
|
|
||||||
g_audioOutput[i * XAUDIO_NUM_CHANNELS + j] = ByteSwap(((uint32_t*)samples)[j * XAUDIO_NUM_SAMPLES + i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
128
UnleashedRecomp/apu/driver/sdl2_driver.cpp
Normal file
128
UnleashedRecomp/apu/driver/sdl2_driver.cpp
Normal file
|
|
@ -0,0 +1,128 @@
|
||||||
|
#include "sdl2_driver.h"
|
||||||
|
#include <cpu/code_cache.h>
|
||||||
|
#include <cpu/guest_thread.h>
|
||||||
|
#include <cpu/guest_code.h>
|
||||||
|
#include <kernel/heap.h>
|
||||||
|
|
||||||
|
static PPCFunc* g_clientCallback{};
|
||||||
|
static uint32_t g_clientCallbackParam{}; // pointer in guest memory
|
||||||
|
static SDL_AudioDeviceID g_audioDevice{};
|
||||||
|
static bool g_downMixToStereo;
|
||||||
|
|
||||||
|
void XAudioInitializeSystem()
|
||||||
|
{
|
||||||
|
SDL_SetHint(SDL_HINT_AUDIO_CATEGORY, "playback");
|
||||||
|
SDL_SetHint(SDL_HINT_AUDIO_DEVICE_APP_NAME, "Unleashed Recompiled");
|
||||||
|
SDL_InitSubSystem(SDL_INIT_AUDIO);
|
||||||
|
|
||||||
|
SDL_AudioSpec desired{}, obtained{};
|
||||||
|
desired.freq = XAUDIO_SAMPLES_HZ;
|
||||||
|
desired.format = AUDIO_F32SYS;
|
||||||
|
desired.channels = XAUDIO_NUM_CHANNELS;
|
||||||
|
desired.samples = XAUDIO_NUM_SAMPLES;
|
||||||
|
g_audioDevice = SDL_OpenAudioDevice(nullptr, 0, &desired, &obtained, SDL_AUDIO_ALLOW_CHANNELS_CHANGE);
|
||||||
|
|
||||||
|
if (obtained.channels != 2 && obtained.channels != XAUDIO_NUM_CHANNELS)
|
||||||
|
{
|
||||||
|
SDL_CloseAudioDevice(g_audioDevice);
|
||||||
|
g_audioDevice = SDL_OpenAudioDevice(nullptr, 0, &desired, &obtained, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_downMixToStereo = (obtained.channels == 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::unique_ptr<std::thread> g_audioThread;
|
||||||
|
|
||||||
|
static void AudioThread()
|
||||||
|
{
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
|
GuestThreadContext ctx(0);
|
||||||
|
|
||||||
|
size_t channels = g_downMixToStereo ? 2 : XAUDIO_NUM_CHANNELS;
|
||||||
|
|
||||||
|
constexpr double INTERVAL = double(XAUDIO_NUM_SAMPLES) / double(XAUDIO_SAMPLES_HZ);
|
||||||
|
auto start = std::chrono::steady_clock::now();
|
||||||
|
size_t iteration = 1;
|
||||||
|
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
uint32_t queuedAudioSize = SDL_GetQueuedAudioSize(g_audioDevice);
|
||||||
|
constexpr size_t MAX_LATENCY = 10;
|
||||||
|
const size_t callbackAudioSize = channels * XAUDIO_NUM_SAMPLES * sizeof(float);
|
||||||
|
|
||||||
|
if ((queuedAudioSize / callbackAudioSize) <= MAX_LATENCY)
|
||||||
|
{
|
||||||
|
ctx.ppcContext.r3.u32 = g_clientCallbackParam;
|
||||||
|
g_clientCallback(ctx.ppcContext, reinterpret_cast<uint8_t*>(g_memory.base));
|
||||||
|
}
|
||||||
|
|
||||||
|
auto next = start + std::chrono::duration<double>(iteration * INTERVAL);
|
||||||
|
auto now = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
if ((next - now) > 1s)
|
||||||
|
next = now;
|
||||||
|
|
||||||
|
std::this_thread::sleep_until(next);
|
||||||
|
|
||||||
|
iteration = std::chrono::duration<double>(std::chrono::steady_clock::now() - start).count() / INTERVAL + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void XAudioRegisterClient(PPCFunc* callback, uint32_t param)
|
||||||
|
{
|
||||||
|
auto* pClientParam = static_cast<uint32_t*>(g_userHeap.Alloc(sizeof(param)));
|
||||||
|
ByteSwapInplace(param);
|
||||||
|
*pClientParam = param;
|
||||||
|
g_clientCallbackParam = g_memory.MapVirtual(pClientParam);
|
||||||
|
g_clientCallback = callback;
|
||||||
|
|
||||||
|
SDL_PauseAudioDevice(g_audioDevice, 0);
|
||||||
|
g_audioThread = std::make_unique<std::thread>(AudioThread);
|
||||||
|
}
|
||||||
|
|
||||||
|
void XAudioSubmitFrame(void* samples)
|
||||||
|
{
|
||||||
|
if (g_downMixToStereo)
|
||||||
|
{
|
||||||
|
// 0: left 1.0f, right 0.0f
|
||||||
|
// 1: left 0.0f, right 1.0f
|
||||||
|
// 2: left 0.75f, right 0.75f
|
||||||
|
// 3: left 0.0f, right 0.0f
|
||||||
|
// 4: left 1.0f, right 0.0f
|
||||||
|
// 5: left 0.0f, right 1.0f
|
||||||
|
|
||||||
|
auto floatSamples = reinterpret_cast<be<float>*>(samples);
|
||||||
|
|
||||||
|
std::array<float, 2 * XAUDIO_NUM_SAMPLES> audioFrames;
|
||||||
|
|
||||||
|
for (size_t i = 0; i < XAUDIO_NUM_SAMPLES; i++)
|
||||||
|
{
|
||||||
|
float ch0 = floatSamples[0 * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
float ch1 = floatSamples[1 * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
float ch2 = floatSamples[2 * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
float ch3 = floatSamples[3 * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
float ch4 = floatSamples[4 * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
float ch5 = floatSamples[5 * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
|
||||||
|
audioFrames[i * 2 + 0] = ch0 + ch2 * 0.75f + ch4;
|
||||||
|
audioFrames[i * 2 + 1] = ch1 + ch2 * 0.75f + ch5;
|
||||||
|
}
|
||||||
|
|
||||||
|
SDL_QueueAudio(g_audioDevice, &audioFrames, sizeof(audioFrames));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
auto rawSamples = reinterpret_cast<be<uint32_t>*>(samples);
|
||||||
|
|
||||||
|
std::array<uint32_t, XAUDIO_NUM_CHANNELS * XAUDIO_NUM_SAMPLES> audioFrames;
|
||||||
|
|
||||||
|
for (size_t i = 0; i < XAUDIO_NUM_SAMPLES; i++)
|
||||||
|
{
|
||||||
|
for (size_t j = 0; j < XAUDIO_NUM_CHANNELS; j++)
|
||||||
|
audioFrames[i * XAUDIO_NUM_CHANNELS + j] = rawSamples[j * XAUDIO_NUM_SAMPLES + i];
|
||||||
|
}
|
||||||
|
|
||||||
|
SDL_QueueAudio(g_audioDevice, &audioFrames, sizeof(audioFrames));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -90,7 +90,7 @@ void XAudioInitializeSystem()
|
||||||
g_sourceVoice->Start();
|
g_sourceVoice->Start();
|
||||||
|
|
||||||
KeInsertHostFunction(XAUDIO_DRIVER_KEY, DriverLoop);
|
KeInsertHostFunction(XAUDIO_DRIVER_KEY, DriverLoop);
|
||||||
GuestThread::Start(XAUDIO_DRIVER_KEY, 0, 0, &g_driverThread);
|
GuestThread::Start({ XAUDIO_DRIVER_KEY, 0, 0 }, nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
void XAudioRegisterClient(PPCFunc* callback, uint32_t param)
|
void XAudioRegisterClient(PPCFunc* callback, uint32_t param)
|
||||||
|
|
|
||||||
|
|
@ -10,82 +10,6 @@
|
||||||
#include <res/sounds/sys_actstg_pausewinclose.ogg.h>
|
#include <res/sounds/sys_actstg_pausewinclose.ogg.h>
|
||||||
#include <res/sounds/sys_actstg_pausewinopen.ogg.h>
|
#include <res/sounds/sys_actstg_pausewinopen.ogg.h>
|
||||||
|
|
||||||
#pragma region libvorbis
|
|
||||||
static ma_result ma_decoding_backend_init__libvorbis(void* pUserData, ma_read_proc onRead, ma_seek_proc onSeek, ma_tell_proc onTell, void* pReadSeekTellUserData, const ma_decoding_backend_config* pConfig, const ma_allocation_callbacks* pAllocationCallbacks, ma_data_source** ppBackend)
|
|
||||||
{
|
|
||||||
ma_result result;
|
|
||||||
ma_libvorbis* pVorbis;
|
|
||||||
|
|
||||||
(void)pUserData;
|
|
||||||
|
|
||||||
pVorbis = (ma_libvorbis*)ma_malloc(sizeof(*pVorbis), pAllocationCallbacks);
|
|
||||||
if (pVorbis == NULL) {
|
|
||||||
return MA_OUT_OF_MEMORY;
|
|
||||||
}
|
|
||||||
|
|
||||||
result = ma_libvorbis_init(onRead, onSeek, onTell, pReadSeekTellUserData, pConfig, pAllocationCallbacks, pVorbis);
|
|
||||||
if (result != MA_SUCCESS) {
|
|
||||||
ma_free(pVorbis, pAllocationCallbacks);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
*ppBackend = pVorbis;
|
|
||||||
|
|
||||||
return MA_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
static ma_result ma_decoding_backend_init_file__libvorbis(void* pUserData, const char* pFilePath, const ma_decoding_backend_config* pConfig, const ma_allocation_callbacks* pAllocationCallbacks, ma_data_source** ppBackend)
|
|
||||||
{
|
|
||||||
ma_result result;
|
|
||||||
ma_libvorbis* pVorbis;
|
|
||||||
|
|
||||||
(void)pUserData;
|
|
||||||
|
|
||||||
pVorbis = (ma_libvorbis*)ma_malloc(sizeof(*pVorbis), pAllocationCallbacks);
|
|
||||||
if (pVorbis == NULL) {
|
|
||||||
return MA_OUT_OF_MEMORY;
|
|
||||||
}
|
|
||||||
|
|
||||||
result = ma_libvorbis_init_file(pFilePath, pConfig, pAllocationCallbacks, pVorbis);
|
|
||||||
if (result != MA_SUCCESS) {
|
|
||||||
ma_free(pVorbis, pAllocationCallbacks);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
*ppBackend = pVorbis;
|
|
||||||
|
|
||||||
return MA_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void ma_decoding_backend_uninit__libvorbis(void* pUserData, ma_data_source* pBackend, const ma_allocation_callbacks* pAllocationCallbacks)
|
|
||||||
{
|
|
||||||
ma_libvorbis* pVorbis = (ma_libvorbis*)pBackend;
|
|
||||||
|
|
||||||
(void)pUserData;
|
|
||||||
|
|
||||||
ma_libvorbis_uninit(pVorbis, pAllocationCallbacks);
|
|
||||||
ma_free(pVorbis, pAllocationCallbacks);
|
|
||||||
}
|
|
||||||
|
|
||||||
static ma_result ma_decoding_backend_get_channel_map__libvorbis(void* pUserData, ma_data_source* pBackend, ma_channel* pChannelMap, size_t channelMapCap)
|
|
||||||
{
|
|
||||||
ma_libvorbis* pVorbis = (ma_libvorbis*)pBackend;
|
|
||||||
|
|
||||||
(void)pUserData;
|
|
||||||
|
|
||||||
return ma_libvorbis_get_data_format(pVorbis, NULL, NULL, NULL, pChannelMap, channelMapCap);
|
|
||||||
}
|
|
||||||
|
|
||||||
static ma_decoding_backend_vtable g_ma_decoding_backend_vtable_libvorbis =
|
|
||||||
{
|
|
||||||
ma_decoding_backend_init__libvorbis,
|
|
||||||
ma_decoding_backend_init_file__libvorbis,
|
|
||||||
NULL, /* onInitFileW() */
|
|
||||||
NULL, /* onInitMemory() */
|
|
||||||
ma_decoding_backend_uninit__libvorbis
|
|
||||||
};
|
|
||||||
#pragma endregion
|
|
||||||
|
|
||||||
enum class EmbeddedSound
|
enum class EmbeddedSound
|
||||||
{
|
{
|
||||||
SysWorldMapCursor,
|
SysWorldMapCursor,
|
||||||
|
|
@ -101,12 +25,10 @@ enum class EmbeddedSound
|
||||||
struct EmbeddedSoundData
|
struct EmbeddedSoundData
|
||||||
{
|
{
|
||||||
static const int SimultaneousLimit = 4;
|
static const int SimultaneousLimit = 4;
|
||||||
std::array<std::unique_ptr<ma_sound>, SimultaneousLimit> sounds;
|
Mix_Chunk* chunk{};
|
||||||
std::array<std::unique_ptr<ma_decoder>, SimultaneousLimit> decoders;
|
int channelIndex{};
|
||||||
int oldestIndex = 0;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static ma_engine g_audioEngine = {};
|
|
||||||
static std::array<EmbeddedSoundData, size_t(EmbeddedSound::Count)> g_embeddedSoundData = {};
|
static std::array<EmbeddedSoundData, size_t(EmbeddedSound::Count)> g_embeddedSoundData = {};
|
||||||
static const std::unordered_map<std::string_view, EmbeddedSound> g_embeddedSoundMap =
|
static const std::unordered_map<std::string_view, EmbeddedSound> g_embeddedSoundMap =
|
||||||
{
|
{
|
||||||
|
|
@ -122,10 +44,7 @@ static const std::unordered_map<std::string_view, EmbeddedSound> g_embeddedSound
|
||||||
static void PlayEmbeddedSound(EmbeddedSound s)
|
static void PlayEmbeddedSound(EmbeddedSound s)
|
||||||
{
|
{
|
||||||
EmbeddedSoundData &data = g_embeddedSoundData[size_t(s)];
|
EmbeddedSoundData &data = g_embeddedSoundData[size_t(s)];
|
||||||
int pickedIndex = -1;
|
if (data.chunk == nullptr)
|
||||||
for (int i = 0; (i < EmbeddedSoundData::SimultaneousLimit) && (pickedIndex < 0); i++)
|
|
||||||
{
|
|
||||||
if (data.sounds[i] == nullptr)
|
|
||||||
{
|
{
|
||||||
// The sound hasn't been created yet, create it and pick it.
|
// The sound hasn't been created yet, create it and pick it.
|
||||||
const void *soundData = nullptr;
|
const void *soundData = nullptr;
|
||||||
|
|
@ -165,68 +84,16 @@ static void PlayEmbeddedSound(EmbeddedSound s)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ma_decoding_backend_vtable* pCustomBackendVTables[] =
|
data.chunk = Mix_LoadWAV_RW(SDL_RWFromConstMem(soundData, soundDataSize), 1);
|
||||||
{
|
|
||||||
&g_ma_decoding_backend_vtable_libvorbis
|
|
||||||
};
|
|
||||||
|
|
||||||
ma_decoder_config decoderConfig = ma_decoder_config_init_default();
|
|
||||||
decoderConfig.pCustomBackendUserData = NULL;
|
|
||||||
decoderConfig.ppCustomBackendVTables = pCustomBackendVTables;
|
|
||||||
decoderConfig.customBackendCount = std::size(pCustomBackendVTables);
|
|
||||||
|
|
||||||
ma_result res;
|
|
||||||
data.decoders[i] = std::make_unique<ma_decoder>();
|
|
||||||
res = ma_decoder_init_memory(soundData, soundDataSize, &decoderConfig, data.decoders[i].get());
|
|
||||||
if (res != MA_SUCCESS)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "ma_decoder_init_memory failed with error code %d.\n", res);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
data.sounds[i] = std::make_unique<ma_sound>();
|
Mix_PlayChannel(data.channelIndex % EmbeddedSoundData::SimultaneousLimit, data.chunk, 0);
|
||||||
res = ma_sound_init_from_data_source(&g_audioEngine, data.decoders[i].get(), MA_SOUND_FLAG_DECODE, nullptr, data.sounds[i].get());
|
++data.channelIndex;
|
||||||
if (res != MA_SUCCESS)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "ma_sound_init_from_data_source failed with error code %d.\n", res);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
pickedIndex = i;
|
|
||||||
}
|
|
||||||
else if (ma_sound_at_end(data.sounds[i].get()))
|
|
||||||
{
|
|
||||||
// A sound has reached the end, pick it.
|
|
||||||
pickedIndex = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pickedIndex < 0)
|
|
||||||
{
|
|
||||||
// No free slots are available, pick the oldest one.
|
|
||||||
pickedIndex = data.oldestIndex;
|
|
||||||
data.oldestIndex = (data.oldestIndex + 1) % EmbeddedSoundData::SimultaneousLimit;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.sounds[pickedIndex] != nullptr)
|
|
||||||
{
|
|
||||||
ma_sound_set_volume(data.sounds[pickedIndex].get(), Config::EffectsVolume);
|
|
||||||
ma_sound_seek_to_pcm_frame(data.sounds[pickedIndex].get(), 0);
|
|
||||||
ma_sound_start(data.sounds[pickedIndex].get());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void EmbeddedPlayer::Init()
|
void EmbeddedPlayer::Init()
|
||||||
{
|
{
|
||||||
ma_engine_config engineConfig = ma_engine_config_init();
|
Mix_OpenAudio(XAUDIO_SAMPLES_HZ, AUDIO_F32SYS, XAUDIO_NUM_CHANNELS, 256);
|
||||||
engineConfig.channels = XAUDIO_NUM_CHANNELS;
|
|
||||||
engineConfig.sampleRate = XAUDIO_SAMPLES_HZ;
|
|
||||||
|
|
||||||
ma_result res = ma_engine_init(&engineConfig, &g_audioEngine);
|
|
||||||
if (res != MA_SUCCESS)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "ma_engine_init failed with error code %d.\n", res);
|
|
||||||
}
|
|
||||||
|
|
||||||
s_isActive = true;
|
s_isActive = true;
|
||||||
}
|
}
|
||||||
|
|
@ -241,11 +108,6 @@ void EmbeddedPlayer::Play(const char *name)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (g_audioEngine.pDevice == nullptr)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
PlayEmbeddedSound(it->second);
|
PlayEmbeddedSound(it->second);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -253,37 +115,12 @@ void EmbeddedPlayer::Shutdown()
|
||||||
{
|
{
|
||||||
for (EmbeddedSoundData &data : g_embeddedSoundData)
|
for (EmbeddedSoundData &data : g_embeddedSoundData)
|
||||||
{
|
{
|
||||||
for (auto &sound : data.sounds)
|
if (data.chunk != nullptr)
|
||||||
{
|
Mix_FreeChunk(data.chunk);
|
||||||
if (sound != nullptr)
|
|
||||||
{
|
|
||||||
if (sound->pDataSource != nullptr)
|
|
||||||
{
|
|
||||||
ma_sound_uninit(sound.get());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sound.reset();
|
Mix_CloseAudio();
|
||||||
}
|
Mix_Quit();
|
||||||
}
|
|
||||||
|
|
||||||
for (auto &decoder : data.decoders)
|
|
||||||
{
|
|
||||||
if (decoder != nullptr)
|
|
||||||
{
|
|
||||||
if (decoder->pBackend != nullptr)
|
|
||||||
{
|
|
||||||
ma_decoder_uninit(decoder.get());
|
|
||||||
}
|
|
||||||
|
|
||||||
decoder.reset();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (g_audioEngine.pDevice != nullptr)
|
|
||||||
{
|
|
||||||
ma_engine_uninit(&g_audioEngine);
|
|
||||||
}
|
|
||||||
|
|
||||||
s_isActive = false;
|
s_isActive = false;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -32,6 +32,7 @@ GuestThreadContext::GuestThreadContext(uint32_t cpuNumber)
|
||||||
ppcContext.fn = (uint8_t*)g_codeCache.bucket;
|
ppcContext.fn = (uint8_t*)g_codeCache.bucket;
|
||||||
ppcContext.r1.u64 = g_memory.MapVirtual(thread + PCR_SIZE + TLS_SIZE + TEB_SIZE + STACK_SIZE); // stack pointer
|
ppcContext.r1.u64 = g_memory.MapVirtual(thread + PCR_SIZE + TLS_SIZE + TEB_SIZE + STACK_SIZE); // stack pointer
|
||||||
ppcContext.r13.u64 = g_memory.MapVirtual(thread);
|
ppcContext.r13.u64 = g_memory.MapVirtual(thread);
|
||||||
|
ppcContext.fpscr.loadFromHost();
|
||||||
|
|
||||||
assert(GetPPCContext() == nullptr);
|
assert(GetPPCContext() == nullptr);
|
||||||
SetPPCContext(ppcContext);
|
SetPPCContext(ppcContext);
|
||||||
|
|
@ -77,7 +78,7 @@ uint32_t GuestThread::Start(const GuestThreadParams& params)
|
||||||
GuestThreadContext ctx(cpuNumber);
|
GuestThreadContext ctx(cpuNumber);
|
||||||
ctx.ppcContext.r3.u64 = params.value;
|
ctx.ppcContext.r3.u64 = params.value;
|
||||||
|
|
||||||
GuestCode::Run(g_codeCache.Find(params.function), &ctx.ppcContext, g_memory.Translate(0));
|
reinterpret_cast<PPCFunc*>(g_codeCache.Find(params.function))(ctx.ppcContext, reinterpret_cast<uint8_t*>(g_memory.base));
|
||||||
|
|
||||||
return ctx.ppcContext.r3.u32;
|
return ctx.ppcContext.r3.u32;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -76,16 +76,16 @@ static std::atomic<uint32_t> g_keSetEventGeneration;
|
||||||
|
|
||||||
struct Semaphore final : KernelObject, HostObject<XKSEMAPHORE>
|
struct Semaphore final : KernelObject, HostObject<XKSEMAPHORE>
|
||||||
{
|
{
|
||||||
std::atomic<uint32_t> count;
|
std::counting_semaphore<> semaphore;
|
||||||
uint32_t maximumCount;
|
uint32_t maximumCount;
|
||||||
|
|
||||||
Semaphore(XKSEMAPHORE* semaphore)
|
Semaphore(XKSEMAPHORE* semaphore)
|
||||||
: count(semaphore->Header.SignalState), maximumCount(semaphore->Limit)
|
: semaphore(semaphore->Header.SignalState), maximumCount(semaphore->Limit)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
Semaphore(uint32_t count, uint32_t maximumCount)
|
Semaphore(uint32_t count, uint32_t maximumCount)
|
||||||
: count(count), maximumCount(maximumCount)
|
: semaphore(count), maximumCount(maximumCount)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -93,32 +93,11 @@ struct Semaphore final : KernelObject, HostObject<XKSEMAPHORE>
|
||||||
{
|
{
|
||||||
if (timeout == 0)
|
if (timeout == 0)
|
||||||
{
|
{
|
||||||
uint32_t currentCount = count.load();
|
return semaphore.try_acquire() ? STATUS_SUCCESS : STATUS_TIMEOUT;
|
||||||
if (currentCount != 0)
|
|
||||||
{
|
|
||||||
if (count.compare_exchange_weak(currentCount, currentCount - 1))
|
|
||||||
return STATUS_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
return STATUS_TIMEOUT;
|
|
||||||
}
|
}
|
||||||
else if (timeout == INFINITE)
|
else if (timeout == INFINITE)
|
||||||
{
|
{
|
||||||
uint32_t currentCount;
|
semaphore.acquire();
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
currentCount = count.load();
|
|
||||||
if (currentCount != 0)
|
|
||||||
{
|
|
||||||
if (count.compare_exchange_weak(currentCount, currentCount - 1))
|
|
||||||
return STATUS_SUCCESS;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
count.wait(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return STATUS_SUCCESS;
|
return STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
@ -130,13 +109,7 @@ struct Semaphore final : KernelObject, HostObject<XKSEMAPHORE>
|
||||||
|
|
||||||
void Release(uint32_t releaseCount, uint32_t* previousCount)
|
void Release(uint32_t releaseCount, uint32_t* previousCount)
|
||||||
{
|
{
|
||||||
if (previousCount != nullptr)
|
semaphore.release(releaseCount);
|
||||||
*previousCount = count;
|
|
||||||
|
|
||||||
assert(count + releaseCount <= maximumCount);
|
|
||||||
|
|
||||||
count += releaseCount;
|
|
||||||
count.notify_all();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -546,7 +519,13 @@ uint32_t KeDelayExecutionThread(uint32_t WaitMode, bool Alertable, be<int64_t>*
|
||||||
if (Alertable)
|
if (Alertable)
|
||||||
return STATUS_USER_APC;
|
return STATUS_USER_APC;
|
||||||
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(GuestTimeoutToMilliseconds(Timeout)));
|
uint32_t timeout = GuestTimeoutToMilliseconds(Timeout);
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
Sleep(timeout);
|
||||||
|
#else
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(timeout));
|
||||||
|
#endif
|
||||||
|
|
||||||
return STATUS_SUCCESS;
|
return STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -137,6 +137,10 @@ uint32_t LdrLoadModule(const std::filesystem::path &path)
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
|
#ifdef _WIN32
|
||||||
|
timeBeginPeriod(1);
|
||||||
|
#endif
|
||||||
|
|
||||||
os::logger::Init();
|
os::logger::Init();
|
||||||
|
|
||||||
bool forceInstaller = false;
|
bool forceInstaller = false;
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,4 @@
|
||||||
#define STB_IMAGE_IMPLEMENTATION
|
#define STB_IMAGE_IMPLEMENTATION
|
||||||
#include <stb_image.h>
|
#include <stb_image.h>
|
||||||
|
|
||||||
#define MINIAUDIO_IMPLEMENTATION
|
|
||||||
#include <miniaudio.h>
|
|
||||||
|
|
||||||
#define ma_offset_pcm_frames_ptr (char*)ma_offset_pcm_frames_ptr
|
|
||||||
#include <extras/miniaudio_libvorbis.h>
|
|
||||||
|
|
||||||
#include "stdafx.h"
|
#include "stdafx.h"
|
||||||
|
|
|
||||||
|
|
@ -36,6 +36,7 @@ using Microsoft::WRL::ComPtr;
|
||||||
#include <stb_image.h>
|
#include <stb_image.h>
|
||||||
#include <blockingconcurrentqueue.h>
|
#include <blockingconcurrentqueue.h>
|
||||||
#include <SDL.h>
|
#include <SDL.h>
|
||||||
|
#include <SDL_mixer.h>
|
||||||
#include <imgui.h>
|
#include <imgui.h>
|
||||||
#include <imgui_internal.h>
|
#include <imgui_internal.h>
|
||||||
#include <backends/imgui_impl_sdl2.h>
|
#include <backends/imgui_impl_sdl2.h>
|
||||||
|
|
@ -43,10 +44,9 @@ using Microsoft::WRL::ComPtr;
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include <smolv.h>
|
#include <smolv.h>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <miniaudio.h>
|
|
||||||
#include <extras/miniaudio_libvorbis.h>
|
|
||||||
#include <fmt/core.h>
|
#include <fmt/core.h>
|
||||||
#include <list>
|
#include <list>
|
||||||
|
#include <semaphore>
|
||||||
|
|
||||||
#include "framework.h"
|
#include "framework.h"
|
||||||
#include "mutex.h"
|
#include "mutex.h"
|
||||||
|
|
|
||||||
15
thirdparty/CMakeLists.txt
vendored
15
thirdparty/CMakeLists.txt
vendored
|
|
@ -1,11 +1,22 @@
|
||||||
|
cmake_policy(SET CMP0077 NEW)
|
||||||
|
|
||||||
set(MSDF_ATLAS_BUILD_STANDALONE OFF)
|
set(MSDF_ATLAS_BUILD_STANDALONE OFF)
|
||||||
set(MSDF_ATLAS_USE_SKIA OFF)
|
set(MSDF_ATLAS_USE_SKIA OFF)
|
||||||
set(MSDF_ATLAS_NO_ARTERY_FONT ON)
|
set(MSDF_ATLAS_NO_ARTERY_FONT ON)
|
||||||
set(MSDFGEN_DISABLE_PNG ON)
|
set(MSDFGEN_DISABLE_PNG ON)
|
||||||
|
|
||||||
|
set(SDL2MIXER_DEPS_SHARED OFF)
|
||||||
|
set(SDL2MIXER_VENDORED ON)
|
||||||
|
set(SDL2MIXER_FLAC OFF)
|
||||||
|
set(SDL2MIXER_MOD OFF)
|
||||||
|
set(SDL2MIXER_MP3 OFF)
|
||||||
|
set(SDL2MIXER_MIDI OFF)
|
||||||
|
set(SDL2MIXER_OPUS OFF)
|
||||||
|
set(SDL2MIXER_VORBIS "VORBISFILE")
|
||||||
|
set(SDL2MIXER_WAVPACK OFF)
|
||||||
|
|
||||||
add_subdirectory("${SWA_THIRDPARTY_ROOT}/msdf-atlas-gen")
|
add_subdirectory("${SWA_THIRDPARTY_ROOT}/msdf-atlas-gen")
|
||||||
add_subdirectory("${SWA_THIRDPARTY_ROOT}/nativefiledialog-extended")
|
add_subdirectory("${SWA_THIRDPARTY_ROOT}/nativefiledialog-extended")
|
||||||
add_subdirectory("${SWA_THIRDPARTY_ROOT}/ogg")
|
|
||||||
add_subdirectory("${SWA_THIRDPARTY_ROOT}/o1heap")
|
add_subdirectory("${SWA_THIRDPARTY_ROOT}/o1heap")
|
||||||
add_subdirectory("${SWA_THIRDPARTY_ROOT}/SDL")
|
add_subdirectory("${SWA_THIRDPARTY_ROOT}/SDL")
|
||||||
add_subdirectory("${SWA_THIRDPARTY_ROOT}/vorbis")
|
add_subdirectory("${SWA_THIRDPARTY_ROOT}/SDL_mixer")
|
||||||
|
|
|
||||||
1
thirdparty/SDL_mixer
vendored
Submodule
1
thirdparty/SDL_mixer
vendored
Submodule
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 437992692cf9300f2b2f04be35adc7445a9055bf
|
||||||
1
thirdparty/miniaudio
vendored
1
thirdparty/miniaudio
vendored
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 12a8d4e4911c5ab4f4c089b4d039433975ed8a66
|
|
||||||
1
thirdparty/ogg
vendored
1
thirdparty/ogg
vendored
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 7cf42ea17aef7bc1b7b21af70724840a96c2e7d0
|
|
||||||
1
thirdparty/vorbis
vendored
1
thirdparty/vorbis
vendored
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 84c023699cdf023a32fa4ded32019f194afcdad0
|
|
||||||
Loading…
Add table
Reference in a new issue