Mesen-SX/Core/SoundMixer.cpp

151 lines
No EOL
3.9 KiB
C++

#include "stdafx.h"
#include "SoundMixer.h"
#include "Console.h"
#include "EmuSettings.h"
#include "SoundResampler.h"
#include "RewindManager.h"
#include "VideoRenderer.h"
#include "WaveRecorder.h"
#include "../Utilities/Equalizer.h"
#include "../Utilities/blip_buf.h"
SoundMixer::SoundMixer(Console *console)
{
_console = console;
_audioDevice = nullptr;
_resampler.reset(new SoundResampler(console));
_sampleBuffer = new int16_t[0x10000];
}
SoundMixer::~SoundMixer()
{
delete[] _sampleBuffer;
}
void SoundMixer::RegisterAudioDevice(IAudioDevice *audioDevice)
{
_audioDevice = audioDevice;
}
AudioStatistics SoundMixer::GetStatistics()
{
if(_audioDevice) {
return _audioDevice->GetStatistics();
} else {
return AudioStatistics();
}
}
void SoundMixer::StopAudio(bool clearBuffer)
{
if(_audioDevice) {
if(clearBuffer) {
_audioDevice->Stop();
} else {
_audioDevice->Pause();
}
}
}
void SoundMixer::PlayAudioBuffer(int16_t* samples, uint32_t sampleCount)
{
AudioConfig cfg = _console->GetSettings()->GetAudioConfig();
if(cfg.EnableEqualizer) {
ProcessEqualizer(samples, sampleCount);
}
uint32_t masterVolume = cfg.MasterVolume;
if(_console->GetSettings()->CheckFlag(EmulationFlags::InBackground)) {
if(cfg.MuteSoundInBackground) {
masterVolume = 0;
} else if(cfg.ReduceSoundInBackground) {
masterVolume = cfg.VolumeReduction == 100 ? 0 : masterVolume * (100 - cfg.VolumeReduction) / 100;
}
} else if(cfg.ReduceSoundInFastForward && _console->GetSettings()->CheckFlag(EmulationFlags::TurboOrRewind)) {
masterVolume = cfg.VolumeReduction == 100 ? 0 : masterVolume * (100 - cfg.VolumeReduction) / 100;
}
_leftSample = samples[0];
_rightSample = samples[1];
if(masterVolume < 100) {
//Apply volume if not using the default value
for(uint32_t i = 0; i < sampleCount * 2; i++) {
samples[i] = (int32_t)samples[i] * (int32_t)masterVolume / 100;
}
}
shared_ptr<RewindManager> rewindManager = _console->GetRewindManager();
if(rewindManager && rewindManager->SendAudio(samples, (uint32_t)sampleCount)) {
int16_t *out = nullptr;
uint32_t count = 0;
if(cfg.SampleRate == SoundResampler::SpcSampleRate && cfg.DisableDynamicSampleRate) {
out = samples;
count = sampleCount;
} else {
count = _resampler->Resample(samples, sampleCount, cfg.SampleRate, _sampleBuffer);
out = _sampleBuffer;
}
bool isRecording = _waveRecorder || _console->GetVideoRenderer()->IsRecording();
if(isRecording) {
if(_waveRecorder) {
_waveRecorder->WriteSamples(out, count, cfg.SampleRate, true);
}
_console->GetVideoRenderer()->AddRecordingSound(out, count, cfg.SampleRate);
}
if(_audioDevice) {
if(!cfg.EnableAudio) {
_audioDevice->Stop();
return;
}
_audioDevice->PlayBuffer(out, count, cfg.SampleRate, true);
_audioDevice->ProcessEndOfFrame();
}
}
}
void SoundMixer::ProcessEqualizer(int16_t* samples, uint32_t sampleCount)
{
AudioConfig cfg = _console->GetSettings()->GetAudioConfig();
if(!_equalizer) {
_equalizer.reset(new Equalizer());
}
vector<double> bandGains = {
cfg.Band1Gain, cfg.Band2Gain, cfg.Band3Gain, cfg.Band4Gain, cfg.Band5Gain,
cfg.Band6Gain, cfg.Band7Gain, cfg.Band8Gain, cfg.Band9Gain, cfg.Band10Gain,
cfg.Band11Gain, cfg.Band12Gain, cfg.Band13Gain, cfg.Band14Gain, cfg.Band15Gain,
cfg.Band16Gain, cfg.Band17Gain, cfg.Band18Gain, cfg.Band19Gain, cfg.Band20Gain
};
_equalizer->UpdateEqualizers(bandGains, SoundResampler::SpcSampleRate);
_equalizer->ApplyEqualizer(sampleCount, samples);
}
double SoundMixer::GetRateAdjustment()
{
return _resampler->GetRateAdjustment();
}
void SoundMixer::StartRecording(string filepath)
{
_waveRecorder.reset(new WaveRecorder(filepath, _console->GetSettings()->GetAudioConfig().SampleRate, true));
}
void SoundMixer::StopRecording()
{
_waveRecorder.reset();
}
bool SoundMixer::IsRecording()
{
return _waveRecorder.get() != nullptr;
}
void SoundMixer::GetLastSamples(int16_t &left, int16_t &right)
{
left = _leftSample;
right = _rightSample;
}