// 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 { struct CubebInput::Impl { // unsigned int sample_rate = 0; // std::vector device_list; cubeb* ctx = nullptr; cubeb_stream* stream = nullptr; bool looped_buffer; u8* buffer; u32 buffer_size; u32 initial_offset; u32 offset; u32 audio_buffer_size; 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; } } 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(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"); } impl->buffer = backing_memory; impl->buffer_size = backing_memory_size; impl->audio_buffer_size = params.buffer_size; impl->offset = params.buffer_offset; impl->looped_buffer = params.buffer_loop; 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); } void CubebInput::StopSampling() { if (impl->stream) { cubeb_stream_stop(impl->stream); } } void CubebInput::AdjustSampleRate(u32 sample_rate) { // TODO This should restart the stream with the new sample rate LOG_ERROR(Audio, "AdjustSampleRate unimplemented!"); } 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); const u8* data = reinterpret_cast(input_buffer); if (!impl) { return 0; } if (!impl->buffer) { return 0; } u64 total_written = 0; u64 to_write = num_frames; u64 remaining_space = impl->audio_buffer_size - impl->offset; if (to_write > remaining_space) { to_write = remaining_space; } std::memcpy(impl->buffer + impl->offset, data, to_write); impl->offset += to_write; total_written += to_write; if (impl->looped_buffer && num_frames > total_written) { impl->offset = impl->initial_offset; to_write = num_frames - to_write; std::memcpy(impl->buffer + impl->offset, data, to_write); impl->offset += to_write; total_written += to_write; } // The last 4 bytes of the shared memory contains the latest offset // so update that as well https://www.3dbrew.org/wiki/MIC_Shared_Memory std::memcpy(impl->buffer + (impl->buffer_size - sizeof(u32)), reinterpret_cast(&impl->offset), sizeof(u32)); // returning less than num_frames here signals cubeb to stop sampling return total_written; } 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 (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