2#include <ossia/detail/config.hpp>
4#if __has_include(<libavcodec/avcodec.h>) && \
5 __has_include(<libavformat/avformat.h>) && \
6 __has_include(<libavdevice/avdevice.h>) && \
7 __has_include(<libavutil/frame.h>) && \
8 __has_include(<libswresample/swresample.h>) && \
9 __has_include(<libswscale/swscale.h>)
11#define OSSIA_HAS_LIBAV 1
12#include <ossia/detail/flicks.hpp>
15#include <libavcodec/avcodec.h>
16#include <libavformat/avformat.h>
17#include <libavutil/channel_layout.h>
18#include <libavutil/version.h>
19#include <libswresample/swresample.h>
25static constexpr const int OSSIA_LIBAV_SEEK_ROUGH = 0b00000100000000000000000000000000;
26inline bool seek_to_flick(
27 AVFormatContext* format, AVCodecContext* codec, AVStream* stream, int64_t flicks,
30 constexpr auto flicks_tb = AVRational{1, ossia::flicks_per_second<int64_t>};
31 constexpr auto av_tb = AVRational{1, AV_TIME_BASE};
33 const auto dts = av_rescale_q_rnd(flicks, flicks_tb, av_tb, AVRounding::AV_ROUND_DOWN);
35 avio_flush(format->pb);
36 avformat_flush(format);
38 avcodec_flush_buffers(codec);
40 if(flags & OSSIA_LIBAV_SEEK_ROUGH)
43 flags &= ~OSSIA_LIBAV_SEEK_ROUGH;
44 if(av_seek_frame(format, -1, dts, flags) < 0)
50 if(avformat_seek_file(format, -1, INT64_MIN, dts, INT64_MAX, flags) < 0)
54 avio_flush(format->pb);
55 avformat_flush(format);
57 avcodec_flush_buffers(codec);
62static inline int avstream_get_audio_channels(AVStream& stream)
noexcept
64#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(57, 24, 100)
65 return stream.codecpar->ch_layout.nb_channels;
67 return stream.codecpar->channels;
73 AVFormatContext* format{};
75 AVCodecContext* codec{};
76 SwrContext* resample{};
78 libav_handle() =
default;
79 libav_handle(
const libav_handle& other) =
delete;
80 libav_handle(libav_handle&& other)
82 format = other.format;
83 other.format =
nullptr;
84 stream = other.stream;
85 other.stream =
nullptr;
87 other.codec =
nullptr;
88 resample = other.resample;
89 other.resample =
nullptr;
91 libav_handle& operator=(
const libav_handle& other) =
delete;
92 libav_handle& operator=(libav_handle&& other)
94 format = other.format;
95 other.format =
nullptr;
96 stream = other.stream;
97 other.stream =
nullptr;
99 other.codec =
nullptr;
100 resample = other.resample;
101 other.resample =
nullptr;
117 avcodec_free_context(&codec);
121 avformat_close_input(&format);
122 avformat_free_context(format);
127 void open(
const std::string& path,
int stream_index,
int target_rate)
noexcept
129 if(avformat_open_input(&format, path.c_str(),
nullptr,
nullptr) != 0)
134 if(avformat_find_stream_info(format,
nullptr) < 0)
139 if(stream_index < 0 || stream_index >=
int(format->nb_streams))
144 if(format->streams[stream_index]->codecpar->codec_type != AVMEDIA_TYPE_AUDIO)
150 stream = format->streams[stream_index];
152 auto cdc = avcodec_find_decoder(stream->codecpar->codec_id);
159 codec = avcodec_alloc_context3(cdc);
160 if(avcodec_parameters_to_context(codec, stream->codecpar) < 0)
166 if(avcodec_open2(codec, cdc,
nullptr) < 0)
172 AVSampleFormat out_sample_fmt{AV_SAMPLE_FMT_FLT};
173 AVSampleFormat in_sample_fmt{(AVSampleFormat)stream->codecpar->format};
174 int in_sample_rate = stream->codecpar->sample_rate;
175 int out_sample_rate = target_rate == 0 ? in_sample_rate : target_rate;
177#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(57, 24, 100)
178 AVChannelLayout out_layout{stream->codecpar->ch_layout};
179 AVChannelLayout in_layout{stream->codecpar->ch_layout};
182 &resample, &out_layout, out_sample_fmt, out_sample_rate, &in_layout,
183 in_sample_fmt, in_sample_rate, 0,
nullptr);
185 int64_t out_layout = stream->codecpar->channel_layout;
186 int64_t in_layout = stream->codecpar->channel_layout;
188 resample = swr_alloc_set_opts(
189 resample, out_layout, out_sample_fmt, out_sample_rate, in_layout, in_sample_fmt,
190 in_sample_rate, 0,
nullptr);
196 int rate() const noexcept {
return stream->codecpar->sample_rate; }
197 int channels() const noexcept {
return avstream_get_audio_channels(*stream); }
199 int64_t totalPCMFrameCount() const noexcept
201 if(stream->duration > 0)
203 return stream->duration;
205 else if(format->duration > 0)
207 double seconds = format->duration / double(AV_TIME_BASE);
208 double frames = seconds * stream->codecpar->sample_rate;
211 else if(stream->nb_frames > 0)
214 stream->nb_frames, stream->r_frame_rate,
215 AVRational{1, stream->codecpar->sample_rate});
223 operator bool() const noexcept {
return bool(format); }
225 void fetch(int64_t frame,
int samples_to_write,
auto func)
228 ossia::seek_to_flick(
229 format, codec, stream,
230 ossia::flicks_per_second<double> * frame / stream->codecpar->sample_rate,
233 const std::size_t channels = this->channels();
234 std::vector<float> tmp;
239 while(processed < samples_to_write)
243 auto packet = av_packet_alloc();
246 ret = av_read_frame(format, packet);
248 while(ret >= 0 && ret != AVERROR(EOF) && packet->stream_index != stream->index)
250 av_packet_unref(packet);
251 ret = av_read_frame(format, packet);
253 if(ret == AVERROR(EOF))
263 ret = avcodec_send_packet(codec, packet);
266 auto avframe = av_frame_alloc();
267 ret = avcodec_receive_frame(codec, avframe);
270 const int av_frame_start = avframe->best_effort_timestamp;
271 const int samples = avframe->nb_samples;
275 const int offset = (frame < av_frame_start) ? 0 : (frame - av_frame_start);
276 if(offset >= samples)
279 av_frame_free(&avframe);
280 av_packet_unref(packet);
281 av_packet_free(&packet);
288 tmp.resize(samples * channels);
289 float* out_ptr = tmp.data();
292 int read_samples = swr_convert(
293 resample, (uint8_t**)&out_ptr, samples,
294 (
const uint8_t**)avframe->extended_data, samples);
296 auto end = tmp.data() + tmp.size();
298 read_samples -= offset;
299 if(read_samples <= 0)
301 av_frame_free(&avframe);
302 av_packet_unref(packet);
303 av_packet_free(&packet);
307 out_ptr += offset * channels;
309 for(
int i = 0; i < read_samples; i++)
314 if(processed == samples_to_write)
316 av_frame_free(&avframe);
317 av_packet_unref(packet);
318 av_packet_free(&packet);
327 frame = avframe->best_effort_timestamp + avframe->nb_samples;
329 av_frame_free(&avframe);
331 av_packet_unref(packet);
332 av_packet_free(&packet);