19 std::shared_ptr<AEffectWrapper> fx{};
25 this->set_not_fp_safe();
34 ossia::value_port* port{};
38 int32_t opcode, int32_t index = 0, intptr_t value = 0,
void* ptr =
nullptr,
41 fx->dispatch(opcode, index, value, ptr, opt);
45 ossia::small_vector<vst_control, 16> controls;
49 for(vst_control& p : controls)
51 const auto& vec = p.port->get_data();
54 auto t = ossia::convert<float>(last(vec));
56 p.value = ossia::clamp<float>(t, 0.f, 1.f);
57 fx->fx->setParameter(fx->fx, p.idx, p.value);
62 auto& prepareInput(int64_t offset, int64_t samples)
64 const auto bs = offset + samples;
65 auto& p = *m_inlets[0]->template target<ossia::audio_port>();
70 p.channel(0).resize(bs);
71 p.channel(1).resize(bs);
76 p.channel(0).resize(bs);
77 p.channel(1).assign(p.channel(0).begin(), p.channel(0).end());
90 auto& prepareOutput(int64_t offset, int64_t samples)
92 const auto bs = offset + samples;
93 auto& p = *m_outlets[0]->template target<ossia::audio_port>();
96 chan.resize(bs, boost::container::default_init);
100 void setupTimeInfo(
const ossia::token_request& tk, ossia::exec_state_facade st)
102 auto& time_info = fx->info;
103 time_info.samplePos = this->m_processed_frames;
104 time_info.sampleRate = st.sampleRate();
105 time_info.nanoSeconds = st.currentDate() - st.startDate();
106 time_info.ppqPos = tk.musical_start_position;
107 time_info.tempo = tk.tempo;
108 time_info.barStartPos = tk.musical_start_last_bar;
109 time_info.cycleStartPos = 0.;
110 time_info.cycleEndPos = 0.;
111 time_info.timeSigNumerator = tk.signature.upper;
112 time_info.timeSigDenominator = tk.signature.lower;
113 time_info.smpteOffset = 0;
114 time_info.smpteFrameRate = 0;
115 time_info.samplesToNextClock = 0;
116 time_info.flags = kVstTransportPlaying | kVstNanosValid | kVstPpqPosValid
117 | kVstTempoValid | kVstBarsValid | kVstTimeSigValid
126 static constexpr bool synth = IsSynth;
127 VstSpeakerArrangement i_arr{};
128 VstSpeakerArrangement o_arr{};
131 vst_node(std::shared_ptr<AEffectWrapper> dat,
int sampleRate,
int bs)
136 m_inlets.push_back(
new ossia::audio_inlet);
137 if constexpr(IsSynth)
138 m_inlets.push_back(
new ossia::midi_inlet);
141 m_outlets.push_back(
new ossia::audio_outlet);
144 memset(&i_arr, 0,
sizeof(i_arr));
145 memset(&o_arr, 0,
sizeof(o_arr));
146 i_arr.type = kSpeakerArrStereo;
147 i_arr.numChannels = 2;
148 i_arr.speakers[0].type = kSpeakerL;
149 i_arr.speakers[1].type = kSpeakerR;
151 o_arr.type = kSpeakerArrStereo;
152 o_arr.numChannels = 2;
153 o_arr.speakers[0].type = kSpeakerL;
154 o_arr.speakers[1].type = kSpeakerR;
155 dispatch(effSetSpeakerArrangement, 0, (intptr_t)&i_arr, (
void*)&o_arr, 0);
158 dispatch(effSetSampleRate, 0, sampleRate,
nullptr, sampleRate);
159 dispatch(effSetBlockSize, 0, bs,
nullptr, bs);
161 effSetProcessPrecision, 0,
162 UseDouble ? kVstProcessPrecision64 : kVstProcessPrecision32);
163 dispatch(effMainsChanged, 0, 1);
164 dispatch(effStartProcess);
166 fx->fx->resvd2 =
reinterpret_cast<intptr_t
>(
this);
172 dispatch(effStopProcess);
173 dispatch(effMainsChanged, 0, 0);
176 std::string label()
const noexcept override
178 char paramName[512] = {0};
179 fx->fx->dispatcher(fx->fx, effGetProductString, 0, 0, paramName, 0.0f);
183 void all_notes_off()
noexcept override
185 if constexpr(IsSynth)
189 constexpr auto sz =
sizeof(VstEvents) +
sizeof(
void*) * 32 * 2;
190 VstEvents* events = (VstEvents*)alloca(sz);
191 std::memset(events, 0, sz);
193 events->numEvents = 32;
195 VstMidiEvent ev[64] = {};
196 memset(&ev, 0,
sizeof(ev));
199 for(
int i = 0; i < 16; i++)
202 e.type = kVstMidiType;
203 e.flags = kVstMidiEventIsRealtime;
204 e.byteSize =
sizeof(VstMidiEvent);
206 e.midiData[0] = (char)(uint8_t)176 + i;
207 e.midiData[1] = (char)(uint8_t)123;
211 events->events[i] =
reinterpret_cast<VstEvent*
>(&e);
215 for(
int i = 0; i < 16; i++)
217 auto& e = ev[16 + i];
218 e.type = kVstMidiType;
219 e.flags = kVstMidiEventIsRealtime;
220 e.byteSize =
sizeof(VstMidiEvent);
222 e.midiData[0] = (char)(uint8_t)176 + i;
223 e.midiData[1] = (char)(uint8_t)121;
227 events->events[16 + i] =
reinterpret_cast<VstEvent*
>(&e);
230 dispatch(effProcessEvents, 0, 0, events, 0.f);
232 if constexpr(!UseDouble)
234 float* dummy = (
float*)alloca(
sizeof(
float) * m_bs);
235 std::fill_n(dummy, m_bs, 0.f);
238 = (
float**)alloca(
sizeof(
float*) * std::max(2, this->fx->fx->numInputs));
239 for(
int i = 0; i < this->fx->fx->numInputs; i++)
243 = (
float**)alloca(
sizeof(
float*) * std::max(2, this->fx->fx->numOutputs));
244 for(
int i = 0; i < this->fx->fx->numOutputs; i++)
247 fx->fx->processReplacing(fx->fx, input, output, m_bs);
251 double* dummy = (
double*)alloca(
sizeof(
double) * m_bs);
252 std::fill_n(dummy, m_bs, 0.);
255 = (
double**)alloca(
sizeof(
double*) * std::max(2, this->fx->fx->numInputs));
256 for(
int i = 0; i < this->fx->fx->numInputs; i++)
260 = (
double**)alloca(
sizeof(
double*) * std::max(2, this->fx->fx->numOutputs));
261 for(
int i = 0; i < this->fx->fx->numOutputs; i++)
264 fx->fx->processDoubleReplacing(fx->fx, input, output, m_bs);
272 template <
typename Fun>
273 void dispatchMidi(int64_t offset, Fun&& f)
276 auto& ip =
static_cast<ossia::midi_inlet*
>(m_inlets[1])->data.messages;
277 const auto n_mess = ip.size();
285 const auto sz =
sizeof(VstEvents) +
sizeof(
void*) * n_mess * 2;
286 VstEvents* events = (VstEvents*)alloca(sz);
287 std::memset(events, 0, sz);
288 events->numEvents = n_mess;
290 ossia::small_vector<VstMidiEvent, 16> vec;
293 for(libremidi::ump& mess : ip)
295 if(
auto type = mess.get_type();
296 (type == libremidi::midi2::message_type::SYSEX7)
297 || (type == libremidi::midi2::message_type::SYSEX8_MDS))
300 VstMidiEvent& e = vec[i];
301 std::memset(&e, 0,
sizeof(VstMidiEvent));
302 e.type = kVstMidiType;
303 e.byteSize =
sizeof(VstMidiEvent);
304 e.deltaFrames = mess.timestamp - offset;
305 e.flags = kVstMidiEventIsRealtime;
307 if(
auto n = cmidi2_convert_single_ump_to_midi1((uint8_t*)e.midiData, 4, mess.data);
310 events->events[i] =
reinterpret_cast<VstEvent*
>(&e);
318 dispatch(effProcessEvents, 0, 0, events, 0.f);
322 void run(
const ossia::token_request& tk, ossia::exec_state_facade st)
noexcept override
324 if(!muted() && tk.date > tk.prev_date)
326 const auto timings = st.timings(tk);
328 this->setupTimeInfo(tk, st);
330 if constexpr(UseDouble)
332 if constexpr(IsSynth)
334 dispatchMidi(timings.start_sample, [
this, timings] {
335 processDouble(timings.start_sample, timings.length);
340 processDouble(timings.start_sample, timings.length);
345 if constexpr(IsSynth)
347 dispatchMidi(timings.start_sample, [
this, timings] {
348 processFloat(timings.start_sample, timings.length);
353 processFloat(timings.start_sample, timings.length);
358 if(this->fx->fx->numOutputs == 1)
360 auto& op = m_outlets[0]->template target<ossia::audio_port>()->get();
361 op[1].assign(op[0].begin(), op[0].end());
366 void processFloat(int64_t offset, int64_t samples)
368 if constexpr(!UseDouble)
373 SCORE_ASSERT(m_bs >= offset + samples);
375 const auto max_i = std::max(2, this->fx->fx->numInputs);
376 const auto max_o = std::max(2, this->fx->fx->numOutputs);
377 const auto max_io = std::max(max_i, max_o);
379 const auto max_samples = std::max(samples, (int64_t)m_bs);
384 auto& ip = prepareInput(offset, samples);
385 SCORE_ASSERT(ip.size() >= 2);
387 auto& op = prepareOutput(offset, samples);
388 SCORE_ASSERT(op.size() >= 2);
391 float_v[0].resize(max_samples);
392 float_v[1].resize(max_samples);
394 std::copy_n(ip[0].data() + offset, samples, float_v[0].data());
395 std::copy_n(ip[1].data() + offset, samples, float_v[1].data());
397 float** io = (
float**)alloca(
sizeof(
float*) * max_io);
398 io[0] = float_v[0].data();
399 io[1] = float_v[1].data();
405 float* dummy = (
float*)alloca(
sizeof(
float) * max_samples);
406 std::fill_n(dummy, max_samples, 0.f);
408 for(
int i = 2; i < max_io; i++)
412 fx->fx->processReplacing(fx->fx, io, io, samples);
414 std::copy_n(float_v[0].data(), samples, op[0].data() + offset);
415 std::copy_n(float_v[1].data(), samples, op[1].data() + offset);
422 void processDouble(int64_t offset, int64_t samples)
424 if constexpr(UseDouble)
429 SCORE_ASSERT(m_bs >= offset + samples);
431 const auto max_i = std::max(2, this->fx->fx->numInputs);
432 const auto max_o = std::max(2, this->fx->fx->numOutputs);
433 const auto max_io = std::max(max_i, max_o);
436 auto& ip = prepareInput(offset, samples);
437 SCORE_ASSERT(ip.size() >= 2);
439 auto& op = prepareOutput(offset, samples);
440 SCORE_ASSERT(op.size() >= 2);
442 double** input = (
double**)alloca(
sizeof(
double*) * max_i);
443 input[0] = ip[0].data() + offset;
444 input[1] = ip[1].data() + offset;
446 double** output = (
double**)alloca(
sizeof(
double*) * max_o);
447 output[0] = op[0].data() + offset;
448 output[1] = op[1].data() + offset;
454 double* dummy = (
double*)alloca(
sizeof(
double) * m_bs);
455 std::fill_n(dummy, m_bs, 0.);
456 for(
int i = 2; i < this->fx->fx->numInputs; i++)
458 for(
int i = 2; i < this->fx->fx->numOutputs; i++)
462 fx->fx->processDoubleReplacing(fx->fx, input, output, samples);
469 std::conditional_t<!UseDouble, std::array<ossia::float_vector, 2>,
dummy_t> float_v;