mirror of
https://github.com/Lime3DS/Lime3DS
synced 2025-01-02 04:20:50 -06:00
90082268dc
We can hide the direct array from external view and instead provide functions to retrieve the necessary info. This has the benefit of completely hiding the makeup of the SinkDetails structure from the rest of the code. Given that this makes the array hidden, we can also make the array constexpr by altering the members slightly. This gets rid of several static constructor calls related to std::vector and std::function. Now we don't have heap allocations here that need to occur before the program can even enter main(). It also has the benefit of saving a little bit of heap space, but this doesn't matter too much, since the savings in that regard are pretty tiny.
193 lines
6.2 KiB
C++
193 lines
6.2 KiB
C++
// Copyright 2018 Citra Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <cstdarg>
|
|
#include <mutex>
|
|
#include <vector>
|
|
#include <cubeb/cubeb.h>
|
|
#include "audio_core/audio_types.h"
|
|
#include "audio_core/cubeb_sink.h"
|
|
#include "common/logging/log.h"
|
|
|
|
namespace AudioCore {
|
|
|
|
struct CubebSink::Impl {
|
|
unsigned int sample_rate = 0;
|
|
|
|
cubeb* ctx = nullptr;
|
|
cubeb_stream* stream = nullptr;
|
|
|
|
std::function<void(s16*, std::size_t)> cb;
|
|
|
|
static long DataCallback(cubeb_stream* stream, void* user_data, const void* input_buffer,
|
|
void* output_buffer, long num_frames);
|
|
static void StateCallback(cubeb_stream* stream, void* user_data, cubeb_state state);
|
|
static void LogCallback(char const* fmt, ...);
|
|
};
|
|
|
|
CubebSink::CubebSink(std::string_view target_device_name) : impl(std::make_unique<Impl>()) {
|
|
if (cubeb_init(&impl->ctx, "Citra", nullptr) != CUBEB_OK) {
|
|
LOG_CRITICAL(Audio_Sink, "cubeb_init failed");
|
|
return;
|
|
}
|
|
cubeb_set_log_callback(CUBEB_LOG_NORMAL, &Impl::LogCallback);
|
|
|
|
impl->sample_rate = native_sample_rate;
|
|
|
|
cubeb_stream_params params;
|
|
params.rate = impl->sample_rate;
|
|
params.channels = 2;
|
|
params.layout = CUBEB_LAYOUT_STEREO;
|
|
params.format = CUBEB_SAMPLE_S16NE;
|
|
params.prefs = CUBEB_STREAM_PREF_NONE;
|
|
|
|
u32 minimum_latency = 100 * impl->sample_rate / 1000; // Firefox default
|
|
if (cubeb_get_min_latency(impl->ctx, ¶ms, &minimum_latency) != CUBEB_OK) {
|
|
LOG_CRITICAL(Audio_Sink, "Error getting minimum latency");
|
|
}
|
|
|
|
cubeb_devid output_device = nullptr;
|
|
if (target_device_name != auto_device_name && !target_device_name.empty()) {
|
|
cubeb_device_collection collection;
|
|
if (cubeb_enumerate_devices(impl->ctx, CUBEB_DEVICE_TYPE_OUTPUT, &collection) != CUBEB_OK) {
|
|
LOG_WARNING(Audio_Sink, "Audio output device enumeration not supported");
|
|
} else {
|
|
const auto collection_end{collection.device + collection.count};
|
|
const auto device{
|
|
std::find_if(collection.device, collection_end, [&](const cubeb_device_info& info) {
|
|
return info.friendly_name != nullptr &&
|
|
target_device_name == info.friendly_name;
|
|
})};
|
|
if (device != collection_end) {
|
|
output_device = device->devid;
|
|
}
|
|
cubeb_device_collection_destroy(impl->ctx, &collection);
|
|
}
|
|
}
|
|
|
|
int stream_err = cubeb_stream_init(impl->ctx, &impl->stream, "CitraAudio", nullptr, nullptr,
|
|
output_device, ¶ms, std::max(512u, minimum_latency),
|
|
&Impl::DataCallback, &Impl::StateCallback, impl.get());
|
|
if (stream_err != CUBEB_OK) {
|
|
switch (stream_err) {
|
|
case CUBEB_ERROR:
|
|
default:
|
|
LOG_CRITICAL(Audio_Sink, "Error initializing cubeb stream ({})", stream_err);
|
|
break;
|
|
case CUBEB_ERROR_INVALID_FORMAT:
|
|
LOG_CRITICAL(Audio_Sink, "Invalid format when initializing cubeb stream");
|
|
break;
|
|
case CUBEB_ERROR_DEVICE_UNAVAILABLE:
|
|
LOG_CRITICAL(Audio_Sink, "Device unavailable when initializing cubeb stream");
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (cubeb_stream_start(impl->stream) != CUBEB_OK) {
|
|
LOG_CRITICAL(Audio_Sink, "Error starting cubeb stream");
|
|
return;
|
|
}
|
|
}
|
|
|
|
CubebSink::~CubebSink() {
|
|
if (!impl->ctx) {
|
|
return;
|
|
}
|
|
|
|
impl->cb = nullptr;
|
|
|
|
if (cubeb_stream_stop(impl->stream) != CUBEB_OK) {
|
|
LOG_CRITICAL(Audio_Sink, "Error stopping cubeb stream");
|
|
}
|
|
|
|
cubeb_stream_destroy(impl->stream);
|
|
cubeb_destroy(impl->ctx);
|
|
}
|
|
|
|
unsigned int CubebSink::GetNativeSampleRate() const {
|
|
if (!impl->ctx)
|
|
return native_sample_rate;
|
|
|
|
return impl->sample_rate;
|
|
}
|
|
|
|
void CubebSink::SetCallback(std::function<void(s16*, std::size_t)> cb) {
|
|
impl->cb = cb;
|
|
}
|
|
|
|
long CubebSink::Impl::DataCallback(cubeb_stream* stream, void* user_data, const void* input_buffer,
|
|
void* output_buffer, long num_frames) {
|
|
Impl* impl = static_cast<Impl*>(user_data);
|
|
s16* buffer = reinterpret_cast<s16*>(output_buffer);
|
|
|
|
if (!impl || !impl->cb) {
|
|
LOG_DEBUG(Audio_Sink, "Emitting zeros");
|
|
std::memset(output_buffer, 0, num_frames * 2 * sizeof(s16));
|
|
return num_frames;
|
|
}
|
|
|
|
impl->cb(buffer, num_frames);
|
|
|
|
return num_frames;
|
|
}
|
|
|
|
void CubebSink::Impl::StateCallback(cubeb_stream* stream, void* user_data, cubeb_state state) {
|
|
switch (state) {
|
|
case CUBEB_STATE_STARTED:
|
|
LOG_INFO(Audio_Sink, "Cubeb Audio Stream Started");
|
|
break;
|
|
case CUBEB_STATE_STOPPED:
|
|
LOG_INFO(Audio_Sink, "Cubeb Audio Stream Stopped");
|
|
break;
|
|
case CUBEB_STATE_DRAINED:
|
|
LOG_INFO(Audio_Sink, "Cubeb Audio Stream Drained");
|
|
break;
|
|
case CUBEB_STATE_ERROR:
|
|
LOG_CRITICAL(Audio_Sink, "Cubeb Audio Stream Errored");
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CubebSink::Impl::LogCallback(char const* format, ...) {
|
|
std::array<char, 512> buffer;
|
|
std::va_list args;
|
|
va_start(args, format);
|
|
#ifdef _MSC_VER
|
|
vsprintf_s(buffer.data(), buffer.size(), format, args);
|
|
#else
|
|
vsnprintf(buffer.data(), buffer.size(), format, args);
|
|
#endif
|
|
va_end(args);
|
|
buffer.back() = '\0';
|
|
LOG_INFO(Audio_Sink, "{}", buffer.data());
|
|
}
|
|
|
|
std::vector<std::string> ListCubebSinkDevices() {
|
|
std::vector<std::string> device_list;
|
|
cubeb* ctx;
|
|
|
|
if (cubeb_init(&ctx, "CitraEnumerator", nullptr) != CUBEB_OK) {
|
|
LOG_CRITICAL(Audio_Sink, "cubeb_init failed");
|
|
return {};
|
|
}
|
|
|
|
cubeb_device_collection collection;
|
|
if (cubeb_enumerate_devices(ctx, CUBEB_DEVICE_TYPE_OUTPUT, &collection) != CUBEB_OK) {
|
|
LOG_WARNING(Audio_Sink, "Audio output device enumeration not supported");
|
|
} else {
|
|
for (std::size_t i = 0; i < collection.count; i++) {
|
|
const cubeb_device_info& device = collection.device[i];
|
|
if (device.friendly_name) {
|
|
device_list.emplace_back(device.friendly_name);
|
|
}
|
|
}
|
|
cubeb_device_collection_destroy(ctx, &collection);
|
|
}
|
|
|
|
cubeb_destroy(ctx);
|
|
return device_list;
|
|
}
|
|
|
|
} // namespace AudioCore
|