#include "frame_on_disk.h"
#include "player.h"
#include "shared/ffmpeg_raii.h"
+#include "shared/shared_defs.h"
#include "shared/timebase.h"
#include <QMessageBox>
extern "C" {
#include <libavformat/avformat.h>
+#include <libavutil/channel_layout.h>
}
using namespace std;
namespace {
// Only used in export_cliplist_clip_multitrack_triggered.
-struct BufferedJPEG {
+struct BufferedFrame {
int64_t pts;
- unsigned stream_idx;
- string jpeg;
+ unsigned video_stream_idx;
+ string data;
};
-bool write_buffered_jpegs(AVFormatContext *avctx, const vector<BufferedJPEG> &buffered_jpegs)
+bool write_buffered_frames(AVFormatContext *avctx, const vector<BufferedFrame> &buffered_frames)
{
- for (const BufferedJPEG &jpeg : buffered_jpegs) {
+ for (const BufferedFrame &frame : buffered_frames) {
AVPacket pkt;
av_init_packet(&pkt);
- pkt.stream_index = jpeg.stream_idx;
- pkt.data = (uint8_t *)jpeg.jpeg.data();
- pkt.size = jpeg.jpeg.size();
- pkt.pts = jpeg.pts;
- pkt.dts = jpeg.pts;
+ pkt.stream_index = frame.video_stream_idx;
+ pkt.data = (uint8_t *)frame.data.data();
+ pkt.size = frame.data.size();
+ pkt.pts = frame.pts;
+ pkt.dts = frame.pts;
pkt.flags = AV_PKT_FLAG_KEY;
if (av_write_frame(avctx, &pkt) < 0) {
AVStream *avstream_video = avformat_new_stream(avctx, nullptr);
if (avstream_video == nullptr) {
fprintf(stderr, "avformat_new_stream() failed\n");
- exit(1);
+ abort();
}
avstream_video->time_base = AVRational{ 1, TIMEBASE };
avstream_video->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
video_streams.push_back(avstream_video);
}
+ // Similar, for audio streams.
+ vector<AVStream *> audio_streams;
+ for (unsigned stream_idx = 0; stream_idx <= last_stream_idx; ++stream_idx) {
+ AVStream *avstream_audio = avformat_new_stream(avctx, nullptr);
+ if (avstream_audio == nullptr) {
+ fprintf(stderr, "avformat_new_stream() failed\n");
+ abort();
+ }
+ avstream_audio->time_base = AVRational{ 1, TIMEBASE };
+ avstream_audio->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
+ avstream_audio->codecpar->codec_id = AV_CODEC_ID_PCM_S32LE;
+ avstream_audio->codecpar->channel_layout = AV_CH_LAYOUT_STEREO;
+ avstream_audio->codecpar->channels = 2;
+ avstream_audio->codecpar->sample_rate = OUTPUT_FREQUENCY;
+ audio_streams.push_back(avstream_audio);
+ }
+
if (avformat_write_header(avctx, nullptr) < 0) {
QMessageBox msgbox;
msgbox.setText("Writing header failed");
// We buffer up to 1000 frames at a time, in a hope that we can reduce
// the amount of seeking needed on rotational media.
- vector<BufferedJPEG> buffered_jpegs;
+ vector<BufferedFrame> buffered_frames;
size_t frames_written = 0;
while (num_streams_with_frames_left > 0) {
// Find the stream with the lowest frame. Lower stream indexes win.
--num_streams_with_frames_left;
}
}
- string jpeg = readers[first_frame_stream_idx].read_frame(first_frame);
- int64_t scaled_pts = av_rescale_q(first_frame.pts, AVRational{ 1, TIMEBASE },
- video_streams[first_frame_stream_idx]->time_base);
- buffered_jpegs.emplace_back(BufferedJPEG{ scaled_pts, first_frame_stream_idx, std::move(jpeg) });
- if (buffered_jpegs.size() >= 1000) {
- if (!write_buffered_jpegs(avctx, buffered_jpegs)) {
+
+ FrameReader::Frame frame = readers[first_frame_stream_idx].read_frame(first_frame, /*read_video=*/true, /*read_audio=*/true);
+
+ // Write audio. (Before video, since that's what we expect on input.)
+ if (!frame.audio.empty()) {
+ unsigned audio_stream_idx = first_frame_stream_idx + video_streams.size();
+ int64_t scaled_audio_pts = av_rescale_q(first_frame.pts, AVRational{ 1, TIMEBASE },
+ audio_streams[first_frame_stream_idx]->time_base);
+ buffered_frames.emplace_back(BufferedFrame{ scaled_audio_pts, audio_stream_idx, std::move(frame.audio) });
+ }
+
+ // Write video.
+ unsigned video_stream_idx = first_frame_stream_idx;
+ int64_t scaled_video_pts = av_rescale_q(first_frame.pts, AVRational{ 1, TIMEBASE },
+ video_streams[first_frame_stream_idx]->time_base);
+ buffered_frames.emplace_back(BufferedFrame{ scaled_video_pts, video_stream_idx, std::move(frame.video) });
+
+ // Flush to disk if required.
+ if (buffered_frames.size() >= 1000) {
+ if (!write_buffered_frames(avctx, buffered_frames)) {
QMessageBox msgbox;
msgbox.setText("Writing frames failed");
msgbox.exec();
unlink(filename.c_str());
return;
}
- frames_written += buffered_jpegs.size();
+ frames_written += buffered_frames.size();
progress.setValue(frames_written);
- buffered_jpegs.clear();
+ buffered_frames.clear();
}
if (progress.wasCanceled()) {
unlink(filename.c_str());
}
}
- if (!write_buffered_jpegs(avctx, buffered_jpegs)) {
+ if (!write_buffered_frames(avctx, buffered_frames)) {
QMessageBox msgbox;
msgbox.setText("Writing frames failed");
msgbox.exec();
unlink(filename.c_str());
return;
}
- frames_written += buffered_jpegs.size();
+ frames_written += buffered_frames.size();
progress.setValue(frames_written);
}
for (const Clip &clip : clips) {
clips_with_id.emplace_back(ClipWithID{ clip, 0 });
}
- double total_length = compute_total_time(clips_with_id);
+ TimeRemaining total_length = compute_total_time(clips_with_id);
promise<void> done_promise;
future<void> done = done_promise.get_future();
player.set_done_callback([&done_promise] {
done_promise.set_value();
});
- player.set_progress_callback([¤t_value, &clips, total_length](const std::map<uint64_t, double> &player_progress, double time_remaining) {
- current_value = 1.0 - time_remaining / total_length;
+ player.set_progress_callback([¤t_value, total_length](const std::map<uint64_t, double> &player_progress, TimeRemaining time_remaining) {
+ current_value = 1.0 - time_remaining.t / total_length.t; // Nothing to do about the infinite clips.
});
player.play(clips_with_id);
while (done.wait_for(std::chrono::milliseconds(100)) != future_status::ready && !progress.wasCanceled()) {