2021-06-05 15:35:57 +02:00
|
|
|
// Copyright 2018 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <atomic>
|
|
|
|
#include <cstring>
|
|
|
|
#include "audio_core/sdl2_sink.h"
|
|
|
|
#include "audio_core/stream.h"
|
|
|
|
#include "audio_core/time_stretch.h"
|
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/logging/log.h"
|
|
|
|
//#include "common/settings.h"
|
|
|
|
|
|
|
|
// Ignore -Wimplicit-fallthrough due to https://github.com/libsdl-org/SDL/issues/4307
|
|
|
|
#ifdef __clang__
|
|
|
|
#pragma clang diagnostic push
|
|
|
|
#pragma clang diagnostic ignored "-Wimplicit-fallthrough"
|
|
|
|
#endif
|
|
|
|
#include <SDL.h>
|
|
|
|
#ifdef __clang__
|
|
|
|
#pragma clang diagnostic pop
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace AudioCore {
|
|
|
|
|
|
|
|
class SDLSinkStream final : public SinkStream {
|
|
|
|
public:
|
|
|
|
SDLSinkStream(u32 sample_rate, u32 num_channels_, const std::string& output_device)
|
|
|
|
: num_channels{std::min(num_channels_, 6u)}, time_stretch{sample_rate, num_channels} {
|
|
|
|
|
|
|
|
SDL_AudioSpec spec;
|
|
|
|
spec.freq = sample_rate;
|
|
|
|
spec.channels = static_cast<u8>(num_channels);
|
|
|
|
spec.format = AUDIO_S16SYS;
|
|
|
|
spec.samples = 4096;
|
|
|
|
spec.callback = nullptr;
|
|
|
|
|
|
|
|
SDL_AudioSpec obtained;
|
2021-06-07 12:51:59 +02:00
|
|
|
if (output_device.empty()) {
|
2021-06-05 15:35:57 +02:00
|
|
|
dev = SDL_OpenAudioDevice(nullptr, 0, &spec, &obtained, 0);
|
2021-06-07 12:51:59 +02:00
|
|
|
} else {
|
2021-06-05 15:35:57 +02:00
|
|
|
dev = SDL_OpenAudioDevice(output_device.c_str(), 0, &spec, &obtained, 0);
|
2021-06-07 12:51:59 +02:00
|
|
|
}
|
2021-06-05 15:35:57 +02:00
|
|
|
|
|
|
|
if (dev == 0) {
|
|
|
|
LOG_CRITICAL(Audio_Sink, "Error opening sdl audio device: {}", SDL_GetError());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_PauseAudioDevice(dev, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
~SDLSinkStream() override {
|
|
|
|
if (dev == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_CloseAudioDevice(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EnqueueSamples(u32 source_num_channels, const std::vector<s16>& samples) override {
|
|
|
|
if (source_num_channels > num_channels) {
|
|
|
|
// Downsample 6 channels to 2
|
|
|
|
ASSERT_MSG(source_num_channels == 6, "Channel count must be 6");
|
|
|
|
|
|
|
|
std::vector<s16> buf;
|
|
|
|
buf.reserve(samples.size() * num_channels / source_num_channels);
|
|
|
|
for (std::size_t i = 0; i < samples.size(); i += source_num_channels) {
|
|
|
|
// Downmixing implementation taken from the ATSC standard
|
|
|
|
const s16 left{samples[i + 0]};
|
|
|
|
const s16 right{samples[i + 1]};
|
|
|
|
const s16 center{samples[i + 2]};
|
|
|
|
const s16 surround_left{samples[i + 4]};
|
|
|
|
const s16 surround_right{samples[i + 5]};
|
|
|
|
// Not used in the ATSC reference implementation
|
|
|
|
[[maybe_unused]] const s16 low_frequency_effects{samples[i + 3]};
|
|
|
|
|
|
|
|
constexpr s32 clev{707}; // center mixing level coefficient
|
|
|
|
constexpr s32 slev{707}; // surround mixing level coefficient
|
|
|
|
|
|
|
|
buf.push_back(static_cast<s16>(left + (clev * center / 1000) +
|
|
|
|
(slev * surround_left / 1000)));
|
|
|
|
buf.push_back(static_cast<s16>(right + (clev * center / 1000) +
|
|
|
|
(slev * surround_right / 1000)));
|
|
|
|
}
|
|
|
|
int ret = SDL_QueueAudio(dev, static_cast<const void*>(buf.data()),
|
|
|
|
static_cast<u32>(buf.size() * sizeof(s16)));
|
|
|
|
if (ret < 0)
|
|
|
|
LOG_WARNING(Audio_Sink, "Could not queue audio buffer: {}", SDL_GetError());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ret = SDL_QueueAudio(dev, static_cast<const void*>(samples.data()),
|
|
|
|
static_cast<u32>(samples.size() * sizeof(s16)));
|
|
|
|
if (ret < 0)
|
|
|
|
LOG_WARNING(Audio_Sink, "Could not queue audio buffer: {}", SDL_GetError());
|
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t SamplesInQueue(u32 channel_count) const override {
|
|
|
|
if (dev == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return SDL_GetQueuedAudioSize(dev) / (channel_count * sizeof(s16));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Flush() override {
|
|
|
|
should_flush = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 GetNumChannels() const {
|
|
|
|
return num_channels;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
SDL_AudioDeviceID dev = 0;
|
|
|
|
u32 num_channels{};
|
|
|
|
std::atomic<bool> should_flush{};
|
|
|
|
TimeStretcher time_stretch;
|
|
|
|
};
|
|
|
|
|
|
|
|
SDLSink::SDLSink(std::string_view target_device_name) {
|
|
|
|
if (!SDL_WasInit(SDL_INIT_AUDIO)) {
|
|
|
|
if (SDL_InitSubSystem(SDL_INIT_AUDIO) < 0) {
|
|
|
|
LOG_CRITICAL(Audio_Sink, "SDL_InitSubSystem audio failed: {}", SDL_GetError());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (target_device_name != auto_device_name && !target_device_name.empty()) {
|
|
|
|
output_device = target_device_name;
|
|
|
|
} else {
|
|
|
|
output_device.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-07 12:51:59 +02:00
|
|
|
SDLSink::~SDLSink() = default;
|
2021-06-05 15:35:57 +02:00
|
|
|
|
|
|
|
SinkStream& SDLSink::AcquireSinkStream(u32 sample_rate, u32 num_channels, const std::string&) {
|
|
|
|
sink_streams.push_back(
|
|
|
|
std::make_unique<SDLSinkStream>(sample_rate, num_channels, output_device));
|
|
|
|
return *sink_streams.back();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> ListSDLSinkDevices() {
|
|
|
|
std::vector<std::string> device_list;
|
|
|
|
|
|
|
|
if (!SDL_WasInit(SDL_INIT_AUDIO)) {
|
|
|
|
if (SDL_InitSubSystem(SDL_INIT_AUDIO) < 0) {
|
|
|
|
LOG_CRITICAL(Audio_Sink, "SDL_InitSubSystem audio failed: {}", SDL_GetError());
|
2021-06-07 12:51:59 +02:00
|
|
|
return {};
|
2021-06-05 15:35:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-07 12:51:59 +02:00
|
|
|
const int device_count = SDL_GetNumAudioDevices(0);
|
2021-06-05 15:35:57 +02:00
|
|
|
for (int i = 0; i < device_count; ++i) {
|
|
|
|
device_list.emplace_back(SDL_GetAudioDeviceName(i, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
return device_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace AudioCore
|