// Copyright 2018 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include #include #include "audio_core/cubeb_input.h" #include "common/logging/log.h" namespace AudioCore { using SampleQueue = Common::SPSCQueue; struct CubebInput::Impl { cubeb* ctx = nullptr; cubeb_stream* stream = nullptr; std::unique_ptr sample_queue{}; u8 sample_size_in_bytes = 0; 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); }; CubebInput::CubebInput() : impl(std::make_unique()) { if (cubeb_init(&impl->ctx, "Citra Input", nullptr) != CUBEB_OK) { LOG_ERROR(Audio, "cubeb_init failed! Mic will not work properly"); return; } impl->sample_queue = std::make_unique(); } CubebInput::~CubebInput() { if (!impl->ctx) return; if (cubeb_stream_stop(impl->stream) != CUBEB_OK) { LOG_ERROR(Audio, "Error stopping cubeb input stream."); } cubeb_destroy(impl->ctx); } void CubebInput::StartSampling(const Frontend::Mic::Parameters& params) { // Cubeb apparently only supports signed 16 bit PCM (and float32 which the 3ds doesn't support) // TODO resample the input stream if (params.sign == Frontend::Mic::Signedness::Unsigned) { LOG_ERROR(Audio, "Application requested unsupported unsigned pcm format. Falling back to signed"); } if (params.sample_size != 16) { LOG_ERROR(Audio, "Application requested unsupported 8 bit pcm format. Falling back to 16 bits"); } parameters = params; is_sampling = true; impl->sample_size_in_bytes = 2; cubeb_devid input_device = nullptr; cubeb_stream_params input_params; input_params.channels = 1; input_params.layout = CUBEB_LAYOUT_UNDEFINED; input_params.prefs = CUBEB_STREAM_PREF_NONE; input_params.format = CUBEB_SAMPLE_S16LE; input_params.rate = params.sample_rate; u32 latency_frames; if (cubeb_get_min_latency(impl->ctx, &input_params, &latency_frames) != CUBEB_OK) { LOG_ERROR(Audio, "Could not get minimum latency"); } if (cubeb_stream_init(impl->ctx, &impl->stream, "Citra Microphone", input_device, &input_params, nullptr, nullptr, latency_frames, Impl::DataCallback, Impl::StateCallback, impl.get()) != CUBEB_OK) { LOG_CRITICAL(Audio, "Error creating cubeb input stream"); } cubeb_stream_start(impl->stream); int ret = cubeb_stream_set_volume(impl->stream, 1.0); if (ret == CUBEB_ERROR_NOT_SUPPORTED) { LOG_WARNING(Audio, "Unabled to set volume for cubeb input"); } } void CubebInput::StopSampling() { if (impl->stream) { cubeb_stream_stop(impl->stream); } is_sampling = false; } void CubebInput::AdjustSampleRate(u32 sample_rate) { // TODO This should restart the stream with the new sample rate LOG_ERROR(Audio, "AdjustSampleRate unimplemented!"); } Frontend::Mic::Samples CubebInput::Read() { Frontend::Mic::Samples samples{}; Frontend::Mic::Samples queue; while (impl->sample_queue->Pop(queue)) { samples.insert(samples.end(), queue.begin(), queue.end()); } return samples; } long CubebInput::Impl::DataCallback(cubeb_stream* stream, void* user_data, const void* input_buffer, void* output_buffer, long num_frames) { Impl* impl = static_cast(user_data); if (!impl) { return 0; } u8 const* data = reinterpret_cast(input_buffer); std::vector samples{data, data + num_frames * impl->sample_size_in_bytes}; impl->sample_queue->Push(samples); // returning less than num_frames here signals cubeb to stop sampling return num_frames; } void CubebInput::Impl::StateCallback(cubeb_stream* stream, void* user_data, cubeb_state state) {} std::vector ListCubebInputDevices() { std::vector device_list; cubeb* ctx; if (cubeb_init(&ctx, "Citra Input Device Enumerator", nullptr) != CUBEB_OK) { LOG_CRITICAL(Audio, "cubeb_init failed"); return {}; } cubeb_device_collection collection; if (cubeb_enumerate_devices(ctx, CUBEB_DEVICE_TYPE_INPUT, &collection) != CUBEB_OK) { LOG_WARNING(Audio_Sink, "Audio input 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.state == CUBEB_DEVICE_STATE_ENABLED && device.friendly_name) { device_list.emplace_back(device.friendly_name); } } cubeb_device_collection_destroy(ctx, &collection); } cubeb_destroy(ctx); return device_list; } } // namespace AudioCore