3 <alsa/asoundlib.h>) && __has_include(<zita-alsa-pcmi-ardour.h>) && !defined(__EMSCRIPTEN__)
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>
12#include <alsa/asoundlib.h>
13#include <kfr/base/conversion.hpp>
22#include <zita-alsa-pcmi-ardour.h>
24#define OSSIA_AUDIO_ALSA 1
28class alsa_engine final :
public audio_engine
32 std::string card_in, std::string card_out,
int inputs,
int outputs,
int rate,
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);
39 if(m_client->state() != 0)
41 throw std::runtime_error(
"Could not open ALSA");
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(); }};
53 ~alsa_engine()
override
59 m_stop_token.store(
false, std::memory_order_relaxed);
60 assert(m_thread.joinable());
67 bool running()
const override
76 auto& client = *this->m_client.get();
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);
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);
97 while(m_stop_token.load(std::memory_order_relaxed))
99 k = client.pcm_wait();
101 while(k >= effective_buffer_size)
103 client.capt_init(effective_buffer_size);
104 for(
int c = 0; c < effective_inputs; c++)
106 c, buf_capture.data() + c, effective_buffer_size, effective_inputs);
107 client.capt_done(effective_buffer_size);
110 ptrs_capture.data(), buf_capture.data(), effective_inputs,
111 effective_buffer_size);
114 buf_playback_deint.data(), effective_buffer_size * effective_outputs, 0);
116 process(ptrs_capture.data(), ptrs_playback.data(), effective_buffer_size);
119 buf_playback.data(), (
const float**)ptrs_playback.data(), effective_outputs,
120 effective_buffer_size);
122 client.play_init(effective_buffer_size);
123 for(
int c = 0; c < effective_outputs; c++)
125 c, buf_playback.data() + c, effective_buffer_size, effective_outputs);
127 for(std::size_t i = effective_outputs; i < client.nplay(); i++)
128 client.clear_chan(i, effective_buffer_size);
130 client.play_done(effective_buffer_size);
132 k -= effective_buffer_size;
142 ossia::set_thread_name(
"ossia audio 0");
143 ossia::set_thread_pinned(thread_type::Audio, 0);
146 void process(
float** float_input,
float** float_output, uint64_t nframes)
148 auto& self = *
static_cast<alsa_engine*
>(
this);
151 const auto inputs = this->effective_inputs;
152 const auto outputs = this->effective_outputs;
153 if(self.stop_processing)
159 using namespace std::chrono;
160 ossia::audio_tick_state ts{
166 duration_cast<nanoseconds>(m_last_time - m_start_time).count() / 1e9};
170 m_last_time = clk::now();
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{};