310 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| // 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_DETAIL_HETEROGENEOUS_CONVERSION_HPP
 | |
| #define BOOST_UNITS_DETAIL_HETEROGENEOUS_CONVERSION_HPP
 | |
| 
 | |
| #include <boost/mpl/minus.hpp>
 | |
| #include <boost/mpl/times.hpp>
 | |
| 
 | |
| #include <boost/units/static_rational.hpp>
 | |
| #include <boost/units/homogeneous_system.hpp>
 | |
| #include <boost/units/detail/linear_algebra.hpp>
 | |
| 
 | |
| namespace boost {
 | |
| 
 | |
| namespace units {
 | |
| 
 | |
| namespace detail {
 | |
| 
 | |
| struct solve_end {
 | |
|     template<class Begin, class Y>
 | |
|     struct apply {
 | |
|         typedef dimensionless_type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| struct no_solution {};
 | |
| 
 | |
| template<class X1, class X2, class Next>
 | |
| struct solve_normal {
 | |
|     template<class Begin, class Y>
 | |
|     struct apply {
 | |
|         typedef typename Begin::next next;
 | |
|         typedef list<
 | |
|             typename mpl::minus<
 | |
|                 typename mpl::times<X1, Y>::type,
 | |
|                 typename mpl::times<X2, typename Begin::item>::type
 | |
|             >::type,
 | |
|             typename Next::template apply<next, Y>::type
 | |
|         > type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<class Next>
 | |
| struct solve_leading_zeroes {
 | |
|     template<class Begin>
 | |
|     struct apply {
 | |
|         typedef list<
 | |
|             typename Begin::item,
 | |
|             typename Next::template apply<typename Begin::next>::type
 | |
|         > type;
 | |
|     };
 | |
|     typedef solve_leading_zeroes type;
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct solve_leading_zeroes<no_solution> {
 | |
|     typedef no_solution type;
 | |
| };
 | |
| 
 | |
| template<class Next>
 | |
| struct solve_first_non_zero {
 | |
|     template<class Begin>
 | |
|     struct apply {
 | |
|         typedef typename Next::template apply<
 | |
|             typename Begin::next,
 | |
|             typename Begin::item
 | |
|         >::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<class Next>
 | |
| struct solve_internal_zero {
 | |
|     template<class Begin, class Y>
 | |
|     struct apply {
 | |
|         typedef list<
 | |
|             typename Begin::item,
 | |
|             typename Next::template apply<typename Begin::next, Y>::type
 | |
|         > type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<class T>
 | |
| struct make_solve_list_internal_zero {
 | |
|     template<class Next, class X>
 | |
|     struct apply {
 | |
|         typedef solve_normal<T, X, Next> type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_solve_list_internal_zero<static_rational<0> > {
 | |
|     template<class Next, class X>
 | |
|     struct apply {
 | |
|         typedef solve_internal_zero<Next> type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct make_solve_list_normal {
 | |
|     template<class Begin, class X>
 | |
|     struct apply {
 | |
|         typedef typename make_solve_list_internal_zero<
 | |
|             typename Begin::item
 | |
|         >::template apply<
 | |
|             typename make_solve_list_normal<N-1>::template apply<typename Begin::next, X>::type,
 | |
|             X
 | |
|         >::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_solve_list_normal<0> {
 | |
|     template<class Begin, class X>
 | |
|     struct apply {
 | |
|         typedef solve_end type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct make_solve_list_leading_zeroes;
 | |
| 
 | |
| template<class T>
 | |
| struct make_solve_list_first_non_zero {
 | |
|     template<class Begin, int N>
 | |
|     struct apply {
 | |
|         typedef solve_first_non_zero<
 | |
|             typename make_solve_list_normal<N-1>::template apply<
 | |
|                 typename Begin::next,
 | |
|                 typename Begin::item
 | |
|             >::type
 | |
|         > type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_solve_list_first_non_zero<static_rational<0> > {
 | |
|     template<class Begin, int N>
 | |
|     struct apply {
 | |
|         typedef typename solve_leading_zeroes<
 | |
|             typename make_solve_list_leading_zeroes<N-1>::template apply<
 | |
|                 typename Begin::next
 | |
|             >::type
 | |
|         >::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct make_solve_list_leading_zeroes {
 | |
|     template<class Begin>
 | |
|     struct apply {
 | |
|         typedef typename make_solve_list_first_non_zero<typename Begin::item>::template apply<Begin, N>::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_solve_list_leading_zeroes<0> {
 | |
|     template<class Begin>
 | |
|     struct apply {
 | |
|         typedef no_solution type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct try_add_unit_impl {
 | |
|     template<class Begin, class L>
 | |
|     struct apply {
 | |
|         typedef typename try_add_unit_impl<N-1>::template apply<typename Begin::next, L>::type next;
 | |
|         typedef typename Begin::item::template apply<next>::type type;
 | |
|         BOOST_STATIC_ASSERT((next::size::value - 1 == type::size::value));
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct try_add_unit_impl<0> {
 | |
|     template<class Begin, class L>
 | |
|     struct apply {
 | |
|         typedef L type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct make_homogeneous_system_impl;
 | |
| 
 | |
| template<class T, bool is_done>
 | |
| struct make_homogeneous_system_func;
 | |
| 
 | |
| template<class T>
 | |
| struct make_homogeneous_system_func<T, false> {
 | |
|     template<class Begin, class Current, class Units, class Dimensions, int N>
 | |
|     struct apply {
 | |
|         typedef typename make_homogeneous_system_impl<N-1>::template apply<
 | |
|             typename Begin::next,
 | |
|             list<T, Current>,
 | |
|             list<typename Begin::item, Units>,
 | |
|             Dimensions
 | |
|         >::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<class T>
 | |
| struct make_homogeneous_system_func<T, true> {
 | |
|     template<class Begin, class Current, class Units, class Dimensions, int N>
 | |
|     struct apply {
 | |
|         typedef list<typename Begin::item, Units> type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_homogeneous_system_func<no_solution, false> {
 | |
|     template<class Begin, class Current, class Units, class Dimensions, int N>
 | |
|     struct apply {
 | |
|         typedef typename make_homogeneous_system_impl<N-1>::template apply<
 | |
|             typename Begin::next,
 | |
|             Current,
 | |
|             Units,
 | |
|             Dimensions
 | |
|         >::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_homogeneous_system_func<no_solution, true> {
 | |
|     template<class Begin, class Current, class Units, class Dimensions, int N>
 | |
|     struct apply {
 | |
|         typedef typename make_homogeneous_system_impl<N-1>::template apply<
 | |
|             typename Begin::next,
 | |
|             Current,
 | |
|             Units,
 | |
|             Dimensions
 | |
|         >::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct make_homogeneous_system_impl {
 | |
|     template<class Begin, class Current, class Units, class Dimensions>
 | |
|     struct apply {
 | |
|         typedef typename expand_dimensions<Dimensions::size::value>::template apply<
 | |
|             Dimensions,
 | |
|             typename Begin::item::dimension_type
 | |
|         >::type dimensions;
 | |
|         typedef typename try_add_unit_impl<Current::size::value>::template apply<Current, dimensions>::type new_element;
 | |
|         typedef typename make_solve_list_leading_zeroes<new_element::size::value>::template apply<new_element>::type new_func;
 | |
|         typedef typename make_homogeneous_system_func<
 | |
|             new_func,
 | |
|             ((Current::size::value)+1) == (Dimensions::size::value)
 | |
|         >::template apply<Begin, Current, Units, Dimensions, N>::type type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct make_homogeneous_system_impl<0> {
 | |
|     template<class Begin, class Current, class Units, class Dimensions>
 | |
|     struct apply {
 | |
|         typedef Units type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<class Units>
 | |
| struct make_homogeneous_system {
 | |
|     typedef typename find_base_dimensions<Units>::type base_dimensions;
 | |
|     typedef homogeneous_system<
 | |
|         typename insertion_sort<
 | |
|             typename make_homogeneous_system_impl<
 | |
|                 Units::size::value
 | |
|             >::template apply<
 | |
|                 Units,
 | |
|                 dimensionless_type,
 | |
|                 dimensionless_type,
 | |
|                 base_dimensions
 | |
|             >::type
 | |
|         >::type
 | |
|     > type;
 | |
| };
 | |
| 
 | |
| template<int N>
 | |
| struct extract_base_units {
 | |
|     template<class Begin, class T>
 | |
|     struct apply {
 | |
|         typedef list<
 | |
|             typename Begin::item::tag_type,
 | |
|             typename extract_base_units<N-1>::template apply<typename Begin::next, T>::type
 | |
|         > type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct extract_base_units<0> {
 | |
|     template<class Begin, class T>
 | |
|     struct apply {
 | |
|         typedef T type;
 | |
|     };
 | |
| };
 | |
| 
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| #endif
 | 
