1 // (C) Copyright Gennadiy Rozental 2001.
2 // Distributed under the Boost Software License, Version 1.0.
3 // (See accompanying file LICENSE_1_0.txt or copy at
4 // http://www.boost.org/LICENSE_1_0.txt)
6 // See http://www.boost.org/libs/test for the library home page.
9 /// Defines monomorphic dataset based on zipping of 2 other monomorphic datasets
10 // ***************************************************************************
12 #ifndef BOOST_TEST_DATA_MONOMORPHIC_ZIP_HPP_102211GER
13 #define BOOST_TEST_DATA_MONOMORPHIC_ZIP_HPP_102211GER
16 #include <boost/test/data/config.hpp>
18 #if !defined(BOOST_TEST_NO_ZIP_COMPOSITION_AVAILABLE) || defined(BOOST_TEST_DOXYGEN_DOC__)
20 #include <boost/test/data/monomorphic/fwd.hpp>
21 #include <boost/test/data/monomorphic/sample_merge.hpp>
23 #include <boost/test/detail/suppress_warnings.hpp>
29 namespace monomorphic {
31 // ************************************************************************** //
32 // ************** zip ************** //
33 // ************************************************************************** //
37 //! A zip of two datasets is a dataset whose arity is the sum of the operand datasets arity. The size is given by
38 //! the function creating the instance (see @c operator^ on datasets).
39 template<typename DataSet1, typename DataSet2>
41 typedef typename boost::decay<DataSet1>::type dataset1_decay;
42 typedef typename boost::decay<DataSet2>::type dataset2_decay;
44 typedef typename dataset1_decay::iterator dataset1_iter;
45 typedef typename dataset2_decay::iterator dataset2_iter;
48 enum { arity = dataset1_decay::arity + dataset2_decay::arity };
52 explicit iterator( dataset1_iter iter1, dataset2_iter iter2 )
53 : m_iter1( std::move( iter1 ) )
54 , m_iter2( std::move( iter2 ) )
57 using iterator_sample = decltype(
58 sample_merge( *std::declval<dataset1_iter>(),
59 *std::declval<dataset2_iter>()) );
61 // forward iterator interface
62 auto operator*() const -> iterator_sample {
63 return sample_merge( *m_iter1, *m_iter2 );
65 void operator++() { ++m_iter1; ++m_iter2; }
69 dataset1_iter m_iter1;
70 dataset2_iter m_iter2;
73 typedef typename iterator::iterator_sample sample;
77 //! The datasets are moved and not copied.
78 zip( DataSet1&& ds1, DataSet2&& ds2, data::size_t size )
79 : m_ds1( std::forward<DataSet1>( ds1 ) )
80 , m_ds2( std::forward<DataSet2>( ds2 ) )
86 : m_ds1( std::forward<DataSet1>( j.m_ds1 ) )
87 , m_ds2( std::forward<DataSet2>( j.m_ds2 ) )
92 data::size_t size() const { return m_size; }
93 iterator begin() const { return iterator( m_ds1.begin(), m_ds2.begin() ); }
102 //____________________________________________________________________________//
104 //! Zipped datasets results in a dataset.
105 template<typename DataSet1, typename DataSet2>
106 struct is_dataset<zip<DataSet1,DataSet2>> : mpl::true_ {};
108 //____________________________________________________________________________//
110 namespace ds_detail {
112 //! Handles the sise of the resulting zipped dataset.
113 template<typename DataSet1, typename DataSet2>
115 zip_size( DataSet1&& ds1, DataSet2&& ds2 )
117 data::size_t ds1_size = ds1.size();
118 data::size_t ds2_size = ds2.size();
120 if( ds1_size == ds2_size )
123 if( ds1_size == 1 || ds1_size.is_inf() )
126 if( ds2_size == 1 || ds2_size.is_inf() )
129 BOOST_TEST_DS_ERROR( "Can't zip datasets of different sizes" );
132 } // namespace ds_detail
134 //____________________________________________________________________________//
136 namespace result_of {
138 //! Result type of the zip operator.
139 template<typename DS1Gen, typename DS2Gen>
141 typedef monomorphic::zip<typename DS1Gen::type,typename DS2Gen::type> type;
144 } // namespace result_of
146 //____________________________________________________________________________//
148 //! Overload operator for zip support
149 template<typename DataSet1, typename DataSet2>
150 inline typename boost::lazy_enable_if_c<is_dataset<DataSet1>::value && is_dataset<DataSet2>::value,
151 result_of::zip<mpl::identity<DataSet1>,mpl::identity<DataSet2>>
153 operator^( DataSet1&& ds1, DataSet2&& ds2 )
155 return zip<DataSet1,DataSet2>( std::forward<DataSet1>( ds1 ),
156 std::forward<DataSet2>( ds2 ),
157 ds_detail::zip_size( ds1, ds2 ) );
160 //____________________________________________________________________________//
162 //! @overload boost::unit_test::data::monomorphic::operator^()
163 template<typename DataSet1, typename DataSet2>
164 inline typename boost::lazy_enable_if_c<is_dataset<DataSet1>::value && !is_dataset<DataSet2>::value,
165 result_of::zip<mpl::identity<DataSet1>,data::result_of::make<DataSet2>>
167 operator^( DataSet1&& ds1, DataSet2&& ds2 )
169 return std::forward<DataSet1>( ds1 ) ^ data::make( std::forward<DataSet2>( ds2 ) );
172 //____________________________________________________________________________//
174 //! @overload boost::unit_test::data::monomorphic::operator^()
175 template<typename DataSet1, typename DataSet2>
176 inline typename boost::lazy_enable_if_c<!is_dataset<DataSet1>::value && is_dataset<DataSet2>::value,
177 result_of::zip<data::result_of::make<DataSet1>,mpl::identity<DataSet2>>
179 operator^( DataSet1&& ds1, DataSet2&& ds2 )
181 return data::make( std::forward<DataSet1>( ds1 ) ) ^ std::forward<DataSet2>( ds2 );
184 } // namespace monomorphic
186 } // namespace unit_test
189 #include <boost/test/detail/enable_warnings.hpp>
191 #endif // BOOST_TEST_NO_ZIP_COMPOSITION_AVAILABLE
193 #endif // BOOST_TEST_DATA_MONOMORPHIC_ZIP_HPP_102211GER