OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
alsa_protocol.hpp
1#pragma once
2#if __has_include( \
3 <alsa/asoundlib.h>) && __has_include(<zita-alsa-pcmi-ardour.h>) && !defined(__EMSCRIPTEN__)
4
5#include <ossia/audio/audio_engine.hpp>
6#include <ossia/dataflow/float_to_sample.hpp>
7#include <ossia/dataflow/sample_to_float.hpp>
9#include <ossia/detail/pod_vector.hpp>
10#include <ossia/detail/thread.hpp>
11
12#include <alsa/asoundlib.h>
13#include <kfr/base/conversion.hpp>
14
15#include <cmath>
16
17#include <atomic>
18#include <string_view>
19#include <thread>
20#include <vector>
21
22#include <zita-alsa-pcmi-ardour.h>
23
24#define OSSIA_AUDIO_ALSA 1
25
26namespace ossia
27{
28class alsa_engine final : public audio_engine
29{
30public:
31 alsa_engine(
32 std::string card_in, std::string card_out, int inputs, int outputs, int rate,
33 int bs)
34 {
35 m_client = std::make_unique<Alsa_pcmi>(
36 card_in.c_str(), card_out.c_str(), nullptr, rate, bs, 2, 2,
37 Alsa_pcmi::DEBUG_ALL);
38
39 if(m_client->state() != 0)
40 {
41 throw std::runtime_error("Could not open ALSA");
42 }
43
44 effective_sample_rate = rate;
45 effective_buffer_size = bs;
46 effective_inputs = std::min(inputs, (int)m_client->ncapt());
47 effective_outputs = std::min(outputs, (int)m_client->nplay());
48 m_thread = std::thread{[this] { thread_duplex(); }};
49
50 m_activated = true;
51 }
52
53 ~alsa_engine() override
54 {
55 stop();
56
57 if(m_client)
58 {
59 m_stop_token.store(false, std::memory_order_relaxed);
60 assert(m_thread.joinable());
61 m_thread.join();
62
63 m_client.reset();
64 }
65 }
66
67 bool running() const override
68 {
69 if(!m_client)
70 return false;
71 return m_activated;
72 }
73
74 void thread_duplex()
75 {
76 auto& client = *this->m_client.get();
77
78 std::vector<float> buf_capture;
79 buf_capture.resize(effective_buffer_size * effective_inputs);
80 std::vector<float> buf_capture_deint;
81 buf_capture_deint.resize(effective_buffer_size * effective_inputs);
82 std::vector<float*> ptrs_capture;
83 for(int i = 0; i < effective_inputs; i++)
84 ptrs_capture.push_back(buf_capture_deint.data() + i * effective_buffer_size);
85
86 std::vector<float> buf_playback;
87 buf_playback.resize(effective_buffer_size * effective_outputs);
88 std::vector<float> buf_playback_deint;
89 buf_playback_deint.resize(effective_buffer_size * effective_outputs);
90 std::vector<float*> ptrs_playback;
91 for(int i = 0; i < effective_outputs; i++)
92 ptrs_playback.push_back(buf_playback_deint.data() + i * effective_buffer_size);
93
94 client.pcm_start();
95
96 int k = 0;
97 while(m_stop_token.load(std::memory_order_relaxed))
98 {
99 k = client.pcm_wait();
100
101 while(k >= effective_buffer_size)
102 {
103 client.capt_init(effective_buffer_size);
104 for(int c = 0; c < effective_inputs; c++)
105 client.capt_chan(
106 c, buf_capture.data() + c, effective_buffer_size, effective_inputs);
107 client.capt_done(effective_buffer_size);
108
109 kfr::deinterleave(
110 ptrs_capture.data(), buf_capture.data(), effective_inputs,
111 effective_buffer_size);
112
113 std::fill_n(
114 buf_playback_deint.data(), effective_buffer_size * effective_outputs, 0);
115
116 process(ptrs_capture.data(), ptrs_playback.data(), effective_buffer_size);
117
118 kfr::interleave(
119 buf_playback.data(), (const float**)ptrs_playback.data(), effective_outputs,
120 effective_buffer_size);
121
122 client.play_init(effective_buffer_size);
123 for(int c = 0; c < effective_outputs; c++)
124 client.play_chan(
125 c, buf_playback.data() + c, effective_buffer_size, effective_outputs);
126
127 for(std::size_t i = effective_outputs; i < client.nplay(); i++)
128 client.clear_chan(i, effective_buffer_size);
129
130 client.play_done(effective_buffer_size);
131
132 k -= effective_buffer_size;
133 }
134 }
135
136 client.pcm_stop();
137 }
138
139private:
140 void init_thread()
141 {
142 ossia::set_thread_name("ossia audio 0");
143 ossia::set_thread_pinned(thread_type::Audio, 0);
144 }
145
146 void process(float** float_input, float** float_output, uint64_t nframes)
147 {
148 auto& self = *static_cast<alsa_engine*>(this);
149 self.tick_start();
150
151 const auto inputs = this->effective_inputs;
152 const auto outputs = this->effective_outputs;
153 if(self.stop_processing)
154 {
155 self.tick_clear();
156 return;
157 }
158
159 using namespace std::chrono;
160 ossia::audio_tick_state ts{
161 float_input,
162 float_output,
163 (int)inputs,
164 (int)outputs,
165 nframes,
166 duration_cast<nanoseconds>(m_last_time - m_start_time).count() / 1e9};
167 self.audio_tick(ts);
168
169 self.tick_end();
170 m_last_time = clk::now();
171 }
172
173 std::unique_ptr<Alsa_pcmi> m_client;
174 std::thread m_thread;
175 using clk = std::chrono::steady_clock;
176 clk::time_point m_start_time{};
177 clk::time_point m_last_time{};
178 std::atomic_bool m_stop_token{true};
179 std::atomic_bool m_activated{};
180};
181}
182
183#endif
Definition git_info.h:7