X-Git-Url: https://git.dogcows.com/gitweb?p=chaz%2Fyoink;a=blobdiff_plain;f=src%2FMoof%2Fcml%2Fmatrix%2Fmatrix_promotions.h;fp=src%2FMoof%2Fcml%2Fmatrix%2Fmatrix_promotions.h;h=18e94bb2509cea057445c4081cf1100817e94013;hp=0000000000000000000000000000000000000000;hb=c2321281bf12a7efaedde930422c7ddbc92080d4;hpb=87bc17e55b0c1dc73ecc66df856d3f08fd7a7724 diff --git a/src/Moof/cml/matrix/matrix_promotions.h b/src/Moof/cml/matrix/matrix_promotions.h new file mode 100644 index 0000000..18e94bb --- /dev/null +++ b/src/Moof/cml/matrix/matrix_promotions.h @@ -0,0 +1,171 @@ +/* -*- C++ -*- ------------------------------------------------------------ + +Copyright (c) 2007 Jesse Anders and Demian Nave http://cmldev.net/ + +The Configurable Math Library (CML) is distributed under the terms of the +Boost Software License, v1.0 (see cml/LICENSE for details). + + *-----------------------------------------------------------------------*/ +/** @file + * @brief + * + * Defines promotions for matrices used in matrix/matrix or matrix/scalar + * expressions. + * + * @sa UnaryMat4_TOp + * @sa BinaryMat4_TOp + */ + +#ifndef matrix_promotions_h +#define matrix_promotions_h + +#include +#include +#include +#include +#include + +/* This is used below to create a more meaningful compile-time error when + * either argument to OuterPromote has the wrong orientation. + */ +struct outer_promote_expects_properly_oriented_args_error; + +namespace cml { +namespace et { + +/* Default matrix type promotion template. */ +template struct MatrixPromote; + +/** Type promotion for two matrix types. + * + * @note This always uses the basis orientation of the left-hand matrix. + * @bug This always uses the basis orientation of the left-hand matrix, + * which is not always correct. + */ +template +struct MatrixPromote< cml::matrix, cml::matrix > +{ + /* Promote the arrays: */ + typedef typename ArrayPromote< + typename cml::matrix::array_type, + typename cml::matrix::array_type + >::type promoted_array; + + /* The deduced matrix result type: */ + typedef cml::matrix< + typename promoted_array::value_type, + typename promoted_array::generator_type, + BO1, + typename promoted_array::layout + > type; + + /* The deduced temporary type: */ + typedef typename type::temporary_type temporary_type; +}; + +/** + * NOTE: MatrixPromote* are somewhat ad hoc, and were added to + * simplify the code for matrix slerp/squad/etc. + */ + +/** Type promotion for two matrix types. */ +template < class Mat1_T, class Mat2_T > +struct MatrixPromote2 +{ + typedef typename MatrixPromote< + typename Mat1_T::temporary_type, typename Mat2_T::temporary_type + >::temporary_type temporary_type; + typedef typename temporary_type::value_type value_type; +}; + +/** Type promotion for three matrix types. */ +template < class Mat1_T, class Mat2_T, class Mat3_T > +struct MatrixPromote3 +{ + typedef typename MatrixPromote< + typename Mat1_T::temporary_type, + typename MatrixPromote< + typename Mat2_T::temporary_type, typename Mat3_T::temporary_type + >::temporary_type + >::temporary_type temporary_type; + typedef typename temporary_type::value_type value_type; +}; + +/** Type promotion for four matrix types. */ +template < class Mat1_T, class Mat2_T, class Mat3_T, class Mat4_T > +struct MatrixPromote4 +{ + typedef typename MatrixPromote< + typename Mat1_T::temporary_type, + typename MatrixPromote< + typename Mat2_T::temporary_type, + typename MatrixPromote< + typename Mat3_T::temporary_type, + typename Mat4_T::temporary_type + >::temporary_type + >::temporary_type + >::temporary_type temporary_type; + typedef typename temporary_type::value_type value_type; +}; + +/** Type promotion for a matrix and a scalar. */ +template +struct MatrixPromote, S> +{ + /* The deduced matrix result type (the array type is the same): */ + typedef cml::matrix::type, AT, BO, L> type; + + /* The deduced temporary type: */ + typedef typename type::temporary_type temporary_type; +}; + +/** Type promotion for a scalar and a matrix. */ +template +struct MatrixPromote > +{ + /* The deduced matrix result type (the array type is the same): */ + typedef cml::matrix::type, AT, BO, L> type; + + /* The deduced temporary type: */ + typedef typename type::temporary_type temporary_type; +}; + +/** Type promotion for outer product. */ +template +struct MatrixPromote< cml::vector, cml::vector > +{ + typedef cml::vector left_type; + typedef cml::vector right_type; + typedef CML_DEFAULT_BASIS_ORIENTATION basis_orient; + + /* Get matrix size: */ + enum { + array_rows = left_type::array_size, + array_cols = right_type::array_size + }; + + /* Deduce the corresponding matrix types for the vectors: */ + typedef CML_DEFAULT_ARRAY_LAYOUT layout; + typedef typename select_if< + array_rows == -1, dynamic<>, fixed + >::result left_storage; + typedef cml::matrix left_matrix; + + typedef typename select_if< + array_cols == -1, dynamic<>, fixed<1,array_cols> + >::result right_storage; + typedef cml::matrix right_matrix; + + /* Finally, promote the matrix types to get the result: */ + typedef typename et::MatrixPromote::type type; + typedef typename type::temporary_type temporary_type; +}; + +} // namespace et +} // namespace cml + +#endif + +// ------------------------------------------------------------------------- +// vim:ft=cpp