summaryrefslogtreecommitdiff
path: root/src/stereo_j2k_picture_asset_writer.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/stereo_j2k_picture_asset_writer.cc')
-rw-r--r--src/stereo_j2k_picture_asset_writer.cc160
1 files changed, 160 insertions, 0 deletions
diff --git a/src/stereo_j2k_picture_asset_writer.cc b/src/stereo_j2k_picture_asset_writer.cc
new file mode 100644
index 00000000..6ee271bc
--- /dev/null
+++ b/src/stereo_j2k_picture_asset_writer.cc
@@ -0,0 +1,160 @@
+/*
+ Copyright (C) 2012-2021 Carl Hetherington <cth@carlh.net>
+
+ This file is part of libdcp.
+
+ libdcp is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ libdcp is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with libdcp. If not, see <http://www.gnu.org/licenses/>.
+
+ In addition, as a special exception, the copyright holders give
+ permission to link the code of portions of this program with the
+ OpenSSL library under certain conditions as described in each
+ individual source file, and distribute linked combinations
+ including the two.
+
+ You must obey the GNU General Public License in all respects
+ for all of the code used other than OpenSSL. If you modify
+ file(s) with this exception, you may extend this exception to your
+ version of the file(s), but you are not obligated to do so. If you
+ do not wish to do so, delete this exception statement from your
+ version. If you delete this exception statement from all source
+ files in the program, then also delete it here.
+*/
+
+
+/** @file src/stereo_picture_asset_writer.cc
+ * @brief StereoPictureAssetWriter class
+ */
+
+
+#include "stereo_picture_asset_writer.h"
+#include "exceptions.h"
+#include "dcp_assert.h"
+#include "picture_asset.h"
+#include "crypto_context.h"
+#include <asdcp/AS_DCP.h>
+#include <asdcp/KM_fileio.h>
+
+
+#include "picture_asset_writer_common.cc"
+
+
+using std::string;
+using std::shared_ptr;
+using namespace dcp;
+
+
+struct StereoPictureAssetWriter::ASDCPState : public ASDCPStateBase
+{
+ ASDCP::JP2K::MXFSWriter mxf_writer;
+};
+
+
+StereoPictureAssetWriter::StereoPictureAssetWriter (PictureAsset* mxf, boost::filesystem::path file, bool overwrite)
+ : PictureAssetWriter (mxf, file, overwrite)
+ , _state (new StereoPictureAssetWriter::ASDCPState)
+{
+
+}
+
+
+StereoPictureAssetWriter::~StereoPictureAssetWriter()
+{
+ try {
+ /* Last-resort finalization to close the file, at least */
+ if (!_finalized) {
+ _state->mxf_writer.Finalize();
+ }
+ } catch (...) {}
+}
+
+
+void
+StereoPictureAssetWriter::start (uint8_t const * data, int size)
+{
+ dcp::start (this, _state, _picture_asset, data, size);
+ _picture_asset->set_frame_rate (Fraction (_picture_asset->edit_rate().numerator * 2, _picture_asset->edit_rate().denominator));
+}
+
+
+FrameInfo
+StereoPictureAssetWriter::write (uint8_t const * data, int size)
+{
+ DCP_ASSERT (!_finalized);
+
+ if (!_started) {
+ start (data, size);
+ }
+
+ if (ASDCP_FAILURE (_state->j2k_parser.OpenReadFrame (data, size, _state->frame_buffer))) {
+ boost::throw_exception (MiscError ("could not parse J2K frame"));
+ }
+
+ _state->frame_buffer.PlaintextOffset(0);
+
+ uint64_t const before_offset = _state->mxf_writer.Tell ();
+
+ string hash;
+ auto r = _state->mxf_writer.WriteFrame (
+ _state->frame_buffer,
+ _next_eye == Eye::LEFT ? ASDCP::JP2K::SP_LEFT : ASDCP::JP2K::SP_RIGHT,
+ _crypto_context->context(),
+ _crypto_context->hmac(),
+ &hash
+ );
+
+ if (ASDCP_FAILURE (r)) {
+ boost::throw_exception (MXFFileError ("error in writing video MXF", _file.string(), r));
+ }
+
+ _next_eye = _next_eye == Eye::LEFT ? Eye::RIGHT : Eye::LEFT;
+
+ if (_next_eye == Eye::LEFT) {
+ ++_frames_written;
+ }
+
+ return FrameInfo (before_offset, _state->mxf_writer.Tell() - before_offset, hash);
+}
+
+
+void
+StereoPictureAssetWriter::fake_write (int size)
+{
+ DCP_ASSERT (_started);
+ DCP_ASSERT (!_finalized);
+
+ auto r = _state->mxf_writer.FakeWriteFrame (size, _next_eye == Eye::LEFT ? ASDCP::JP2K::SP_LEFT : ASDCP::JP2K::SP_RIGHT);
+ if (ASDCP_FAILURE(r)) {
+ boost::throw_exception (MXFFileError("error in writing video MXF", _file.string(), r));
+ }
+
+ _next_eye = _next_eye == Eye::LEFT ? Eye::RIGHT : Eye::LEFT;
+ if (_next_eye == Eye::LEFT) {
+ ++_frames_written;
+ }
+}
+
+
+bool
+StereoPictureAssetWriter::finalize ()
+{
+ if (_started) {
+ auto r = _state->mxf_writer.Finalize();
+ if (ASDCP_FAILURE(r)) {
+ boost::throw_exception (MXFFileError("error in finalizing video MXF", _file.string(), r));
+ }
+ }
+
+ _picture_asset->_intrinsic_duration = _frames_written;
+ return PictureAssetWriter::finalize ();
+}