X-Git-Url: https://git.carlh.net/gitweb/?a=blobdiff_plain;ds=sidebyside;f=test%2Ftest.cc;h=76a6c4c572034ea8e4e5881741efc5b85acfe286;hb=d39880eef211a296fa8ef4712cdef5945d08527c;hp=0f60e0f1a36219fee8ca88d143901845d9bfc1f7;hpb=8af7b48d8831cf348163a2f61c14b059cd67a8fd;p=libdcp.git diff --git a/test/test.cc b/test/test.cc index 0f60e0f1..76a6c4c5 100644 --- a/test/test.cc +++ b/test/test.cc @@ -1,50 +1,439 @@ /* - Copyright (C) 2012 Carl Hetherington + Copyright (C) 2012-2020 Carl Hetherington - This program is free software; you can redistribute it and/or modify + 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. - This program is distributed in the hope that it will be useful, + 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 this program; if not, write to the Free Software - Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + along with libdcp. If not, see . + + 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. */ #define BOOST_TEST_DYN_LINK #define BOOST_TEST_MODULE libdcp_test -#include +#include "compose.hpp" +#include "cpl.h" +#include "dcp.h" +#include "file.h" +#include "interop_subtitle_asset.h" +#include "mono_picture_asset.h" +#include "picture_asset_writer.h" +#include "reel.h" +#include "reel_mono_picture_asset.h" +#include "reel_sound_asset.h" +#include "reel_closed_caption_asset.h" +#include "reel_subtitle_asset.h" +#include "sound_asset.h" +#include "sound_asset_writer.h" +#include "smpte_subtitle_asset.h" +#include "mono_picture_asset.h" +#include "openjpeg_image.h" +#include "j2k.h" +#include "picture_asset_writer.h" +#include "reel_mono_picture_asset.h" +#include "reel_asset.h" +#include "test.h" #include "util.h" +#include +#include +#include +#include +#include +#include +#include using std::string; +using std::min; +using std::list; +using std::vector; +using std::shared_ptr; +using boost::optional; + + +boost::filesystem::path private_test; +boost::filesystem::path xsd_test = "build/test/xsd with spaces"; + struct TestConfig { TestConfig() { - libdcp::init (); + dcp::init (); + if (boost::unit_test::framework::master_test_suite().argc >= 2) { + private_test = boost::unit_test::framework::master_test_suite().argv[1]; + } + + using namespace boost::filesystem; + boost::system::error_code ec; + remove_all (xsd_test, ec); + boost::filesystem::create_directory (xsd_test); + for (directory_iterator i = directory_iterator("xsd"); i != directory_iterator(); ++i) { + copy_file (*i, xsd_test / i->path().filename()); + } } }; -BOOST_GLOBAL_FIXTURE (TestConfig); +void +check_xml (xmlpp::Element* ref, xmlpp::Element* test, list ignore_tags, bool ignore_whitespace) +{ + BOOST_CHECK_EQUAL (ref->get_name (), test->get_name ()); + BOOST_CHECK_EQUAL (ref->get_namespace_prefix (), test->get_namespace_prefix ()); + + if (find(ignore_tags.begin(), ignore_tags.end(), ref->get_name()) != ignore_tags.end()) { + return; + } + + xmlpp::Element::NodeList ref_children = ref->get_children (); + xmlpp::Element::NodeList test_children = test->get_children (); + BOOST_REQUIRE_MESSAGE ( + ref_children.size () == test_children.size (), + "child counts of " << ref->get_name() << " differ; ref has " << ref_children.size() << ", test has " << test_children.size() + ); + + xmlpp::Element::NodeList::iterator k = ref_children.begin (); + xmlpp::Element::NodeList::iterator l = test_children.begin (); + while (k != ref_children.end ()) { -boost::filesystem::path -j2c (int) + /* XXX: should be doing xmlpp::EntityReference, xmlpp::XIncludeEnd, xmlpp::XIncludeStart */ + + xmlpp::Element* ref_el = dynamic_cast (*k); + xmlpp::Element* test_el = dynamic_cast (*l); + BOOST_CHECK ((ref_el && test_el) || (!ref_el && !test_el)); + if (ref_el && test_el) { + check_xml (ref_el, test_el, ignore_tags, ignore_whitespace); + } + + xmlpp::ContentNode* ref_cn = dynamic_cast (*k); + xmlpp::ContentNode* test_cn = dynamic_cast (*l); + BOOST_CHECK ((ref_cn && test_cn) || (!ref_cn && !test_cn)); + if (ref_cn && test_cn) { + if ( + !ignore_whitespace || + ref_cn->get_content().find_first_not_of(" \t\r\n") != string::npos || + test_cn->get_content().find_first_not_of(" \t\r\n") != string::npos) { + + BOOST_CHECK_EQUAL (ref_cn->get_content(), test_cn->get_content ()); + } + } + + ++k; + ++l; + } + + xmlpp::Element::AttributeList ref_attributes = ref->get_attributes (); + xmlpp::Element::AttributeList test_attributes = test->get_attributes (); + BOOST_CHECK_EQUAL (ref_attributes.size(), test_attributes.size ()); + + xmlpp::Element::AttributeList::const_iterator m = ref_attributes.begin(); + xmlpp::Element::AttributeList::const_iterator n = test_attributes.begin(); + while (m != ref_attributes.end ()) { + BOOST_CHECK_EQUAL ((*m)->get_name(), (*n)->get_name()); + BOOST_CHECK_EQUAL ((*m)->get_value(), (*n)->get_value()); + + ++m; + ++n; + } +} + +void +check_xml (string ref, string test, list ignore) { - return "test/data/32x32_red_square.j2c"; + xmlpp::DomParser* ref_parser = new xmlpp::DomParser (); + ref_parser->parse_memory (ref); + xmlpp::Element* ref_root = ref_parser->get_document()->get_root_node (); + xmlpp::DomParser* test_parser = new xmlpp::DomParser (); + test_parser->parse_memory (test); + xmlpp::Element* test_root = test_parser->get_document()->get_root_node (); + + check_xml (ref_root, test_root, ignore); } -boost::filesystem::path -wav (libdcp::Channel) +void +check_file (boost::filesystem::path ref, boost::filesystem::path check) { - return "test/data/1s_24-bit_48k_silence.wav"; + uintmax_t N = boost::filesystem::file_size (ref); + BOOST_CHECK_EQUAL (N, boost::filesystem::file_size (check)); + FILE* ref_file = dcp::fopen_boost (ref, "rb"); + BOOST_REQUIRE (ref_file); + FILE* check_file = dcp::fopen_boost (check, "rb"); + BOOST_REQUIRE (check_file); + + int const buffer_size = 65536; + uint8_t* ref_buffer = new uint8_t[buffer_size]; + uint8_t* check_buffer = new uint8_t[buffer_size]; + + string error; + error = "File " + check.string() + " differs from reference " + ref.string(); + + while (N) { + uintmax_t this_time = min (uintmax_t (buffer_size), N); + size_t r = fread (ref_buffer, 1, this_time, ref_file); + BOOST_CHECK_EQUAL (r, this_time); + r = fread (check_buffer, 1, this_time, check_file); + BOOST_CHECK_EQUAL (r, this_time); + + BOOST_CHECK_MESSAGE (memcmp (ref_buffer, check_buffer, this_time) == 0, error); + if (memcmp (ref_buffer, check_buffer, this_time)) { + break; + } + + N -= this_time; + } + + delete[] ref_buffer; + delete[] check_buffer; + + fclose (ref_file); + fclose (check_file); +} + + +RNGFixer::RNGFixer () +{ + Kumu::cth_test = true; + Kumu::FortunaRNG().Reset(); +} + + +RNGFixer::~RNGFixer () +{ + Kumu::cth_test = false; } -string private_test = "../libdcp-test-private"; +shared_ptr +simple_picture (boost::filesystem::path path, string suffix) +{ + dcp::MXFMetadata mxf_meta; + mxf_meta.company_name = "OpenDCP"; + mxf_meta.product_name = "OpenDCP"; + mxf_meta.product_version = "0.0.25"; + + shared_ptr mp (new dcp::MonoPictureAsset (dcp::Fraction (24, 1), dcp::SMPTE)); + mp->set_metadata (mxf_meta); + shared_ptr picture_writer = mp->start_write (path / dcp::String::compose("video%1.mxf", suffix), false); + dcp::File j2c ("test/data/32x32_red_square.j2c"); + for (int i = 0; i < 24; ++i) { + picture_writer->write (j2c.data (), j2c.size ()); + } + picture_writer->finalize (); + + return mp; +} + + +shared_ptr +make_simple (boost::filesystem::path path, int reels) +{ + /* Some known metadata */ + dcp::MXFMetadata mxf_meta; + mxf_meta.company_name = "OpenDCP"; + mxf_meta.product_name = "OpenDCP"; + mxf_meta.product_version = "0.0.25"; + + boost::filesystem::remove_all (path); + boost::filesystem::create_directories (path); + shared_ptr d (new dcp::DCP (path)); + shared_ptr cpl (new dcp::CPL ("A Test DCP", dcp::FEATURE)); + cpl->set_annotation_text ("A Test DCP"); + cpl->set_issuer ("OpenDCP 0.0.25"); + cpl->set_creator ("OpenDCP 0.0.25"); + cpl->set_issue_date ("2012-07-17T04:45:18+00:00"); + cpl->set_content_version ( + dcp::ContentVersion("urn:uuid:75ac29aa-42ac-1234-ecae-49251abefd11", "content-version-label-text") + ); + + for (int i = 0; i < reels; ++i) { + string suffix = reels == 1 ? "" : dcp::String::compose("%1", i); + + shared_ptr mp = simple_picture (path, suffix); + + shared_ptr ms (new dcp::SoundAsset(dcp::Fraction(24, 1), 48000, 1, dcp::LanguageTag("en-US"), dcp::SMPTE)); + ms->set_metadata (mxf_meta); + vector active_channels; + active_channels.push_back (dcp::LEFT); + shared_ptr sound_writer = ms->start_write (path / dcp::String::compose("audio%1.mxf", suffix), active_channels); + + SF_INFO info; + info.format = 0; + SNDFILE* sndfile = sf_open ("test/data/1s_24-bit_48k_silence.wav", SFM_READ, &info); + BOOST_CHECK (sndfile); + float buffer[4096*6]; + float* channels[1]; + channels[0] = buffer; + while (true) { + sf_count_t N = sf_readf_float (sndfile, buffer, 4096); + sound_writer->write (channels, N); + if (N < 4096) { + break; + } + } + + sound_writer->finalize (); + + cpl->add (shared_ptr ( + new dcp::Reel ( + shared_ptr(new dcp::ReelMonoPictureAsset(mp, 0)), + shared_ptr(new dcp::ReelSoundAsset(ms, 0)) + ) + )); + } + + d->add (cpl); + return d; +} + + +shared_ptr +simple_subtitle () +{ + return shared_ptr( + new dcp::SubtitleString( + optional(), + false, + false, + false, + dcp::Colour(255, 255, 255), + 42, + 1, + dcp::Time(0, 0, 4, 0, 24), + dcp::Time(0, 0, 8, 0, 24), + 0.5, + dcp::HALIGN_CENTER, + 0.8, + dcp::VALIGN_TOP, + dcp::DIRECTION_LTR, + "Hello world", + dcp::NONE, + dcp::Colour(255, 255, 255), + dcp::Time(), + dcp::Time() + ) + ); +} + + +shared_ptr +make_simple_with_interop_subs (boost::filesystem::path path) +{ + shared_ptr dcp = make_simple (path); + + shared_ptr subs(new dcp::InteropSubtitleAsset()); + subs->add (simple_subtitle()); + + boost::filesystem::create_directory (path / "subs"); + dcp::ArrayData data(4096); + subs->add_font ("afont", data); + subs->write (path / "subs" / "subs.xml"); + + shared_ptr reel_subs(new dcp::ReelSubtitleAsset(subs, dcp::Fraction(24, 1), 240, 0)); + dcp->cpls().front()->reels().front()->add (reel_subs); + + return dcp; +} + + +shared_ptr +make_simple_with_smpte_subs (boost::filesystem::path path) +{ + shared_ptr dcp = make_simple (path); + + shared_ptr subs(new dcp::SMPTESubtitleAsset()); + subs->add (simple_subtitle()); + + dcp::ArrayData data(4096); + subs->write (path / "subs.mxf"); + + shared_ptr reel_subs(new dcp::ReelSubtitleAsset(subs, dcp::Fraction(24, 1), 240, 0)); + dcp->cpls().front()->reels().front()->add (reel_subs); + + return dcp; +} + + +shared_ptr +make_simple_with_interop_ccaps (boost::filesystem::path path) +{ + shared_ptr dcp = make_simple (path); + + shared_ptr subs(new dcp::InteropSubtitleAsset()); + subs->add (simple_subtitle()); + subs->write (path / "ccap.xml"); + + shared_ptr reel_caps(new dcp::ReelClosedCaptionAsset(subs, dcp::Fraction(24, 1), 240, 0)); + dcp->cpls().front()->reels().front()->add (reel_caps); + + return dcp; +} + + +shared_ptr +make_simple_with_smpte_ccaps (boost::filesystem::path path) +{ + shared_ptr dcp = make_simple (path); + + shared_ptr subs(new dcp::SMPTESubtitleAsset()); + subs->add (simple_subtitle()); + subs->write (path / "ccap.mxf"); + + shared_ptr reel_caps(new dcp::ReelClosedCaptionAsset(subs, dcp::Fraction(24, 1), 240, 0)); + dcp->cpls().front()->reels().front()->add (reel_caps); + + return dcp; +} + + +shared_ptr +black_image () +{ + shared_ptr image(new dcp::OpenJPEGImage(dcp::Size(1998, 1080))); + int const pixels = 1998 * 1080; + for (int i = 0; i < 3; ++i) { + memset (image->data(i), 0, pixels * sizeof(int)); + } + return image; +} + + +shared_ptr +black_picture_asset (boost::filesystem::path dir, int frames) +{ + shared_ptr image = black_image (); + dcp::ArrayData frame = dcp::compress_j2k (image, 100000000, 24, false, false); + BOOST_REQUIRE (frame.size() < 230000000 / (24 * 8)); + + shared_ptr asset(new dcp::MonoPictureAsset(dcp::Fraction(24, 1), dcp::SMPTE)); + boost::filesystem::create_directories (dir); + shared_ptr writer = asset->start_write (dir / "pic.mxf", true); + for (int i = 0; i < frames; ++i) { + writer->write (frame.data(), frame.size()); + } + writer->finalize (); + + return shared_ptr(new dcp::ReelMonoPictureAsset(asset, 0)); +} + + +BOOST_GLOBAL_FIXTURE (TestConfig);