Basics of FFmpeg examiner works.
[dcpomatic.git] / src / lib / ffmpeg.cc
1 /*
2     Copyright (C) 2013 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 extern "C" {
21 #include <libavcodec/avcodec.h>
22 #include <libavformat/avformat.h>
23 #include <libswscale/swscale.h>
24 #include <libpostproc/postprocess.h>
25 }
26 #include "ffmpeg.h"
27 #include "ffmpeg_content.h"
28 #include "exceptions.h"
29
30 #include "i18n.h"
31
32 using std::string;
33 using std::stringstream;
34 using boost::shared_ptr;
35
36 boost::mutex FFmpeg::_mutex;
37
38 FFmpeg::FFmpeg (boost::shared_ptr<const FFmpegContent> c)
39         : _ffmpeg_content (c)
40         , _format_context (0)
41         , _frame (0)
42         , _video_stream (-1)
43 {
44         setup_general ();
45         setup_video ();
46         setup_audio ();
47 }
48
49 FFmpeg::~FFmpeg ()
50 {
51         boost::mutex::scoped_lock lm (_mutex);
52
53         for (uint32_t i = 0; i < _format_context->nb_streams; ++i) {
54                 AVCodecContext* context = _format_context->streams[i]->codec;
55                 if (context->codec_type == AVMEDIA_TYPE_VIDEO || context->codec_type == AVMEDIA_TYPE_AUDIO) {
56                         avcodec_close (context);
57                 }
58         }
59
60         av_free (_frame);
61         
62         avformat_close_input (&_format_context);
63 }
64
65 void
66 FFmpeg::setup_general ()
67 {
68         av_register_all ();
69
70         if (avformat_open_input (&_format_context, _ffmpeg_content->file().string().c_str(), 0, 0) < 0) {
71                 throw OpenFileError (_ffmpeg_content->file().string ());
72         }
73
74         if (avformat_find_stream_info (_format_context, 0) < 0) {
75                 throw DecodeError (_("could not find stream information"));
76         }
77
78         /* Find video stream */
79
80         for (uint32_t i = 0; i < _format_context->nb_streams; ++i) {
81                 AVStream* s = _format_context->streams[i];
82                 if (s->codec->codec_type == AVMEDIA_TYPE_VIDEO) {
83                         _video_stream = i;
84                 }
85         }
86
87         if (_video_stream < 0) {
88                 throw DecodeError (N_("could not find video stream"));
89         }
90
91         _frame = avcodec_alloc_frame ();
92         if (_frame == 0) {
93                 throw DecodeError (N_("could not allocate frame"));
94         }
95 }
96
97 void
98 FFmpeg::setup_video ()
99 {
100         boost::mutex::scoped_lock lm (_mutex);
101         
102         AVCodecContext* context = _format_context->streams[_video_stream]->codec;
103         AVCodec* codec = avcodec_find_decoder (context->codec_id);
104
105         if (codec == 0) {
106                 throw DecodeError (_("could not find video decoder"));
107         }
108
109         if (avcodec_open2 (context, codec, 0) < 0) {
110                 throw DecodeError (N_("could not open video decoder"));
111         }
112 }
113
114 void
115 FFmpeg::setup_audio ()
116 {
117         boost::mutex::scoped_lock lm (_mutex);
118
119         for (uint32_t i = 0; i < _format_context->nb_streams; ++i) {
120                 AVCodecContext* context = _format_context->streams[i]->codec;
121                 if (context->codec_type != AVMEDIA_TYPE_AUDIO) {
122                         continue;
123                 }
124                 
125                 AVCodec* codec = avcodec_find_decoder (context->codec_id);
126                 if (codec == 0) {
127                         throw DecodeError (_("could not find audio decoder"));
128                 }
129                 
130                 if (avcodec_open2 (context, codec, 0) < 0) {
131                         throw DecodeError (N_("could not open audio decoder"));
132                 }
133         }
134 }
135
136
137 AVCodecContext *
138 FFmpeg::video_codec_context () const
139 {
140         return _format_context->streams[_video_stream]->codec;
141 }
142
143 AVCodecContext *
144 FFmpeg::audio_codec_context () const
145 {
146         return _format_context->streams[_ffmpeg_content->audio_stream()->id]->codec;
147 }