2#include <ossia/detail/config.hpp>
4#if defined(OSSIA_ENABLE_PORTAUDIO)
5#if __has_include(<portaudio.h>)
6#include <ossia/audio/audio_engine.hpp>
7#include <ossia/detail/thread.hpp>
13#define OSSIA_AUDIO_PORTAUDIO 1
17class portaudio_engine final :
public audio_engine
21 std::string name, std::string card_in, std::string card_out,
int inputs,
22 int outputs,
int rate,
int bs, PaHostApiTypeId hostApi)
24 if(Pa_Initialize() != paNoError)
25 throw std::runtime_error(
"Audio error");
27 int card_in_idx = paNoDevice;
28 int card_out_idx = paNoDevice;
30 auto hostApiIndex = Pa_HostApiTypeIdToHostApiIndex(hostApi);
32 for(
int i = 0; i < Pa_GetDeviceCount(); i++)
34 auto info = Pa_GetDeviceInfo(i);
35 if(info->hostApi != hostApiIndex && hostApiIndex != paInDevelopment)
38 auto raw_name = info->name;
40 if(raw_name == card_in && info->maxInputChannels > 0)
46 if(raw_name == card_out && info->maxOutputChannels > 0)
52 if(card_in_idx != paNoDevice && card_out_idx != paNoDevice)
56 auto devInInfo = Pa_GetDeviceInfo(card_in_idx);
59 std::cerr <<
"Audio error: no input device" << std::endl;
64 inputs = std::min(inputs, devInInfo->maxInputChannels);
67 auto devOutInfo = Pa_GetDeviceInfo(card_out_idx);
70 std::cerr <<
"Audio error: no output device" << std::endl;
75 outputs = std::min(outputs, devOutInfo->maxOutputChannels);
78 PaStreamParameters inputParameters;
79 inputParameters.device = card_in_idx;
80 inputParameters.channelCount = inputs;
81 inputParameters.sampleFormat = paFloat32 | paNonInterleaved;
82 inputParameters.suggestedLatency = 0.01;
83 inputParameters.hostApiSpecificStreamInfo =
nullptr;
85 PaStreamParameters outputParameters;
86 outputParameters.device = card_out_idx;
87 outputParameters.channelCount = outputs;
88 outputParameters.sampleFormat = paFloat32 | paNonInterleaved;
89 outputParameters.suggestedLatency = 0.01;
90 outputParameters.hostApiSpecificStreamInfo =
nullptr;
92 PaStreamParameters* actualInput{};
93 if(card_in_idx != paNoDevice && inputs > 0)
94 actualInput = &inputParameters;
95 PaStreamParameters* actualOutput{};
96 if(card_out_idx != paNoDevice && outputs > 0)
97 actualOutput = &outputParameters;
109 auto ec = Pa_OpenStream(
110 &stream, actualInput, actualOutput, rate,
112 paNoFlag, &PortAudioCallback,
this);
113 m_stream.store(stream);
114 if(ec == PaErrorCode::paNoError)
116 ec = Pa_StartStream(stream);
118 if(ec != PaErrorCode::paNoError)
120 std::cerr <<
"Error while starting audio stream: " << Pa_GetErrorText(ec)
122 Pa_CloseStream(stream);
123 m_stream.store(
nullptr);
127 auto info = Pa_GetStreamInfo(stream);
128 this->effective_sample_rate = info->sampleRate;
129 this->effective_buffer_size = bs;
130 this->effective_inputs = actualInput ? actualInput->channelCount : 0;
131 this->effective_outputs = actualOutput ? actualOutput->channelCount : 0;
136 std::cerr <<
"Error while opening audio stream: " << Pa_GetErrorText(ec)
138 m_stream.store(
nullptr);
144 throw std::runtime_error(
"Could not start PortAudio stream");
148 bool running()
const override
150 auto s = m_stream.load();
151 return s && Pa_IsStreamActive(m_stream);
154 ~portaudio_engine()
override
158 if(
auto stream = m_stream.load())
160 auto ec = Pa_StopStream(stream);
161 std::cerr <<
"=== stream stop ===\n";
163 if(ec != PaErrorCode::paNoError)
165 std::cerr <<
"Error while stopping audio stream: " << Pa_GetErrorText(ec)
173 static int clearBuffers(
float** float_output,
unsigned long nframes,
int outs)
175 for(
int i = 0; i < outs; i++)
177 auto chan = float_output[i];
178 for(std::size_t j = 0; j < nframes; j++)
185 static int PortAudioCallback(
186 const void* input,
void* output,
unsigned long nframes,
187 const PaStreamCallbackTimeInfo* timeInfo, PaStreamCallbackFlags statusFlags,
191 static const thread_local auto _
193 ossia::set_thread_name(
"ossia audio 0");
194 ossia::set_thread_pinned(thread_type::Audio, 0);
199 auto& self = *
static_cast<portaudio_engine*
>(userData);
201 auto clt = self.m_stream.load();
203 if(self.stop_processing || !clt)
206 return clearBuffers(((
float**)output), nframes, self.effective_outputs);
209 auto float_input = ((
float*
const*)input);
210 auto float_output = ((
float**)output);
212 ossia::audio_tick_state ts{
213 float_input, float_output, self.effective_inputs, self.effective_outputs,
214 nframes, timeInfo->currentTime};
227 std::atomic<PaStream*> m_stream{};