Logging improvements to allow prettier displays in the server GUI.
[dcpomatic.git] / src / lib / transcoder.cc
1 /*
2     Copyright (C) 2012-2015 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 /** @file  src/transcoder.cc
21  *  @brief A class which takes a Film and some Options, then uses those to transcode the film.
22  *
23  *  A decoder is selected according to the content type, and the encoder can be specified
24  *  as a parameter to the constructor.
25  */
26
27 #include "transcoder.h"
28 #include "encoder.h"
29 #include "film.h"
30 #include "video_decoder.h"
31 #include "audio_decoder.h"
32 #include "player.h"
33 #include "job.h"
34 #include "writer.h"
35 #include "compose.hpp"
36 #include "subtitle_content.h"
37 #include <boost/signals2.hpp>
38 #include <boost/foreach.hpp>
39 #include <iostream>
40
41 using std::string;
42 using std::cout;
43 using std::list;
44 using boost::shared_ptr;
45 using boost::weak_ptr;
46 using boost::dynamic_pointer_cast;
47
48 /** Construct a transcoder.
49  *  @param f Film that we are transcoding.
50  *  @param j Job that this transcoder is being used in.
51  */
52 Transcoder::Transcoder (shared_ptr<const Film> film, shared_ptr<Job> j)
53         : _film (film)
54         , _player (new Player (film, film->playlist ()))
55         , _writer (new Writer (film, j))
56         , _encoder (new Encoder (film, j, _writer))
57         , _finishing (false)
58 {
59
60 }
61
62 void
63 Transcoder::go ()
64 {
65         _writer->start ();
66         _encoder->begin ();
67
68         DCPTime const frame = DCPTime::from_frames (1, _film->video_frame_rate ());
69         DCPTime const length = _film->length ();
70
71         int burnt_subtitles = 0;
72         int non_burnt_subtitles = 0;
73         BOOST_FOREACH (shared_ptr<const Content> c, _film->content ()) {
74                 shared_ptr<const SubtitleContent> sc = dynamic_pointer_cast<const SubtitleContent> (c);
75                 if (sc && sc->use_subtitles()) {
76                         if (sc->burn_subtitles()) {
77                                 ++burnt_subtitles;
78                         } else {
79                                 ++non_burnt_subtitles;
80                         }
81                 }
82         }
83
84         if (non_burnt_subtitles) {
85                 _writer->write (_player->get_subtitle_fonts ());
86         }
87
88         for (DCPTime t; t < length; t += frame) {
89                 list<shared_ptr<PlayerVideo> > v = _player->get_video (t, true);
90                 for (list<shared_ptr<PlayerVideo> >::const_iterator i = v.begin(); i != v.end(); ++i) {
91                         _encoder->enqueue (*i);
92                 }
93
94                 shared_ptr<AudioBuffers> audio = _player->get_audio (t, frame, true);
95                 if (audio) {
96                         _writer->write (audio);
97                 }
98
99                 if (non_burnt_subtitles) {
100                         _writer->write (_player->get_subtitles (t, frame, true, false));
101                 }
102         }
103
104         /* XXX: we should be passing through details of positions, at least... */
105         BOOST_FOREACH (shared_ptr<dcp::ReelAsset> i, _player->get_reel_assets ()) {
106                 _writer->write (i);
107         }
108
109         _finishing = true;
110         _encoder->end ();
111         _writer->finish ();
112 }
113
114 float
115 Transcoder::current_encoding_rate () const
116 {
117         return _encoder->current_encoding_rate ();
118 }
119
120 int
121 Transcoder::video_frames_out () const
122 {
123         return _encoder->video_frames_out ();
124 }