2 Copyright (C) 2012 Carl Hetherington <cth@carlh.net>
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.
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.
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.
20 /** @file src/make_dcp_job.cc
21 * @brief A job to create DCPs.
25 #include <boost/filesystem.hpp>
26 #include <libdcp/dcp.h>
27 #include <libdcp/picture_asset.h>
28 #include <libdcp/sound_asset.h>
29 #include <libdcp/reel.h>
31 #include <libavutil/pixdesc.h>
33 #include "make_dcp_job.h"
34 #include "dcp_content_type.h"
35 #include "exceptions.h"
37 #include "imagemagick_decoder.h"
43 using boost::shared_ptr;
45 /** @param f Film we are making the DCP for.
48 MakeDCPJob::MakeDCPJob (shared_ptr<Film> f, shared_ptr<Job> req)
55 MakeDCPJob::name () const
57 return String::compose ("Make DCP for %1", _film->name());
60 /** @param f DCP frame index */
62 MakeDCPJob::j2c_path (int f, int offset) const
64 return _film->frame_out_path (f, false);
68 MakeDCPJob::wav_path (libdcp::Channel c) const
70 return _film->multichannel_audio_out_path (int (c), false);
76 if (!_film->dcp_intrinsic_duration()) {
77 throw EncodeError ("cannot make a DCP when its intrinsic duration is not known");
82 string const dcp_path = _film->dir (_film->dcp_name());
84 /* Remove any old DCP */
85 boost::filesystem::remove_all (dcp_path);
87 int const frames = _film->dcp_intrinsic_duration().get();
88 int const duration = frames - _film->trim_start() - _film->trim_end();
89 DCPFrameRate const dfr (_film->frames_per_second ());
91 libdcp::DCP dcp (_film->dir (_film->dcp_name()));
92 dcp.Progress.connect (boost::bind (&MakeDCPJob::dcp_progress, this, _1));
94 shared_ptr<libdcp::CPL> cpl (
95 new libdcp::CPL (_film->dir (_film->dcp_name()), _film->dcp_name(), _film->dcp_content_type()->libdcp_kind (), frames, dfr.frames_per_second)
100 int frames_per_reel = 0;
101 if (_film->reel_size()) {
102 frames_per_reel = (_film->reel_size().get() / (_film->j2k_bandwidth() / 8)) * dfr.frames_per_second;
104 frames_per_reel = frames;
110 while (frames_done < frames) {
112 descend (float (frames_per_reel) / frames);
114 int this_time = std::min (frames_per_reel, (frames - frames_done));
118 shared_ptr<libdcp::MonoPictureAsset> pa (
119 new libdcp::MonoPictureAsset (
120 boost::bind (&MakeDCPJob::j2c_path, this, _1, frames_done),
121 _film->dir (_film->dcp_name()),
122 String::compose ("video_%1.mxf", reel),
124 dfr.frames_per_second,
126 _film->format()->dcp_size()
130 pa->set_entry_point (_film->trim_start ());
131 pa->set_duration (duration);
135 shared_ptr<libdcp::SoundAsset> sa;
137 if (_film->audio_channels() > 0) {
140 new libdcp::SoundAsset (
141 boost::bind (&MakeDCPJob::wav_path, this, _1),
142 _film->dir (_film->dcp_name()),
143 String::compose ("audio_%1.mxf", reel),
145 dfr.frames_per_second,
148 dcp_audio_channels (_film->audio_channels())
152 sa->set_entry_point (_film->trim_start ());
153 sa->set_duration (duration);
159 cpl->add_reel (shared_ptr<libdcp::Reel> (new libdcp::Reel (pa, sa, shared_ptr<libdcp::SubtitleAsset> ())));
162 frames_done += frames_per_reel;
175 set_state (FINISHED_OK);
179 MakeDCPJob::dcp_progress (float p)