From 3b6190b69a9437c3ad7ae05668d35faeb25af9c6 Mon Sep 17 00:00:00 2001 From: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com> Date: Thu, 28 Nov 2024 19:01:55 +0200 Subject: [PATCH] Add volume slider & mute audio settings --- include/audio/miniaudio_device.hpp | 8 ++++-- include/config.hpp | 14 ++++++++++ src/config.cpp | 7 +++++ src/core/audio/aac_decoder.cpp | 6 ++--- src/core/audio/hle_core.cpp | 5 +--- src/core/audio/miniaudio_device.cpp | 40 ++++++++++++++++++++++++++++- src/emulator.cpp | 2 +- 7 files changed, 70 insertions(+), 12 deletions(-) diff --git a/include/audio/miniaudio_device.hpp b/include/audio/miniaudio_device.hpp index f3b212e2..769f0a78 100644 --- a/include/audio/miniaudio_device.hpp +++ b/include/audio/miniaudio_device.hpp @@ -3,6 +3,7 @@ #include #include +#include "config.hpp" #include "helpers.hpp" #include "miniaudio.h" #include "ring_buffer.hpp" @@ -12,12 +13,14 @@ class MiniAudioDevice { static constexpr ma_uint32 sampleRate = 32768; // 3DS sample rate static constexpr ma_uint32 channelCount = 2; // Audio output is stereo + ma_device device; ma_context context; ma_device_config deviceConfig; - ma_device device; ma_resampler resampler; Samples* samples = nullptr; + AudioDeviceConfig& audioSettings; + bool initialized = false; bool running = false; @@ -26,7 +29,8 @@ class MiniAudioDevice { std::vector audioDevices; public: - MiniAudioDevice(); + MiniAudioDevice(AudioDeviceConfig& audioSettings); + // If safe is on, we create a null audio device void init(Samples& samples, bool safe = false); void close(); diff --git a/include/config.hpp b/include/config.hpp index 55fcbe4f..9364a33d 100644 --- a/include/config.hpp +++ b/include/config.hpp @@ -4,6 +4,19 @@ #include "audio/dsp_core.hpp" #include "renderer.hpp" +struct AudioDeviceConfig { + float volumeRaw = 1.0f; + bool muteAudio = false; + + float getVolume() const { + if (muteAudio) { + return 0.0f; + } + + return volumeRaw; + } +}; + // Remember to initialize every field here to its default value otherwise bad things will happen struct EmulatorConfig { // Only enable the shader JIT by default on platforms where it's completely tested @@ -71,6 +84,7 @@ struct EmulatorConfig { }; WindowSettings windowSettings; + AudioDeviceConfig audioDeviceConfig; EmulatorConfig(const std::filesystem::path& path); void load(); diff --git a/src/config.cpp b/src/config.cpp index e4fc9a13..833a1b4e 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -99,6 +99,11 @@ void EmulatorConfig::load() { audioEnabled = toml::find_or(audio, "EnableAudio", false); aacEnabled = toml::find_or(audio, "EnableAACAudio", true); + + audioDeviceConfig.muteAudio = toml::find_or(audio, "MuteAudio", false); + // Our volume ranges from 0.0 (muted) to 2.0 (boosted, using a logarithmic scale). 1.0 is the "default" volume, ie we don't adjust the PCM + // samples at all. + audioDeviceConfig.volumeRaw = float(std::clamp(toml::find_or(audio, "AudioVolume", 1.0), 0.0, 2.0)); } } @@ -170,6 +175,8 @@ void EmulatorConfig::save() { data["Audio"]["DSPEmulation"] = std::string(Audio::DSPCore::typeToString(dspType)); data["Audio"]["EnableAudio"] = audioEnabled; data["Audio"]["EnableAACAudio"] = aacEnabled; + data["Audio"]["MuteAudio"] = audioDeviceConfig.muteAudio; + data["Audio"]["AudioVolume"] = double(audioDeviceConfig.volumeRaw); data["Battery"]["ChargerPlugged"] = chargerPlugged; data["Battery"]["BatteryPercentage"] = batteryPercentage; diff --git a/src/core/audio/aac_decoder.cpp b/src/core/audio/aac_decoder.cpp index 5b9ecee4..af88485c 100644 --- a/src/core/audio/aac_decoder.cpp +++ b/src/core/audio/aac_decoder.cpp @@ -103,10 +103,8 @@ void AAC::Decoder::decode(AAC::Message& response, const AAC::Message& request, A } } else { // If audio is not enabled, push 0s - for (int sample = 0; sample < info->frameSize; sample++) { - for (int stream = 0; stream < channels; stream++) { - audioStreams[stream].push_back(0); - } + for (int stream = 0; stream < channels; stream++) { + audioStreams[stream].resize(audioStreams[stream].size() + info->frameSize, 0); } } } else { diff --git a/src/core/audio/hle_core.cpp b/src/core/audio/hle_core.cpp index ddb893e7..26e566b4 100644 --- a/src/core/audio/hle_core.cpp +++ b/src/core/audio/hle_core.cpp @@ -715,10 +715,7 @@ namespace Audio { response.command = request.command; response.mode = request.mode; - // We allow disabling AAC audio. Mostly useful for debugging & figuring out if an audio track is AAC or not. - if (settings.aacEnabled) { - aacDecoder->decode(response, request, [this](u32 paddr) { return getPointerPhys(paddr); }); - } + aacDecoder->decode(response, request, [this](u32 paddr) { return getPointerPhys(paddr); }, settings.aacEnabled); break; } diff --git a/src/core/audio/miniaudio_device.cpp b/src/core/audio/miniaudio_device.cpp index cd537aef..8965e71c 100644 --- a/src/core/audio/miniaudio_device.cpp +++ b/src/core/audio/miniaudio_device.cpp @@ -1,10 +1,14 @@ #include "audio/miniaudio_device.hpp" +#include +#include #include +#include #include "helpers.hpp" -MiniAudioDevice::MiniAudioDevice() : initialized(false), running(false), samples(nullptr) {} +MiniAudioDevice::MiniAudioDevice(AudioDeviceConfig& audioSettings) + : initialized(false), running(false), samples(nullptr), audioSettings(audioSettings) {} void MiniAudioDevice::init(Samples& samples, bool safe) { this->samples = &samples; @@ -106,6 +110,40 @@ void MiniAudioDevice::init(Samples& samples, bool safe) { std::memcpy(&self->lastStereoSample[0], &output[(samplesWritten - 1) * 2], sizeof(lastStereoSample)); } + // Adjust the volume of our samples based on the emulator's volume slider + float audioVolume = self->audioSettings.getVolume(); + // If volume is 1.0 we don't need to do anything + if (audioVolume != 1.0f) { + s16* sample = output; + + // If our volume is > 1.0 then we boost samples using a logarithmic scale, + // In this case we also have to clamp samples to make sure they don't wrap around + if (audioVolume > 1.0f) { + audioVolume = 0.6 + 20 * std::log10(audioVolume); + + constexpr s32 min = s32(std::numeric_limits::min()); + constexpr s32 max = s32(std::numeric_limits::max()); + + for (usize i = 0; i < samplesWritten; i += 2) { + s16 l = s16(std::clamp(s32(float(sample[0]) * audioVolume), min, max)); + s16 r = s16(std::clamp(s32(float(sample[1]) * audioVolume), min, max)); + + *sample++ = l; + *sample++ = r; + } + } else { + // If our volume is in [0.0, 1.0) then just multiply by the volume. No need to clamp, since there is no danger of our samples wrapping + // around due to overflow + for (usize i = 0; i < samplesWritten; i += 2) { + s16 l = s16(float(sample[0]) * audioVolume); + s16 r = s16(float(sample[1]) * audioVolume); + + *sample++ = l; + *sample++ = r; + } + } + } + // If underruning, copy the last output sample { s16* pointer = &output[samplesWritten * 2]; diff --git a/src/emulator.cpp b/src/emulator.cpp index 3b91390e..fc25eacb 100644 --- a/src/emulator.cpp +++ b/src/emulator.cpp @@ -20,7 +20,7 @@ __declspec(dllexport) DWORD AmdPowerXpressRequestHighPerformance = 1; Emulator::Emulator() : config(getConfigPath()), kernel(cpu, memory, gpu, config), cpu(memory, kernel, *this), gpu(memory, config), memory(cpu.getTicksRef(), config), - cheats(memory, kernel.getServiceManager().getHID()), lua(*this), running(false) + cheats(memory, kernel.getServiceManager().getHID()), audioDevice(config.audioDeviceConfig), lua(*this), running(false) #ifdef PANDA3DS_ENABLE_HTTP_SERVER , httpServer(this)