OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
sdl_protocol.hpp
1#pragma once
2#include <ossia/detail/config.hpp>
3
4#if defined(OSSIA_ENABLE_SDL)
5#if __has_include(<SDL2/SDL_audio.h>)
6#include <SDL2/SDL_config.h>
7#if !defined(SDL_AUDIO_DISABLED)
8#include <ossia/audio/audio_engine.hpp>
9#include <ossia/detail/thread.hpp>
10
11#include <SDL2/SDL.h>
12#include <SDL2/SDL_audio.h>
13
14#define OSSIA_AUDIO_SDL 1
15
16namespace ossia
17{
18class sdl_protocol final : public audio_engine
19{
20 static constexpr int inputs = 0;
21 static constexpr int outputs = 2;
22
23public:
24 sdl_protocol(int rate, int bs)
25 {
26 SDL_Init(SDL_INIT_AUDIO);
27 m_desired.freq = rate;
28 m_desired.format = AUDIO_F32SYS;
29 m_desired.channels = outputs;
30 m_desired.samples = bs;
31 m_desired.callback = SDLCallback;
32 m_desired.userdata = this;
33
34 m_deviceId = SDL_OpenAudioDevice(nullptr, 0, &m_desired, &m_obtained, 0);
35
36 if(m_deviceId < 2)
37 {
38 using namespace std::literals;
39 throw std::runtime_error("SDL: Couldn't open audio: "s + SDL_GetError());
40 }
41
42 this->effective_sample_rate = m_obtained.freq;
43 this->effective_buffer_size = m_obtained.samples;
44 this->effective_inputs = 0;
45 this->effective_outputs = m_obtained.channels;
46
47 SDL_PauseAudioDevice(m_deviceId, 0);
48 }
49
50 ~sdl_protocol() override
51 {
52 stop();
53 SDL_CloseAudioDevice(m_deviceId);
54 SDL_Quit();
55 }
56
57 bool running() const override
58 {
59 return SDL_GetAudioDeviceStatus(m_deviceId) == SDL_AUDIO_PLAYING;
60 }
61
62private:
63 static void SDLCallback(void* userData, Uint8* data, int bytes)
64 {
65 [[maybe_unused]]
66 static const thread_local auto _
67 = [] {
68 ossia::set_thread_name("ossia audio 0");
69 ossia::set_thread_pinned(thread_type::Audio, 0);
70 return 0;
71 }();
72
73 auto& self = *static_cast<sdl_protocol*>(userData);
74 self.tick_start();
75 if(!self.m_start)
76 self.m_start = std::chrono::steady_clock::now();
77
78 auto audio_out = reinterpret_cast<float*>(data);
79 const int out_chan = self.m_obtained.channels;
80 const int frames = self.m_obtained.samples;
81 assert(out_chan > 0);
82 assert(frames > 0);
83 assert(frames * out_chan * sizeof(float) == bytes);
84
85 if(self.stop_processing)
86 {
87 self.tick_clear();
88 memset(data, 0, bytes);
89 return;
90 }
91
92 {
93 auto float_data = (float*)alloca(sizeof(float) * frames * out_chan);
94 memset(float_data, 0, sizeof(sizeof(float) * frames * out_chan));
95
96 auto float_output = (float**)alloca(sizeof(float*) * out_chan);
97
98 for(int c = 0; c < out_chan; c++)
99 {
100 float_output[c] = float_data + c * frames;
101 }
102
103 // if one day there's input... samples[j++] / 32768.;
104
105 auto now = std::chrono::steady_clock::now();
106 auto nsecs
107 = std::chrono::duration_cast<std::chrono::nanoseconds>(now - *self.m_start)
108 .count()
109 / 1e9;
110
111 ossia::audio_tick_state ts{nullptr, float_output, 0,
112 out_chan, (uint64_t)frames, nsecs};
113 self.audio_tick(ts);
114
115 for(int j = 0; j < frames; j++)
116 for(int c = 0; c < out_chan; c++)
117 *audio_out++ = float_output[c][j];
118
119 self.tick_end();
120 self.m_total_frames += frames;
121 }
122 }
123
124 SDL_AudioDeviceID m_deviceId{};
125 SDL_AudioSpec m_desired, m_obtained;
126 uint64_t m_total_frames{};
127 std::optional<std::chrono::steady_clock::time_point> m_start;
128};
129}
130
131#endif
132#endif
133#endif
Definition git_info.h:7