X-Git-Url: https://main.carlh.net/gitweb/?p=dcpomatic.git;a=blobdiff_plain;f=test%2Fdcpomatic_time_test.cc;h=5b58c369b047fe3a9b05ae10777d819258d302d3;hp=2b42d22b880788907e276f33a1329c4cb5c1f13d;hb=d0555983d175e76585b2409d982e09f444eab391;hpb=a0d1dd5d91c81ec9907cbc7b890905c463c18f62 diff --git a/test/dcpomatic_time_test.cc b/test/dcpomatic_time_test.cc index 2b42d22b8..5b58c369b 100644 --- a/test/dcpomatic_time_test.cc +++ b/test/dcpomatic_time_test.cc @@ -1,24 +1,37 @@ /* - Copyright (C) 2015 Carl Hetherington + Copyright (C) 2015-2017 Carl Hetherington - This program is free software; you can redistribute it and/or modify + This file is part of DCP-o-matic. + + DCP-o-matic 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, + DCP-o-matic 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 DCP-o-matic. If not, see . */ -#include +/** @file test/dcpomatic_time_test.cc + * @brief Test Time and TimePeriod classes. + * @ingroup selfcontained + */ + #include "lib/dcpomatic_time.h" +#include "lib/dcpomatic_time_coalesce.h" +#include +#include +#include + +using std::list; +using std::cout; +using namespace dcpomatic; BOOST_AUTO_TEST_CASE (dcpomatic_time_test) { @@ -36,3 +49,292 @@ BOOST_AUTO_TEST_CASE (dcpomatic_time_test) } } } + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_overlaps_test) +{ + /* Taking times as the start of a sampling interval + + |--|--|--|--|--|--|--|--|--|--| + 0 1 2 3 4 5 6 7 8 9 | + |--|--|--|--|--|--|--|--|--|--| + + <------a----><----b-----> + + and saying `from' is the start of the first sampling + interval and `to' is the start of the interval after + the period... a and b do not overlap. + */ + + TimePeriod a (DCPTime (0), DCPTime (4)); + TimePeriod b (DCPTime (4), DCPTime (8)); + BOOST_CHECK (!a.overlap (b)); + + /* Some more obvious non-overlaps */ + a = TimePeriod (DCPTime (0), DCPTime (4)); + b = TimePeriod (DCPTime (5), DCPTime (8)); + BOOST_CHECK (!a.overlap (b)); + + /* Some overlaps */ + a = TimePeriod (DCPTime (0), DCPTime (4)); + b = TimePeriod (DCPTime (3), DCPTime (8)); + BOOST_CHECK (a.overlap(b)); + BOOST_CHECK (a.overlap(b).get() == DCPTimePeriod(DCPTime(3), DCPTime(4))); + a = TimePeriod (DCPTime (1), DCPTime (9)); + b = TimePeriod (DCPTime (0), DCPTime (10)); + BOOST_CHECK (a.overlap(b)); + BOOST_CHECK (a.overlap(b).get() == DCPTimePeriod(DCPTime(1), DCPTime(9))); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test1) +{ + DCPTimePeriod A (DCPTime (0), DCPTime (106)); + list B; + B.push_back (DCPTimePeriod (DCPTime (0), DCPTime (42))); + B.push_back (DCPTimePeriod (DCPTime (52), DCPTime (91))); + B.push_back (DCPTimePeriod (DCPTime (94), DCPTime (106))); + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (42)); + BOOST_CHECK (i->to == DCPTime (52)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (91)); + BOOST_CHECK (i->to == DCPTime (94)); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test2) +{ + DCPTimePeriod A (DCPTime (0), DCPTime (106)); + list B; + B.push_back (DCPTimePeriod (DCPTime (14), DCPTime (42))); + B.push_back (DCPTimePeriod (DCPTime (52), DCPTime (91))); + B.push_back (DCPTimePeriod (DCPTime (94), DCPTime (106))); + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (0)); + BOOST_CHECK (i->to == DCPTime (14)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (42)); + BOOST_CHECK (i->to == DCPTime (52)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (91)); + BOOST_CHECK (i->to == DCPTime (94)); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test3) +{ + DCPTimePeriod A (DCPTime (0), DCPTime (106)); + list B; + B.push_back (DCPTimePeriod (DCPTime (14), DCPTime (42))); + B.push_back (DCPTimePeriod (DCPTime (52), DCPTime (91))); + B.push_back (DCPTimePeriod (DCPTime (94), DCPTime (99))); + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (0)); + BOOST_CHECK (i->to == DCPTime (14)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (42)); + BOOST_CHECK (i->to == DCPTime (52)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (91)); + BOOST_CHECK (i->to == DCPTime (94)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (99)); + BOOST_CHECK (i->to == DCPTime (106)); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test4) +{ + DCPTimePeriod A (DCPTime (0), DCPTime (106)); + list B; + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (0)); + BOOST_CHECK (i->to == DCPTime (106)); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test5) +{ + DCPTimePeriod A (DCPTime (0), DCPTime (106)); + list B; + B.push_back (DCPTimePeriod (DCPTime (14), DCPTime (42))); + B.push_back (DCPTimePeriod (DCPTime (42), DCPTime (91))); + B.push_back (DCPTimePeriod (DCPTime (94), DCPTime (99))); + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (0)); + BOOST_CHECK (i->to == DCPTime (14)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from ==DCPTime (91)); + BOOST_CHECK (i->to == DCPTime (94)); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (99)); + BOOST_CHECK (i->to == DCPTime (106)); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test6) +{ + DCPTimePeriod A (DCPTime (0), DCPTime (106)); + list B; + B.push_back (DCPTimePeriod (DCPTime (0), DCPTime (42))); + B.push_back (DCPTimePeriod (DCPTime (42), DCPTime (91))); + B.push_back (DCPTimePeriod (DCPTime (91), DCPTime (106))); + list r = subtract (A, B); + BOOST_CHECK (r.empty()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test7) +{ + DCPTimePeriod A (DCPTime (228), DCPTime (356)); + list B; + B.push_back (DCPTimePeriod (DCPTime (34), DCPTime (162))); + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (i->from == DCPTime (228)); + BOOST_CHECK (i->to == DCPTime (356)); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_subtract_test8) +{ + DCPTimePeriod A (DCPTime(0), DCPTime(32000)); + list B; + B.push_back (DCPTimePeriod (DCPTime(8000), DCPTime(20000))); + B.push_back (DCPTimePeriod (DCPTime(28000), DCPTime(32000))); + list r = subtract (A, B); + list::const_iterator i = r.begin (); + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (*i == DCPTimePeriod(DCPTime(0), DCPTime(8000))); + ++i; + BOOST_REQUIRE (i != r.end ()); + BOOST_CHECK (*i == DCPTimePeriod(DCPTime(20000), DCPTime(28000))); + ++i; + BOOST_REQUIRE (i == r.end ()); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_coalesce_test1) +{ + DCPTimePeriod A (DCPTime(14), DCPTime(29)); + DCPTimePeriod B (DCPTime(45), DCPTime(91)); + list p; + p.push_back (A); + p.push_back (B); + list q = coalesce (p); + BOOST_REQUIRE_EQUAL (q.size(), 2); + BOOST_CHECK (q.front() == DCPTimePeriod(DCPTime(14), DCPTime(29))); + BOOST_CHECK (q.back () == DCPTimePeriod(DCPTime(45), DCPTime(91))); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_coalesce_test2) +{ + DCPTimePeriod A (DCPTime(14), DCPTime(29)); + DCPTimePeriod B (DCPTime(26), DCPTime(91)); + list p; + p.push_back (A); + p.push_back (B); + list q = coalesce (p); + BOOST_REQUIRE_EQUAL (q.size(), 1); + BOOST_CHECK (q.front() == DCPTimePeriod(DCPTime(14), DCPTime(91))); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_coalesce_test3) +{ + DCPTimePeriod A (DCPTime(14), DCPTime(29)); + DCPTimePeriod B (DCPTime(29), DCPTime(91)); + list p; + p.push_back (A); + p.push_back (B); + list q = coalesce (p); + BOOST_REQUIRE_EQUAL (q.size(), 1); + BOOST_CHECK (q.front() == DCPTimePeriod(DCPTime(14), DCPTime(91))); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_coalesce_test4) +{ + DCPTimePeriod A (DCPTime(14), DCPTime(29)); + DCPTimePeriod B (DCPTime(20), DCPTime(91)); + DCPTimePeriod C (DCPTime(35), DCPTime(106)); + list p; + p.push_back (A); + p.push_back (B); + p.push_back (C); + list q = coalesce (p); + BOOST_REQUIRE_EQUAL (q.size(), 1); + BOOST_CHECK (q.front() == DCPTimePeriod(DCPTime(14), DCPTime(106))); +} + +BOOST_AUTO_TEST_CASE (dcpomatic_time_period_coalesce_test5) +{ + DCPTimePeriod A (DCPTime(14), DCPTime(29)); + DCPTimePeriod B (DCPTime(20), DCPTime(91)); + DCPTimePeriod C (DCPTime(100), DCPTime(106)); + list p; + p.push_back (A); + p.push_back (B); + p.push_back (C); + list q = coalesce (p); + BOOST_REQUIRE_EQUAL (q.size(), 2); + BOOST_CHECK (q.front() == DCPTimePeriod(DCPTime(14), DCPTime(91))); + BOOST_CHECK (q.back() == DCPTimePeriod(DCPTime(100), DCPTime(106))); +} + +/* Straightforward test of DCPTime::ceil */ +BOOST_AUTO_TEST_CASE (dcpomatic_time_ceil_test) +{ + BOOST_CHECK_EQUAL (DCPTime(0).ceil(DCPTime::HZ / 2).get(), 0); + BOOST_CHECK_EQUAL (DCPTime(1).ceil(DCPTime::HZ / 2).get(), 2); + BOOST_CHECK_EQUAL (DCPTime(2).ceil(DCPTime::HZ / 2).get(), 2); + BOOST_CHECK_EQUAL (DCPTime(3).ceil(DCPTime::HZ / 2).get(), 4); + + BOOST_CHECK_EQUAL (DCPTime(0).ceil(DCPTime::HZ / 42).get(), 0); + BOOST_CHECK_EQUAL (DCPTime(1).ceil(DCPTime::HZ / 42).get(), 42); + BOOST_CHECK_EQUAL (DCPTime(42).ceil(DCPTime::HZ / 42).get(), 42); + BOOST_CHECK_EQUAL (DCPTime(43).ceil(DCPTime::HZ / 42).get(), 84); + + /* Check that rounding up to non-integer frame rates works */ + BOOST_CHECK_EQUAL (DCPTime(45312).ceil(29.976).get(), 48038); + + /* Check another tricky case that used to fail */ + BOOST_CHECK_EQUAL (DCPTime(212256039).ceil(23.976).get(), 212256256); +} + +/* Straightforward test of DCPTime::floor */ +BOOST_AUTO_TEST_CASE (dcpomatic_time_floor_test) +{ + BOOST_CHECK_EQUAL (DCPTime(0).floor(DCPTime::HZ / 2).get(), 0); + BOOST_CHECK_EQUAL (DCPTime(1).floor(DCPTime::HZ / 2).get(), 0); + BOOST_CHECK_EQUAL (DCPTime(2).floor(DCPTime::HZ / 2).get(), 2); + BOOST_CHECK_EQUAL (DCPTime(3).floor(DCPTime::HZ / 2).get(), 2); + + BOOST_CHECK_EQUAL (DCPTime(0).floor(DCPTime::HZ / 42).get(), 0); + BOOST_CHECK_EQUAL (DCPTime(1).floor(DCPTime::HZ / 42).get(), 0); + BOOST_CHECK_EQUAL (DCPTime(42).floor(DCPTime::HZ / 42.0).get(), 42); + BOOST_CHECK_EQUAL (DCPTime(43).floor(DCPTime::HZ / 42.0).get(), 42); + + /* Check that rounding down to non-integer frame rates works */ + BOOST_CHECK_EQUAL (DCPTime(45312).floor(29.976).get(), 44836); +}