2017-12-20 10:44:32 -08:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <cstddef>
|
|
|
|
#include "audio_core/dsp_interface.h"
|
|
|
|
#include "audio_core/sink.h"
|
|
|
|
#include "audio_core/sink_details.h"
|
|
|
|
#include "common/assert.h"
|
2022-12-08 03:27:25 -08:00
|
|
|
#include "common/settings.h"
|
2019-01-26 06:20:21 -08:00
|
|
|
#include "core/core.h"
|
|
|
|
#include "core/dumping/backend.h"
|
2017-12-20 10:44:32 -08:00
|
|
|
|
|
|
|
namespace AudioCore {
|
|
|
|
|
|
|
|
DspInterface::DspInterface() = default;
|
2018-09-08 13:07:28 -07:00
|
|
|
DspInterface::~DspInterface() = default;
|
2017-12-20 10:44:32 -08:00
|
|
|
|
2023-05-01 12:17:45 -07:00
|
|
|
void DspInterface::SetSink(AudioCore::SinkType sink_type, std::string_view audio_device) {
|
2023-05-25 15:44:56 -07:00
|
|
|
// Dispose of the current sink first to avoid contention.
|
|
|
|
sink.reset();
|
|
|
|
|
2023-05-01 12:17:45 -07:00
|
|
|
sink = CreateSinkFromID(sink_type, audio_device);
|
2018-09-08 13:07:28 -07:00
|
|
|
sink->SetCallback(
|
|
|
|
[this](s16* buffer, std::size_t num_frames) { OutputCallback(buffer, num_frames); });
|
2017-12-20 10:44:32 -08:00
|
|
|
time_stretcher.SetOutputSampleRate(sink->GetNativeSampleRate());
|
|
|
|
}
|
|
|
|
|
|
|
|
Sink& DspInterface::GetSink() {
|
|
|
|
ASSERT(sink);
|
|
|
|
return *sink.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DspInterface::EnableStretching(bool enable) {
|
|
|
|
if (perform_time_stretching == enable)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!enable) {
|
2018-09-08 13:28:19 -07:00
|
|
|
flushing_time_stretcher = true;
|
2017-12-20 10:44:32 -08:00
|
|
|
}
|
|
|
|
perform_time_stretching = enable;
|
|
|
|
}
|
|
|
|
|
2020-01-28 06:19:36 -08:00
|
|
|
void DspInterface::OutputFrame(StereoFrame16 frame) {
|
2017-12-20 10:44:32 -08:00
|
|
|
if (!sink)
|
|
|
|
return;
|
|
|
|
|
2018-09-08 13:07:28 -07:00
|
|
|
fifo.Push(frame.data(), frame.size());
|
2019-01-26 06:20:21 -08:00
|
|
|
|
2023-06-16 16:06:18 -07:00
|
|
|
auto video_dumper = Core::System::GetInstance().GetVideoDumper();
|
|
|
|
if (video_dumper && video_dumper->IsDumping()) {
|
|
|
|
video_dumper->AddAudioFrame(std::move(frame));
|
2019-01-26 06:20:21 -08:00
|
|
|
}
|
2017-12-20 10:44:32 -08:00
|
|
|
}
|
|
|
|
|
2018-12-06 09:19:58 -08:00
|
|
|
void DspInterface::OutputSample(std::array<s16, 2> sample) {
|
|
|
|
if (!sink)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fifo.Push(&sample, 1);
|
2019-01-26 06:20:21 -08:00
|
|
|
|
2023-06-16 16:06:18 -07:00
|
|
|
auto video_dumper = Core::System::GetInstance().GetVideoDumper();
|
|
|
|
if (video_dumper && video_dumper->IsDumping()) {
|
|
|
|
video_dumper->AddAudioSample(std::move(sample));
|
2019-01-26 06:20:21 -08:00
|
|
|
}
|
2018-12-06 09:19:58 -08:00
|
|
|
}
|
|
|
|
|
2018-09-08 13:28:19 -07:00
|
|
|
void DspInterface::OutputCallback(s16* buffer, std::size_t num_frames) {
|
|
|
|
std::size_t frames_written;
|
|
|
|
if (perform_time_stretching) {
|
|
|
|
const std::vector<s16> in{fifo.Pop()};
|
|
|
|
const std::size_t num_in{in.size() / 2};
|
|
|
|
frames_written = time_stretcher.Process(in.data(), num_in, buffer, num_frames);
|
|
|
|
} else if (flushing_time_stretcher) {
|
|
|
|
time_stretcher.Flush();
|
|
|
|
frames_written = time_stretcher.Process(nullptr, 0, buffer, num_frames);
|
|
|
|
frames_written += fifo.Pop(buffer, num_frames - frames_written);
|
|
|
|
flushing_time_stretcher = false;
|
|
|
|
} else {
|
|
|
|
frames_written = fifo.Pop(buffer, num_frames);
|
|
|
|
}
|
2018-09-08 13:07:28 -07:00
|
|
|
|
|
|
|
if (frames_written > 0) {
|
|
|
|
std::memcpy(&last_frame[0], buffer + 2 * (frames_written - 1), 2 * sizeof(s16));
|
|
|
|
}
|
2017-12-20 10:44:32 -08:00
|
|
|
|
2018-09-08 13:07:28 -07:00
|
|
|
// Hold last emitted frame; this prevents popping.
|
2018-09-08 13:28:19 -07:00
|
|
|
for (std::size_t i = frames_written; i < num_frames; i++) {
|
2018-09-08 13:07:28 -07:00
|
|
|
std::memcpy(buffer + 2 * i, &last_frame[0], 2 * sizeof(s16));
|
2017-12-20 10:44:32 -08:00
|
|
|
}
|
2018-09-08 14:21:49 -07:00
|
|
|
|
2022-05-20 10:17:37 -07:00
|
|
|
// Implementation of the hardware volume slider
|
|
|
|
// A cubic curve is used to approximate a linear change in human-perceived loudness
|
2022-11-04 12:25:57 -07:00
|
|
|
const float linear_volume = std::clamp(Settings::Volume(), 0.0f, 1.0f);
|
2018-09-08 14:21:49 -07:00
|
|
|
if (linear_volume != 1.0) {
|
2022-05-20 10:17:37 -07:00
|
|
|
const float volume_scale_factor = linear_volume * linear_volume * linear_volume;
|
2018-09-08 14:21:49 -07:00
|
|
|
for (std::size_t i = 0; i < num_frames; i++) {
|
|
|
|
buffer[i * 2 + 0] = static_cast<s16>(buffer[i * 2 + 0] * volume_scale_factor);
|
|
|
|
buffer[i * 2 + 1] = static_cast<s16>(buffer[i * 2 + 1] * volume_scale_factor);
|
|
|
|
}
|
|
|
|
}
|
2017-12-20 10:44:32 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace AudioCore
|