// Boost.Units - A C++ library for zero-overhead dimensional analysis and // unit/quantity manipulation and conversion // // Copyright (C) 2003-2008 Matthias Christian Schabel // Copyright (C) 2008 Steven Watanabe // // Distributed under the Boost Software License, Version 1.0. (See // accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_UNITS_DIMENSION_IMPL_HPP #define BOOST_UNITS_DIMENSION_IMPL_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include /// \file /// \brief Core class and metaprogramming utilities for compile-time dimensional analysis. namespace boost { namespace units { namespace detail { template struct insertion_sort_dims_insert; template struct insertion_sort_dims_comparison_impl; // have to recursively add the element to the next sequence. template<> struct insertion_sort_dims_comparison_impl { template struct apply { typedef list< typename Begin::item, typename insertion_sort_dims_insert::template apply< typename Begin::next, T >::type > type; }; }; // either prepend the current element or join it to // the first remaining element of the sequence. template<> struct insertion_sort_dims_comparison_impl { template struct apply { typedef typename push_front_or_add::type type; }; }; template struct insertion_sort_dims_insert { template struct apply { typedef typename insertion_sort_dims_comparison_impl::value>::template apply< Begin, N, T >::type type; }; }; template<> struct insertion_sort_dims_insert<0> { template struct apply { typedef list type; }; }; template struct insertion_sort_dims_mpl_sequence { template struct apply { typedef typename insertion_sort_dims_mpl_sequence::template apply::type>::type next; typedef typename insertion_sort_dims_insert<(next::size::value)>::template apply::type>::type type; }; }; template<> struct insertion_sort_dims_mpl_sequence<0> { template struct apply { typedef dimensionless_type type; }; }; template struct insertion_sort_dims_impl { template struct apply { typedef typename insertion_sort_dims_impl::template apply::type next; typedef typename insertion_sort_dims_insert<(next::size::value)>::template apply::type type; }; }; template<> struct insertion_sort_dims_impl<0> { template struct apply { typedef dimensionless_type type; }; }; template struct sort_dims { typedef typename insertion_sort_dims_mpl_sequence::value>::template apply::type>::type type; }; template struct sort_dims > { typedef typename insertion_sort_dims_impl::size::value>::template apply >::type type; }; /// sorted sequences can be merged in linear time template struct merge_dimensions_func; template struct merge_dimensions_impl; template<> struct merge_dimensions_func { template struct apply { typedef list< typename Begin1::item, typename merge_dimensions_impl::template apply< typename Begin1::next, Begin2 >::type > type; }; }; template<> struct merge_dimensions_func { template struct apply { typedef list< typename Begin2::item, typename merge_dimensions_impl::template apply< typename Begin2::next, Begin1 >::type > type; }; }; template<> struct merge_dimensions_func { template struct apply { typedef typename mpl::plus::type combined; typedef typename push_front_if::value>::template apply< typename merge_dimensions_impl::template apply< typename Begin1::next, typename Begin2::next >::type, combined >::type type; }; }; template struct merge_dimensions_impl { template struct apply { typedef typename Begin1::item dim1; typedef typename Begin2::item dim2; typedef typename merge_dimensions_func<(mpl::less::value == true), (mpl::less::value == true)>::template apply< Begin1, Begin2, N1, N2 >::type type; }; }; template struct merge_dimensions { typedef typename detail::merge_dimensions_impl::template apply< Sequence1, Sequence2 >::type type; }; template struct iterator_to_list { template struct apply { typedef list< typename Begin::item, typename iterator_to_list::template apply< typename Begin::next >::type > type; }; }; template<> struct iterator_to_list<0> { template struct apply { typedef dimensionless_type type; }; }; template struct merge_dimensions_impl { template struct apply { typedef typename iterator_to_list::template apply::type type; }; }; template struct merge_dimensions_impl<0, N> { template struct apply { typedef typename iterator_to_list::template apply::type type; }; }; template<> struct merge_dimensions_impl<0, 0> { template struct apply { typedef dimensionless_type type; }; }; template struct static_inverse_impl { template struct apply { typedef list< typename mpl::negate::type, typename static_inverse_impl::template apply< typename Begin::next >::type > type; }; }; template<> struct static_inverse_impl<0> { template struct apply { typedef dimensionless_type type; }; }; template struct static_power_impl { template struct apply { typedef list< typename mpl::times::type, typename detail::static_power_impl::template apply::type > type; }; }; template<> struct static_power_impl<0> { template struct apply { typedef dimensionless_type type; }; }; template struct static_root_impl { template struct apply { typedef list< typename mpl::divides::type, typename detail::static_root_impl::template apply::type > type; }; }; template<> struct static_root_impl<0> { template struct apply { typedef dimensionless_type type; }; }; } // namespace detail } // namespace units } // namespace boost #endif // BOOST_UNITS_DIMENSION_IMPL_HPP