forked from eden-emu/eden
		
	
		
			
				
	
	
		
			163 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			163 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // 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;
 | |
|         if (output_device.empty()) {
 | |
|             dev = SDL_OpenAudioDevice(nullptr, 0, &spec, &obtained, 0);
 | |
|         } else {
 | |
|             dev = SDL_OpenAudioDevice(output_device.c_str(), 0, &spec, &obtained, 0);
 | |
|         }
 | |
| 
 | |
|         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();
 | |
|     }
 | |
| }
 | |
| 
 | |
| SDLSink::~SDLSink() = default;
 | |
| 
 | |
| 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());
 | |
|             return {};
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const int device_count = SDL_GetNumAudioDevices(0);
 | |
|     for (int i = 0; i < device_count; ++i) {
 | |
|         device_list.emplace_back(SDL_GetAudioDeviceName(i, 0));
 | |
|     }
 | |
| 
 | |
|     return device_list;
 | |
| }
 | |
| 
 | |
| } // namespace AudioCore
 | 
