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 dataset join operation
10 // ***************************************************************************
12 #ifndef BOOST_TEST_DATA_MONOMORPHIC_JOIN_HPP_112711GER
13 #define BOOST_TEST_DATA_MONOMORPHIC_JOIN_HPP_112711GER
16 #include <boost/test/data/config.hpp>
17 #include <boost/test/data/monomorphic/fwd.hpp>
19 #include <boost/core/enable_if.hpp>
20 #include <boost/mpl/identity.hpp>
22 #include <boost/test/detail/suppress_warnings.hpp>
24 //____________________________________________________________________________//
29 namespace monomorphic {
31 // ************************************************************************** //
32 // ************** join ************** //
33 // ************************************************************************** //
35 //! Defines a new dataset from the concatenation of two datasets
37 //! The size of the resulting dataset is the sum of the two underlying datasets. The arity of the 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;
47 using iter1_ret = decltype(*std::declval<DataSet1>().begin());
48 using iter2_ret = decltype(*std::declval<DataSet2>().begin());
52 enum { arity = dataset1_decay::arity };
54 using sample_t = typename std::conditional<
55 std::is_reference<iter1_ret>::value && std::is_reference<iter2_ret>::value && std::is_same<iter1_ret, iter2_ret>::value,
57 typename std::remove_reference<iter1_ret>::type
63 explicit iterator( dataset1_iter&& it1, dataset2_iter&& it2, data::size_t first_size )
64 : m_it1( std::move( it1 ) )
65 , m_it2( std::move( it2 ) )
66 , m_first_size( first_size )
69 // forward iterator interface
70 // The returned sample should be by value, as the operator* may return a temporary object
71 sample_t operator*() const { return m_first_size > 0 ? *m_it1 : *m_it2; }
72 void operator++() { if( m_first_size > 0 ) { --m_first_size; ++m_it1; } else ++m_it2; }
78 data::size_t m_first_size;
82 join( DataSet1&& ds1, DataSet2&& ds2 )
83 : m_ds1( std::forward<DataSet1>( ds1 ) )
84 , m_ds2( std::forward<DataSet2>( ds2 ) )
89 : m_ds1( std::forward<DataSet1>( j.m_ds1 ) )
90 , m_ds2( std::forward<DataSet2>( j.m_ds2 ) )
94 data::size_t size() const { return m_ds1.size() + m_ds2.size(); }
95 iterator begin() const { return iterator( m_ds1.begin(), m_ds2.begin(), m_ds1.size() ); }
103 //____________________________________________________________________________//
105 // A joined dataset is a dataset.
106 template<typename DataSet1, typename DataSet2>
107 struct is_dataset<join<DataSet1,DataSet2>> : mpl::true_ {};
109 //____________________________________________________________________________//
111 namespace result_of {
113 //! Result type of the join operation on datasets.
114 template<typename DataSet1Gen, typename DataSet2Gen>
116 typedef monomorphic::join<typename DataSet1Gen::type,typename DataSet2Gen::type> type;
119 } // namespace result_of
121 //____________________________________________________________________________//
123 template<typename DataSet1, typename DataSet2>
124 inline typename boost::lazy_enable_if_c<is_dataset<DataSet1>::value && is_dataset<DataSet2>::value,
125 result_of::join<mpl::identity<DataSet1>,mpl::identity<DataSet2>>
127 operator+( DataSet1&& ds1, DataSet2&& ds2 )
129 return join<DataSet1,DataSet2>( std::forward<DataSet1>( ds1 ), std::forward<DataSet2>( ds2 ) );
132 //____________________________________________________________________________//
134 template<typename DataSet1, typename DataSet2>
135 inline typename boost::lazy_enable_if_c<is_dataset<DataSet1>::value && !is_dataset<DataSet2>::value,
136 result_of::join<mpl::identity<DataSet1>,data::result_of::make<DataSet2>>
138 operator+( DataSet1&& ds1, DataSet2&& ds2 )
140 return std::forward<DataSet1>( ds1 ) + data::make( std::forward<DataSet2>( ds2 ) );
143 //____________________________________________________________________________//
145 template<typename DataSet1, typename DataSet2>
146 inline typename boost::lazy_enable_if_c<!is_dataset<DataSet1>::value && is_dataset<DataSet2>::value,
147 result_of::join<data::result_of::make<DataSet1>,mpl::identity<DataSet2>>
149 operator+( DataSet1&& ds1, DataSet2&& ds2 )
151 return data::make( std::forward<DataSet1>(ds1) ) + std::forward<DataSet2>( ds2 );
154 } // namespace monomorphic
156 } // namespace unit_test
159 #include <boost/test/detail/enable_warnings.hpp>
161 #endif // BOOST_TEST_DATA_MONOMORPHIC_JOIN_HPP_112711GER