| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | // Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include "audio_core/cubeb_sink.h"
 | 
					
						
							|  |  |  | #include "audio_core/stream.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  | #include "common/ring_buffer.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace AudioCore { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  | class CubebSinkStream final : public SinkStream { | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |     CubebSinkStream(cubeb* ctx, u32 sample_rate, u32 num_channels_, cubeb_devid output_device, | 
					
						
							|  |  |  |                     const std::string& name) | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |         : ctx{ctx}, num_channels{num_channels_} { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (num_channels == 6) { | 
					
						
							|  |  |  |             // 6-channel audio does not seem to work with cubeb + SDL, so we downsample this to 2
 | 
					
						
							|  |  |  |             // channel for now
 | 
					
						
							|  |  |  |             is_6_channel = true; | 
					
						
							|  |  |  |             num_channels = 2; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         cubeb_stream_params params{}; | 
					
						
							|  |  |  |         params.rate = sample_rate; | 
					
						
							|  |  |  |         params.channels = num_channels; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |         params.format = CUBEB_SAMPLE_S16NE; | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |         params.layout = num_channels == 1 ? CUBEB_LAYOUT_MONO : CUBEB_LAYOUT_STEREO; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |         u32 minimum_latency{}; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |         if (cubeb_get_min_latency(ctx, ¶ms, &minimum_latency) != CUBEB_OK) { | 
					
						
							|  |  |  |             LOG_CRITICAL(Audio_Sink, "Error getting minimum latency"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-02 18:54:25 -04:00
										 |  |  |         if (cubeb_stream_init(ctx, &stream_backend, name.c_str(), nullptr, nullptr, output_device, | 
					
						
							|  |  |  |                               ¶ms, std::max(512u, minimum_latency), | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |                               &CubebSinkStream::DataCallback, &CubebSinkStream::StateCallback, | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |                               this) != CUBEB_OK) { | 
					
						
							|  |  |  |             LOG_CRITICAL(Audio_Sink, "Error initializing cubeb stream"); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cubeb_stream_start(stream_backend) != CUBEB_OK) { | 
					
						
							|  |  |  |             LOG_CRITICAL(Audio_Sink, "Error starting cubeb stream"); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |     ~CubebSinkStream() { | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |         if (!ctx) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cubeb_stream_stop(stream_backend) != CUBEB_OK) { | 
					
						
							|  |  |  |             LOG_CRITICAL(Audio_Sink, "Error stopping cubeb stream"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         cubeb_stream_destroy(stream_backend); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-04 00:03:12 -04:00
										 |  |  |     void EnqueueSamples(u32 num_channels, const std::vector<s16>& samples) override { | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |         if (is_6_channel) { | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |             // Downsample 6 channels to 2
 | 
					
						
							| 
									
										
										
										
											2018-08-04 00:03:12 -04:00
										 |  |  |             const size_t sample_count_copy_size = samples.size() * 2; | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |             std::vector<s16> buf; | 
					
						
							|  |  |  |             buf.reserve(sample_count_copy_size); | 
					
						
							| 
									
										
										
										
											2018-08-04 00:03:12 -04:00
										 |  |  |             for (size_t i = 0; i < samples.size(); i += num_channels) { | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |                 buf.push_back(samples[i]); | 
					
						
							|  |  |  |                 buf.push_back(samples[i + 1]); | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |             queue.Push(buf); | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         queue.Push(samples); | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |     size_t SamplesInQueue(u32 num_channels) const override { | 
					
						
							| 
									
										
										
										
											2018-08-23 14:33:03 +02:00
										 |  |  |         if (!ctx) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |         return queue.Size() / num_channels; | 
					
						
							| 
									
										
										
										
											2018-08-23 14:33:03 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |     u32 GetNumChannels() const { | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |         return num_channels; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 16:15:40 +01:00
										 |  |  |     u32 GetNumChannelsInQueue() const { | 
					
						
							|  |  |  |         return num_channels == 1 ? 1 : 2; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | private: | 
					
						
							|  |  |  |     std::vector<std::string> device_list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cubeb* ctx{}; | 
					
						
							|  |  |  |     cubeb_stream* stream_backend{}; | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |     u32 num_channels{}; | 
					
						
							|  |  |  |     bool is_6_channel{}; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |     Common::RingBuffer<s16, 0x10000> queue; | 
					
						
							| 
									
										
										
										
											2018-09-08 16:15:40 +01:00
										 |  |  |     std::array<s16, 2> last_frame; | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CubebSink::CubebSink(std::string target_device_name) { | 
					
						
							|  |  |  |     if (cubeb_init(&ctx, "yuzu", nullptr) != CUBEB_OK) { | 
					
						
							|  |  |  |         LOG_CRITICAL(Audio_Sink, "cubeb_init failed"); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (target_device_name != auto_device_name && !target_device_name.empty()) { | 
					
						
							|  |  |  |         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 { | 
					
						
							|  |  |  |             const auto collection_end{collection.device + collection.count}; | 
					
						
							|  |  |  |             const auto device{std::find_if(collection.device, collection_end, | 
					
						
							|  |  |  |                                            [&](const cubeb_device_info& device) { | 
					
						
							|  |  |  |                                                return target_device_name == device.friendly_name; | 
					
						
							|  |  |  |                                            })}; | 
					
						
							|  |  |  |             if (device != collection_end) { | 
					
						
							|  |  |  |                 output_device = device->devid; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             cubeb_device_collection_destroy(ctx, &collection); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CubebSink::~CubebSink() { | 
					
						
							|  |  |  |     if (!ctx) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& sink_stream : sink_streams) { | 
					
						
							|  |  |  |         sink_stream.reset(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cubeb_destroy(ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-02 18:54:25 -04:00
										 |  |  | SinkStream& CubebSink::AcquireSinkStream(u32 sample_rate, u32 num_channels, | 
					
						
							|  |  |  |                                          const std::string& name) { | 
					
						
							| 
									
										
										
										
											2018-08-03 14:50:53 -04:00
										 |  |  |     sink_streams.push_back( | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |         std::make_unique<CubebSinkStream>(ctx, sample_rate, num_channels, output_device, name)); | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |     return *sink_streams.back(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  | long CubebSinkStream::DataCallback(cubeb_stream* stream, void* user_data, const void* input_buffer, | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |                                   void* output_buffer, long num_frames) { | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |     CubebSinkStream* impl = static_cast<CubebSinkStream*>(user_data); | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |     u8* buffer = reinterpret_cast<u8*>(output_buffer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!impl) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 16:15:40 +01:00
										 |  |  |     const size_t num_channels = impl->GetNumChannelsInQueue(); | 
					
						
							|  |  |  |     const size_t max_samples_to_write = num_channels * num_frames; | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  |     const size_t samples_written = impl->queue.Pop(buffer, max_samples_to_write); | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 16:15:40 +01:00
										 |  |  |     if (samples_written >= num_channels) { | 
					
						
							|  |  |  |         std::memcpy(&impl->last_frame[0], buffer + (samples_written - num_channels) * sizeof(s16), | 
					
						
							|  |  |  |                     num_channels * sizeof(s16)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Fill the rest of the frames with last_frame
 | 
					
						
							|  |  |  |     for (size_t i = samples_written; i < max_samples_to_write; i += num_channels) { | 
					
						
							|  |  |  |         std::memcpy(buffer + i * sizeof(s16), &impl->last_frame[0], num_channels * sizeof(s16)); | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return num_frames; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-08 14:55:11 +01:00
										 |  |  | void CubebSinkStream::StateCallback(cubeb_stream* stream, void* user_data, cubeb_state state) {} | 
					
						
							| 
									
										
										
										
											2018-07-28 13:44:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | std::vector<std::string> ListCubebSinkDevices() { | 
					
						
							|  |  |  |     std::vector<std::string> device_list; | 
					
						
							|  |  |  |     cubeb* ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cubeb_init(&ctx, "Citra Device Enumerator", 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 (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
 |