13 #include <libavformat/avformat.h>
16 #include <QApplication>
18 #include "mainwindow.h"
19 #include "ffmpeg_raii.h"
20 #include "post_to_main_thread.h"
21 #include "ui_mainwindow.h"
23 #define MAX_STREAMS 16
26 using namespace std::chrono;
28 // TODO: Replace by some sort of GUI control, I guess.
29 int64_t current_pts = 0;
31 string filename_for_frame(unsigned stream_idx, int64_t pts)
34 snprintf(filename, sizeof(filename), "frames/cam%d-pts%09ld.jpeg", stream_idx, pts);
39 vector<int64_t> frames[MAX_STREAMS];
43 int main(int argc, char **argv)
46 avformat_network_init();
48 QApplication app(argc, argv);
49 MainWindow mainWindow;
52 thread(thread_func).detach();
59 auto format_ctx = avformat_open_input_unique("multiangle.mp4", nullptr, nullptr);
60 if (format_ctx == nullptr) {
61 fprintf(stderr, "%s: Error opening file\n", "example.mp4");
65 int64_t last_pts = -1;
69 unique_ptr<AVPacket, decltype(av_packet_unref)*> pkt_cleanup(
70 &pkt, av_packet_unref);
74 if (av_read_frame(format_ctx.get(), &pkt) != 0) {
77 fprintf(stderr, "Got a frame from camera %d, pts = %ld, size = %d\n",
78 pkt.stream_index, pkt.pts, pkt.size);
79 string filename = filename_for_frame(pkt.stream_index, pkt.pts);
80 FILE *fp = fopen(filename.c_str(), "wb");
82 perror(filename.c_str());
85 fwrite(pkt.data, pkt.size, 1, fp);
88 post_to_main_thread([pkt] {
89 if (pkt.stream_index == 0) {
90 global_mainwindow->ui->input1_display->setFrame(pkt.stream_index, pkt.pts);
91 } else if (pkt.stream_index == 1) {
92 global_mainwindow->ui->input2_display->setFrame(pkt.stream_index, pkt.pts);
93 } else if (pkt.stream_index == 2) {
94 global_mainwindow->ui->input3_display->setFrame(pkt.stream_index, pkt.pts);
98 assert(pkt.stream_index < MAX_STREAMS);
99 frames[pkt.stream_index].push_back(pkt.pts);
101 // Hack. Assumes a given timebase.
102 if (last_pts != -1) {
103 this_thread::sleep_for(microseconds((pkt.pts - last_pts) * 1000000 / 12800));
106 current_pts = pkt.pts;