2#include <ossia/detail/config.hpp>
4#include <ossia/audio/audio_parameter.hpp>
5#include <ossia/dataflow/audio_stretch_mode.hpp>
6#include <ossia/dataflow/graph_node.hpp>
7#include <ossia/dataflow/nodes/media.hpp>
8#include <ossia/dataflow/nodes/sound.hpp>
9#include <ossia/dataflow/port.hpp>
10#include <ossia/detail/libav.hpp>
11#include <ossia/detail/pod_vector.hpp>
17#include <libavcodec/avcodec.h>
18#include <libavformat/avformat.h>
19#include <libavutil/frame.h>
20#include <libavutil/mem.h>
21#include <libswresample/swresample.h>
26class sound_libav final :
public ossia::sound_node
33 : packet{av_packet_alloc()}
34 , frame{av_frame_alloc()}
36 m_outlets.push_back(&audio_out);
43 av_frame_free(&frame);
44 av_packet_free(&packet);
47 std::string label() const noexcept
override {
return "sound_libav"; }
49 void set_start(std::size_t v) { start = v; }
51 void set_upmix(std::size_t v) { upmix = v; }
53 void set_sound(libav_handle hdl)
57 m_handle = std::move(hdl);
60 m_channel_q = boost::circular_buffer<float>(8192 * m_handle.channels());
63 void transport(time_value flicks)
override
67 m_handle.format, m_handle.codec, m_handle.stream, flicks.impl, AVSEEK_FLAG_ANY);
70 void fetch_from_libav(
int samples_to_write)
72 const std::size_t channels = this->channels();
76 auto floats_to_write = channels * samples_to_write;
77 while(m_channel_q.size() < floats_to_write)
80 if(m_channel_q.capacity() < 4 * floats_to_write)
82 m_channel_q.set_capacity(4 * floats_to_write);
86 auto fmt_ctx = m_handle.format;
87 auto codec_ctx = m_handle.codec;
88 auto stream = m_handle.stream;
92 av_packet_unref(packet);
93 ret = av_read_frame(fmt_ctx, packet);
95 while(ret >= 0 && ret != AVERROR(EOF) && packet->stream_index != stream->index)
97 av_packet_unref(packet);
98 ret = av_read_frame(fmt_ctx, packet);
100 if(ret == AVERROR(EOF))
110 ret = avcodec_send_packet(codec_ctx, packet);
113 ret = avcodec_receive_frame(codec_ctx, frame);
116 const int samples = frame->nb_samples;
117 m_tmp.resize(samples * channels, boost::container::default_init);
118 float* out_ptr = m_tmp.data();
119 const int read_samples = swr_convert(
120 m_handle.resample, (uint8_t**)&out_ptr, samples,
121 (
const uint8_t**)frame->extended_data, samples);
124 m_channel_q.end(), out_ptr, out_ptr + read_samples * channels);
131 template <
typename T>
133 fetch_audio(int64_t start, int64_t samples_to_write, T** audio_array_base)
noexcept
135 const std::size_t channels = this->channels();
139 fetch_from_libav(samples_to_write);
142 for(
int k = 0; k < samples_to_write; k++)
144 for(std::size_t chan = 0; chan < channels; chan++)
146 if(m_channel_q.size() > 0)
148 audio_array_base[chan][k] = m_channel_q.front();
149 m_channel_q.pop_front();
153 audio_array_base[chan][k] = 0.;
159 template <
typename T>
160 void fetch_audio_backward(
161 int64_t start, int64_t samples_to_write, T** audio_array_base)
noexcept
163 const std::size_t channels = this->channels();
174 int64_t backward_start = start - samples_to_write + 1;
175 if(backward_start < 0)
180 const int64_t sample_rate = m_handle.stream->codecpar->sample_rate;
184 constexpr int64_t flicks_per_second = 705600000LL;
185 int64_t flicks_pos = backward_start * flicks_per_second / sample_rate;
188 ossia::seek_to_flick(
189 m_handle.format, m_handle.codec, m_handle.stream, flicks_pos, AVSEEK_FLAG_BACKWARD);
193 fetch_from_libav(samples_to_write);
196 for(int64_t k = 0; k < samples_to_write; k++)
198 for(std::size_t chan = 0; chan < channels; chan++)
200 if(m_channel_q.size() > 0)
202 audio_array_base[chan][k] = m_channel_q.front();
203 m_channel_q.pop_front();
207 audio_array_base[chan][k] = 0.;
213 for(std::size_t chan = 0; chan < channels; chan++)
215 std::reverse(audio_array_base[chan], audio_array_base[chan] + samples_to_write);
219 void run(
const ossia::token_request& t, ossia::exec_state_facade e)
noexcept override
224 const auto channels = m_handle.channels();
225 const auto len = m_handle.totalPCMFrameCount();
227 ossia::audio_port& ap = *audio_out;
228 ap.set_channels(std::max((std::size_t)upmix, (std::size_t)channels));
230 const auto [samples_to_read, samples_to_write]
231 = snd::sample_info(e.bufferSize(), e.modelToSamples(), t);
232 if(samples_to_write <= 0)
235 assert(samples_to_write > 0);
237 const auto samples_offset = t.physical_start(e.modelToSamples());
242 if(t.prev_date < m_prev_date)
247 if(t.prev_date != 0_tv)
249 transport(t.prev_date);
259 transport(t.prev_date);
266 if(t.prev_date > m_prev_date)
271 transport(t.prev_date);
275 transport(t.prev_date);
280 for(
int chan = 0; chan < channels; chan++)
282 ap.channel(chan).resize(e.bufferSize());
285 const double stretch_ratio = update_stretch(t, e);
286 const double abs_stretch_ratio = std::abs(stretch_ratio);
290 *
this, t, e, stretch_ratio, channels, len, samples_to_read, samples_to_write,
293 const bool start_discontinuous = t.start_discontinuous || (m_last_stretch > 70.);
294 const bool end_discontinuous = t.end_discontinuous || (abs_stretch_ratio > 70.);
295 if(abs_stretch_ratio > 70. && m_last_stretch > 70.)
298 for(std::size_t i = 0; i < channels; i++)
300 ossia::snd::do_zero(ap.channel(i), samples_offset, samples_to_write);
306 for(
int chan = 0; chan < channels; chan++)
310 start_discontinuous, end_discontinuous, ap.channel(chan), samples_offset,
315 ossia::snd::perform_upmix(this->upmix, channels, ap);
316 ossia::snd::perform_start_offset(this->start, ap);
318 m_prev_date = t.date;
319 m_last_stretch = abs_stretch_ratio;
322 [[nodiscard]] std::size_t channels()
const
324 return m_handle ? m_handle.channels() : 0;
326 [[nodiscard]] std::size_t duration()
const
328 return m_handle ? m_handle.totalPCMFrameCount() : 0;
332 libav_handle m_handle{};
334 ossia::audio_outlet audio_out;
339 ossia::pod_vector<float> m_tmp{};
340 boost::circular_buffer<float> m_channel_q;
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30