1899 lines
65 KiB
C++
1899 lines
65 KiB
C++
//Copyright (c) 2008-2017 Emil Dotchevski and Reverge Studios, Inc.
|
|
|
|
//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)
|
|
|
|
#include "boost/throw_exception.hpp"
|
|
#include "boost/exception/info.hpp"
|
|
#include "boost/exception/diagnostic_information.hpp"
|
|
#include "boost/bind.hpp"
|
|
#include <string>
|
|
#include <map>
|
|
#include <vector>
|
|
#include <iostream>
|
|
#include <fstream>
|
|
#include <sstream>
|
|
#include <set>
|
|
#include <algorithm>
|
|
#include <limits>
|
|
#include <assert.h>
|
|
|
|
#define NL "\n"
|
|
#define TAB " "
|
|
#define TAB1 TAB
|
|
#define TAB2 TAB TAB
|
|
#define INCLUDE_MAT_ASSIGN "boost/qvm/gen/mat_assign%d.hpp"
|
|
#define INCLUDE_VEC_ASSIGN "boost/qvm/gen/vec_assign%d.hpp"
|
|
#define INCLUDE_STATIC_ASSERT "boost/qvm/static_assert.hpp"
|
|
#define INCLUDE_MATH "boost/qvm/math.hpp"
|
|
#define INCLUDE_THROW_EXCEPTION "boost/qvm/throw_exception.hpp"
|
|
#define INCLUDE_ERROR "boost/qvm/error.hpp"
|
|
#define INCLUDE_INLINE "boost/qvm/inline.hpp"
|
|
#define INCLUDE_M_TRAITS "boost/qvm/mat_traits.hpp"
|
|
#define INCLUDE_V_TRAITS "boost/qvm/vec_traits.hpp"
|
|
#define INCLUDE_Q_TRAITS "boost/qvm/quat_traits.hpp"
|
|
#define INCLUDE_S_TRAITS "boost/qvm/scalar_traits.hpp"
|
|
#define INCLUDE_DEDUCE_M "boost/qvm/deduce_mat.hpp"
|
|
#define INCLUDE_DEDUCE_V "boost/qvm/deduce_vec.hpp"
|
|
#define INCLUDE_DEDUCE_Q "boost/qvm/deduce_quat.hpp"
|
|
#define INCLUDE_DEDUCE_S "boost/qvm/deduce_scalar.hpp"
|
|
#define INCLUDE_SWIZZLE_TRAITS "boost/qvm/detail/swizzle_traits.hpp"
|
|
#define INCLUDE_ENABLE_IF "boost/qvm/enable_if.hpp"
|
|
#define INCLUDE_ASSERT "boost/qvm/assert.hpp"
|
|
|
|
namespace
|
|
{
|
|
struct exception_base: virtual std::exception, virtual boost::exception { };
|
|
struct bad_command_line: virtual exception_base { };
|
|
typedef boost::error_info<struct cmd_arg_,std::string> cmd_arg;
|
|
|
|
struct
|
|
null_deleter
|
|
{
|
|
template <class T>
|
|
void
|
|
operator()( T * ) const
|
|
{
|
|
}
|
|
};
|
|
|
|
std::string
|
|
get_include_guard( std::string file_name )
|
|
{
|
|
std::transform(file_name.begin(), file_name.end(),file_name.begin(), ::toupper);
|
|
std::replace(file_name.begin(), file_name.end(), '.', '_');
|
|
return "BOOST_QVM_GEN_" + file_name + "_INCLUDED";
|
|
}
|
|
|
|
template <class T>
|
|
std::string
|
|
to_string( T const & x )
|
|
{
|
|
std::ostringstream s;
|
|
s<<x;
|
|
return s.str();
|
|
}
|
|
|
|
struct
|
|
command_line_options
|
|
{
|
|
bool con;
|
|
std::string output_directory;
|
|
|
|
command_line_options():
|
|
con(false)
|
|
{
|
|
}
|
|
};
|
|
|
|
class
|
|
output_file
|
|
{
|
|
output_file( output_file const & );
|
|
output_file & operator=( output_file const & );
|
|
|
|
std::string const output_directory;
|
|
bool const con;
|
|
std::ostringstream out_;
|
|
std::set<std::string> includes_;
|
|
|
|
public:
|
|
|
|
explicit
|
|
output_file( command_line_options const & opt ):
|
|
output_directory(opt.output_directory),
|
|
con(opt.con)
|
|
{
|
|
}
|
|
|
|
void
|
|
require_include( std::string const & fn )
|
|
{
|
|
assert(!strchr(fn.c_str(),'%'));
|
|
includes_.insert(fn);
|
|
};
|
|
|
|
std::ostream &
|
|
stream()
|
|
{
|
|
return out_;
|
|
}
|
|
|
|
void
|
|
dump( std::string const & name ) const
|
|
{
|
|
std::ostream * out = &std::cout;
|
|
boost::shared_ptr<std::ofstream> f;
|
|
if( !con )
|
|
{
|
|
std::string path;
|
|
if( !output_directory.empty() )
|
|
{
|
|
path+=output_directory;
|
|
path+='/';
|
|
path+=name;
|
|
}
|
|
boost::shared_ptr<std::ofstream>(new std::ofstream(path.c_str())).swap(f);
|
|
out = f.get();
|
|
std::cout << "Writing " << path << "..." << std::endl;
|
|
}
|
|
out->exceptions(std::ofstream::eofbit|std::ofstream::failbit|std::ofstream::badbit);
|
|
std::string include_guard=get_include_guard(name);
|
|
*out <<
|
|
"#ifndef " << include_guard << NL
|
|
"#define " << include_guard << NL
|
|
NL
|
|
"/// Copyright (c) 2008-2021 Emil Dotchevski and Reverge Studios, Inc." NL
|
|
NL
|
|
"/// Distributed under the Boost Software License, Version 1.0. (See accompanying" NL
|
|
"/// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)" NL
|
|
NL
|
|
"/// This file was generated by a program. Do not edit manually." NL
|
|
NL
|
|
;
|
|
for( std::set<std::string>::const_iterator i=includes_.begin(),e=includes_.end(); i!=e; ++i )
|
|
*out << "#include <" << *i << ">" NL;
|
|
*out <<
|
|
NL
|
|
"namespace boost { namespace qvm {" NL
|
|
NL
|
|
<<out_.str() <<
|
|
"} }" NL
|
|
NL
|
|
"#endif" NL
|
|
;
|
|
}
|
|
};
|
|
|
|
void
|
|
replace( std::string & s, char const * substr, char const * newstr )
|
|
{
|
|
assert(substr && *substr);
|
|
assert(newstr && *newstr);
|
|
std::string::size_type f=s.find(substr);
|
|
if( s.npos!=f )
|
|
s.replace(f,f+strlen(substr),newstr);
|
|
}
|
|
|
|
std::string
|
|
deduce_name( std::string const & fn, char const * suffix )
|
|
{
|
|
std::string s=fn;
|
|
replace(s,"operator==","eq");
|
|
replace(s,"operator!=","neq");
|
|
replace(s,"operator+=","plus_eq");
|
|
replace(s,"operator-=","minus_eq");
|
|
replace(s,"operator*=","mul_eq");
|
|
replace(s,"operator/=","div_eq");
|
|
replace(s,"operator+","plus");
|
|
replace(s,"operator-","minus");
|
|
replace(s,"operator*","mul");
|
|
replace(s,"operator/","div");
|
|
if( suffix )
|
|
{
|
|
s += '_';
|
|
s += suffix;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
void
|
|
header_mr_ma_mb_same_size( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.stream() <<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<B>::rows=="<<r<<" &&" NL
|
|
TAB1 "mat_traits<A>::cols=="<<c<<" && mat_traits<B>::cols=="<<c<<"," NL
|
|
TAB1 "deduce_mat2<A,B,"<<r<<','<<c<<"> >::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_mr_ma_mb_mult( output_file & out, int m, int n, int p, std::string const & name )
|
|
{
|
|
assert(m>0);
|
|
assert(n>0);
|
|
assert(p>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<m<<" && mat_traits<B>::rows=="<<n<<" &&" NL
|
|
TAB1 "mat_traits<A>::cols=="<<n<<" && mat_traits<B>::cols=="<<p<<"," NL
|
|
TAB1 "deduce_mat2<A,B,"<<m<<','<<p<<"> >::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_ma_vb_mult( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<A>::cols=="<<c<<" &&" NL
|
|
TAB1 "vec_traits<B>::dim=="<<c<<"," NL
|
|
TAB1 "deduce_vec2<A,B,"<<c<<"> >::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_va_mb_mult( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<B>::rows=="<<r<<" && mat_traits<B>::cols=="<<c<<" &&" NL
|
|
TAB1 "vec_traits<A>::dim=="<<c<<"," NL
|
|
TAB1 "deduce_vec2<A,B,"<<r<<"> >::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_va_vb_same_size( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<" && vec_traits<B>::dim=="<<d<<"," NL
|
|
TAB1 "deduce_vec2<A,B,"<<d<<"> >::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_bool_ma_mb_same_size( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<B>::rows=="<<r<<" &&" NL
|
|
TAB1 "mat_traits<A>::cols=="<<c<<" && mat_traits<B>::cols=="<<c<<"," NL
|
|
TAB1 "bool>::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_bool_va_vb_same_size( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<" && vec_traits<B>::dim=="<<d<<"," NL
|
|
"bool>::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_ma_mb_same_size( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<B>::rows=="<<r<<" &&" NL
|
|
TAB1 "mat_traits<A>::cols=="<<c<<" && mat_traits<B>::cols=="<<c<<"," NL
|
|
TAB1 "A &>::type" NL
|
|
<<name<<"( A & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_va_vb_same_size( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<" && vec_traits<B>::dim=="<<d<<"," NL
|
|
TAB1 "A &>::type" NL
|
|
<<name<<"( A & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_sr_ma( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<A>::cols=="<<c<<"," NL
|
|
TAB1 "typename mat_traits<A>::scalar_type>::type" NL
|
|
<<name<<"( A const & a )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_sr_va_vb( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_S);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<" && vec_traits<B>::dim=="<<d<<"," NL
|
|
TAB1 "deduce_scalar<typename vec_traits<A>::scalar_type,typename vec_traits<B>::scalar_type> >::type" NL
|
|
<<name<<"( A const & a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_sr_va( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "is_vec<A>::value && vec_traits<A>::dim=="<<d<<"," NL
|
|
TAB1 "typename vec_traits<A>::scalar_type>::type" NL
|
|
<<name<<"( A const & a )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_mr_ma( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.stream()<<
|
|
"template <class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<A>::cols=="<<c<<"," NL
|
|
TAB1 "deduce_mat<A> >::type" NL
|
|
<<name<<"( A const & a )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_va( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
out.stream()<<
|
|
"template <class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<"," NL
|
|
TAB1 "deduce_vec<A> >::type" NL
|
|
<<name<<"( A const & a )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_va_same_size( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class R,class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "is_vec<A>::value &&" NL
|
|
TAB1 "vec_traits<R>::dim=="<<d<<" && vec_traits<A>::dim=="<<d<<"," NL
|
|
TAB1 "R>::type" NL
|
|
<<name<<"( A const & a )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_mr_ma_sb( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<A>::cols=="<<c<<" && is_scalar<B>::value," NL
|
|
TAB1 "deduce_mat2<A,B,mat_traits<A>::rows,mat_traits<A>::cols> >::type" NL
|
|
<<name<<"( A const & a, B b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_mr_sa_mb( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "is_scalar<A>::value && mat_traits<B>::rows=="<<r<<" && mat_traits<B>::cols=="<<c<<"," NL
|
|
TAB1 "deduce_mat2<A,B,mat_traits<B>::rows,mat_traits<B>::cols> >::type" NL
|
|
<<name<<"( A a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_va_sb( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<" && is_scalar<B>::value," NL
|
|
TAB1 "deduce_vec2<A,B,vec_traits<A>::dim> >::type" NL
|
|
<<name<<"( A const & a, B b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_vr_sa_vb( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "is_scalar<A>::value && vec_traits<B>::dim=="<<d<<"," NL
|
|
TAB1 "deduce_vec2<A,B,vec_traits<B>::dim> >::type" NL
|
|
<<name<<"( A a, B const & b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_ma_sb( output_file & out, int r, int c, std::string const & name )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<r<<" && mat_traits<A>::cols=="<<c<<" && is_scalar<B>::value," NL
|
|
TAB1 "A &>::type" NL
|
|
<<name<<"( A & a, B b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
header_va_sb( output_file & out, int d, std::string const & name )
|
|
{
|
|
assert(d>0);
|
|
assert(!name.empty());
|
|
out.stream()<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<" && is_scalar<B>::value," NL
|
|
TAB1 "A &>::type" NL
|
|
<<name<<"( A & a, B b )" NL
|
|
;
|
|
}
|
|
|
|
void
|
|
defined( std::ostream & g, int r, int cr, int c, std::string fn, char const * suffix )
|
|
{
|
|
assert(r>0);
|
|
assert(cr>0);
|
|
assert(c>0);
|
|
assert(!fn.empty());
|
|
std::string dn=deduce_name(fn,suffix);
|
|
std::string name=dn+"_defined";
|
|
g<<
|
|
NL
|
|
"namespace" NL
|
|
"sfinae" NL
|
|
TAB1 "{" NL
|
|
TAB1 "using ::boost::qvm::"<<fn<<";" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
"namespace" NL
|
|
"qvm_detail" NL
|
|
TAB1 "{" NL
|
|
TAB1 "template <int R,int /*CR*/,int C>" NL
|
|
TAB1 "struct "<<name<<";" NL
|
|
NL
|
|
TAB1 "template <>" NL
|
|
TAB1 "struct" NL
|
|
TAB1<<name<<'<'<<r<<','<<cr<<','<<c<<">" NL
|
|
TAB2"{" NL
|
|
TAB2"static bool const value=true;" NL
|
|
TAB2"};" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
;
|
|
}
|
|
|
|
void
|
|
defined( std::ostream & g, int r, int c, std::string const & fn, char const * suffix )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!fn.empty());
|
|
std::string dn=deduce_name(fn,suffix);
|
|
std::string name=dn+"_defined";
|
|
g<<
|
|
NL
|
|
"namespace" NL
|
|
"sfinae" NL
|
|
TAB1 "{" NL
|
|
TAB1 "using ::boost::qvm::"<<fn<<";" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
"namespace" NL
|
|
"qvm_detail" NL
|
|
TAB1 "{" NL
|
|
TAB1 "template <int R,int C>" NL
|
|
TAB1 "struct "<<name<<";" NL
|
|
NL
|
|
TAB1 "template <>" NL
|
|
TAB1 "struct" NL
|
|
TAB1<<name<<"<"<<r<<","<<c<<">" NL
|
|
TAB2"{" NL
|
|
TAB2"static bool const value=true;" NL
|
|
TAB2"};" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
;
|
|
}
|
|
|
|
void
|
|
defined( std::ostream & g, int d, std::string const & fn, char const * suffix )
|
|
{
|
|
assert(d>0);
|
|
assert(!fn.empty());
|
|
std::string dn=deduce_name(fn,suffix);
|
|
std::string name=dn+"_defined";
|
|
g<<
|
|
NL
|
|
"namespace" NL
|
|
"sfinae" NL
|
|
TAB1 "{" NL
|
|
TAB1 "using ::boost::qvm::"<<fn<<";" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
"namespace" NL
|
|
"qvm_detail" NL
|
|
TAB1 "{" NL
|
|
TAB1 "template <int D>" NL
|
|
TAB1 "struct "<<name<<";" NL
|
|
NL
|
|
TAB1 "template <>" NL
|
|
TAB1 "struct" NL
|
|
TAB1<<name<<"<"<<d<<">" NL
|
|
TAB2"{" NL
|
|
TAB2"static bool const value=true;" NL
|
|
TAB2"};" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
;
|
|
}
|
|
|
|
void
|
|
mr_mult_ma_mb( output_file & out, int m, int n, int p, char const * suffix )
|
|
{
|
|
assert(m>0);
|
|
assert(n>0);
|
|
assert(p>0);
|
|
header_mr_ma_mb_mult(out,m,n,p,"operator*");
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<A>::scalar_type Ta;" NL
|
|
TAB1 "typedef typename mat_traits<B>::scalar_type Tb;" NL
|
|
;
|
|
for( int i=0; i!=m; ++i )
|
|
for( int j=0; j!=n; ++j )
|
|
g<<TAB1 "Ta const a"<<i<<j<<" = mat_traits<A>::template read_element<"<<i<<','<<j<<">(a);" NL;
|
|
for( int i=0; i!=n; ++i )
|
|
for( int j=0; j!=p; ++j )
|
|
g<<TAB1 "Tb const b"<<i<<j<<" = mat_traits<B>::template read_element<"<<i<<','<<j<<">(b);" NL;
|
|
g<<
|
|
TAB1 "typedef typename deduce_mat2<A,B,"<<m<<','<<p<<">::type R;" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(mat_traits<R>::rows=="<<m<<");" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(mat_traits<R>::cols=="<<p<<");" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=m; ++i )
|
|
for( int j=0; j!=p; ++j )
|
|
{
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r)=";
|
|
for( int k=0; k!=n; ++k )
|
|
{
|
|
if( k )
|
|
g<<'+';
|
|
g<<'a'<<i<<k<<"*b"<<k<<j;
|
|
}
|
|
g<<";" NL;
|
|
}
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,m,n,p,"operator*",suffix);
|
|
}
|
|
|
|
void
|
|
ma_mult_ma_mb( output_file & out, int d, char const * suffix )
|
|
{
|
|
assert(d>0);
|
|
header_ma_mb_same_size(out,d,d,"operator*=");
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<A>::scalar_type Ta;" NL
|
|
TAB1 "typedef typename mat_traits<B>::scalar_type Tb;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
for( int j=0; j!=d; ++j )
|
|
g<<TAB1 "Ta const a"<<i<<j<<" = mat_traits<A>::template read_element<"<<i<<','<<j<<">(a);" NL;
|
|
for( int i=0; i!=d; ++i )
|
|
for( int j=0; j!=d; ++j )
|
|
g<<TAB1 "Tb const b"<<i<<j<<" = mat_traits<B>::template read_element<"<<i<<','<<j<<">(b);" NL;
|
|
for( int i=0; i!=d; ++i )
|
|
for( int j=0; j!=d; ++j )
|
|
{
|
|
g<<TAB1 "mat_traits<A>::template write_element<"<<i<<","<<j<<">(a)=";
|
|
for( int k=0; k!=d; ++k )
|
|
{
|
|
if( k )
|
|
g<<'+';
|
|
g<<'a'<<i<<k<<"*b"<<k<<j;
|
|
}
|
|
g<<";" NL;
|
|
}
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"operator*=",suffix);
|
|
}
|
|
|
|
void
|
|
vr_mult_ma_vb( output_file & out, int r, int c, char const * suffix )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
header_vr_ma_vb_mult(out,r,c,"operator*");
|
|
out.require_include(INCLUDE_INLINE);
|
|
out.require_include(INCLUDE_V_TRAITS);
|
|
out.require_include(INCLUDE_M_TRAITS);
|
|
out.require_include(INCLUDE_ENABLE_IF);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<A>::scalar_type Ta;" NL
|
|
TAB1 "typedef typename vec_traits<B>::scalar_type Tb;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "Ta const a"<<i<<j<<" = mat_traits<A>::template read_element<"<<i<<','<<j<<">(a);" NL;
|
|
for( int i=0; i!=c; ++i )
|
|
g<<TAB1 "Tb const b"<<i<<" = vec_traits<B>::template read_element<"<<i<<">(b);" NL;
|
|
g<<
|
|
TAB1 "typedef typename deduce_vec2<A,B,"<<c<<">::type R;" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(vec_traits<R>::dim=="<<c<<");" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
{
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=";
|
|
for( int j=0; j!=c; ++j )
|
|
{
|
|
if( j )
|
|
g<<'+';
|
|
g<<'a'<<i<<j<<"*b"<<j;
|
|
}
|
|
g<<";" NL;
|
|
}
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,"operator*",suffix);
|
|
}
|
|
|
|
void
|
|
vr_mult_va_mb( output_file & out, int r, int c, char const * suffix )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
header_vr_va_mb_mult(out,r,c,"operator*");
|
|
out.require_include(INCLUDE_INLINE);
|
|
out.require_include(INCLUDE_V_TRAITS);
|
|
out.require_include(INCLUDE_M_TRAITS);
|
|
out.require_include(INCLUDE_ENABLE_IF);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename vec_traits<A>::scalar_type Ta;" NL
|
|
TAB1 "typedef typename mat_traits<B>::scalar_type Tb;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
g<<TAB1 "Ta const a"<<i<<" = vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "Tb const b"<<i<<j<<" = mat_traits<B>::template read_element<"<<i<<','<<j<<">(b);" NL;
|
|
g<<
|
|
TAB1 "typedef typename deduce_vec2<A,B,"<<r<<">::type R;" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(vec_traits<R>::dim=="<<r<<");" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=c; ++i )
|
|
{
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=";
|
|
for( int j=0; j!=r; ++j )
|
|
{
|
|
if( j )
|
|
g<<'+';
|
|
g<<'a'<<j<<"*b"<<j<<i;
|
|
}
|
|
g<<";" NL;
|
|
}
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,"operator*",suffix);
|
|
}
|
|
|
|
void
|
|
vr_op_va_vb_same_size( output_file & out, int d, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_vr_va_vb_same_size(out,d,fn);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_vec2<A,B,"<<d<<">::type R;" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(vec_traits<R>::dim=="<<d<<");" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=vec_traits<A>::template read_element<"<<i<<">(a)"<<op<<"vec_traits<B>::template read_element<"<<i<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,fn,suffix);
|
|
}
|
|
|
|
void
|
|
bool_eq_ma_mb( output_file & out, int r, int c, char const * suffix )
|
|
{
|
|
header_bool_ma_mb_same_size(out,r,c,"operator==");
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "return" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<
|
|
TAB2"mat_traits<A>::template read_element<"<<i<<','<<j<<">(a)==mat_traits<B>::template read_element<"<<i<<','<<j<<">(b)"<<(i!=r-1||j!=c-1?" &&":";")<<NL;
|
|
;
|
|
g<<
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,"operator==",suffix);
|
|
}
|
|
|
|
void
|
|
bool_eq_va_vb( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_bool_va_vb_same_size(out,d,"operator==");
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "return" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<
|
|
TAB2"vec_traits<A>::template read_element<"<<i<<">(a)==vec_traits<B>::template read_element<"<<i<<">(b)"<<(i!=d-1?" &&":";")<<NL;
|
|
;
|
|
g<<
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"operator==",suffix);
|
|
}
|
|
|
|
void
|
|
bool_neq_ma_mb( output_file & out, int r, int c, char const * suffix )
|
|
{
|
|
header_bool_ma_mb_same_size(out,r,c,"operator!=");
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "return" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<
|
|
TAB2"!(mat_traits<A>::template read_element<"<<i<<','<<j<<">(a)==mat_traits<B>::template read_element<"<<i<<','<<j<<">(b))"<<(i!=r-1||j!=c-1?" ||":";")<<NL;
|
|
;
|
|
g<<
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,"operator!=",suffix);
|
|
}
|
|
|
|
void
|
|
bool_neq_va_vb( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_bool_va_vb_same_size(out,d,"operator!=");
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "return" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<
|
|
TAB2"!(vec_traits<A>::template read_element<"<<i<<">(a)==vec_traits<B>::template read_element<"<<i<<">(b))"<<(i!=d-1?" ||":";")<<NL;
|
|
;
|
|
g<<
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"operator!=",suffix);
|
|
}
|
|
|
|
void
|
|
mr_op_ma_mb_same_size( output_file & out, int r, int c, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(r>0);
|
|
assert(c>0);
|
|
assert(!op.empty());
|
|
header_mr_ma_mb_same_size(out,r,c,fn);
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_mat2<A,B,"<<r<<','<<c<<">::type R;" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(mat_traits<R>::rows=="<<r<<");" NL
|
|
TAB1 "BOOST_QVM_STATIC_ASSERT(mat_traits<R>::cols=="<<c<<");" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r)=mat_traits<A>::template read_element<"<<i<<","<<j<<">(a)"<<op<<"mat_traits<B>::template read_element<"<<i<<","<<j<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,fn,suffix);
|
|
}
|
|
|
|
void
|
|
ma_op_ma_mb_same_size( output_file & out, int r, int c, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_ma_mb_same_size(out,r,c,fn);
|
|
std::ostream & g=out.stream();
|
|
g<<TAB1 "{" NL;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<A>::template write_element<"<<i<<","<<j<<">(a)"<<op<<"mat_traits<B>::template read_element<"<<i<<","<<j<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,fn,suffix);
|
|
}
|
|
|
|
void
|
|
va_op_va_vb_same_size( output_file & out, int d, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_va_vb_same_size(out,d,fn);
|
|
std::ostream & g=out.stream();
|
|
g<<TAB1 "{" NL;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<A>::template write_element<"<<i<<">(a)"<<op<<"vec_traits<B>::template read_element<"<<i<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,fn,suffix);
|
|
}
|
|
|
|
void
|
|
mr_op_ma( output_file & out, int r, int c, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_mr_ma(out,r,c,fn);
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_mat<A>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r)="<<op<<"mat_traits<A>::template read_element<"<<i<<","<<j<<">(a);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,fn,suffix);
|
|
}
|
|
|
|
void
|
|
vr_op_va( output_file & out, int d, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_vr_va(out,d,fn);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_vec<A>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)="<<op<<"vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,fn,suffix);
|
|
}
|
|
|
|
void
|
|
mr_op_ma_sb( output_file & out, int r, int c, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_mr_ma_sb(out,r,c,fn);
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_mat2<A,B,mat_traits<A>::rows,mat_traits<A>::cols>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r)=mat_traits<A>::template read_element<"<<i<<","<<j<<">(a)"<<op<<"b;" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,fn,suffix);
|
|
}
|
|
|
|
void
|
|
mr_op_sa_mb( output_file & out, int r, int c, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_mr_sa_mb(out,r,c,fn);
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_mat2<A,B,mat_traits<B>::rows,mat_traits<B>::cols>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r)=a"<<op<<"mat_traits<B>::template read_element<"<<i<<","<<j<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,fn,suffix);
|
|
}
|
|
|
|
void
|
|
vr_op_va_sb( output_file & out, int d, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_vr_va_sb(out,d,fn);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_vec2<A,B,vec_traits<A>::dim>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=vec_traits<A>::template read_element<"<<i<<">(a)"<<op<<"b;" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,fn,suffix);
|
|
}
|
|
|
|
void
|
|
vr_op_sa_vb( output_file & out, int d, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_vr_sa_vb(out,d,fn);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename deduce_vec2<A,B,vec_traits<B>::dim>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=a"<<op<<"vec_traits<B>::template read_element<"<<i<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,fn,suffix);
|
|
}
|
|
|
|
void
|
|
ma_op_ma_sb( output_file & out, int r, int c, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_ma_sb(out,r,c,fn);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<A>::template write_element<"<<i<<","<<j<<">(a)"<<op<<"b;" NL;
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,fn,suffix);
|
|
}
|
|
|
|
void
|
|
va_op_va_sb( output_file & out, int d, std::string const & fn, std::string const & op, char const * suffix )
|
|
{
|
|
assert(!op.empty());
|
|
header_va_sb(out,d,fn);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<A>::template write_element<"<<i<<">(a)"<<op<<"b;" NL;
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,fn,suffix);
|
|
}
|
|
|
|
void
|
|
ma_assign_ma_mb( output_file & out, int r, int c, char const * suffix )
|
|
{
|
|
header_ma_mb_same_size(out,r,c,"assign");
|
|
out.require_include(INCLUDE_M_TRAITS);
|
|
out.require_include(INCLUDE_INLINE);
|
|
out.require_include(INCLUDE_ENABLE_IF);
|
|
std::ostream & g=out.stream();
|
|
g<<TAB1 "{" NL;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<A>::template write_element<"<<i<<","<<j<<">(a)=mat_traits<B>::template read_element<"<<i<<","<<j<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,r,c,"assign",suffix);
|
|
}
|
|
|
|
void
|
|
va_assign_va_vb( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_va_vb_same_size(out,d,"assign");
|
|
out.require_include(INCLUDE_V_TRAITS);
|
|
out.require_include(INCLUDE_INLINE);
|
|
out.require_include(INCLUDE_ENABLE_IF);
|
|
std::ostream & g=out.stream();
|
|
g<<TAB1 "{" NL;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<A>::template write_element<"<<i<<">(a)=vec_traits<B>::template read_element<"<<i<<">(b);" NL;
|
|
g<<
|
|
TAB1 "return a;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"assign",suffix);
|
|
}
|
|
|
|
void
|
|
mr_convert_to_ma( output_file & out, int r, int c, char const * suffix )
|
|
{
|
|
if( r==c && r>=3 )
|
|
{
|
|
out.require_include(INCLUDE_Q_TRAITS);
|
|
out.require_include(INCLUDE_S_TRAITS);
|
|
}
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
"template <class R,class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "mat_traits<R>::rows=="<<r<<" && mat_traits<A>::rows=="<<r<<" &&" NL
|
|
TAB1 "mat_traits<R>::cols=="<<c<<" && mat_traits<A>::cols=="<<c<<"," NL
|
|
TAB1 "R>::type" NL
|
|
<<"convert_to( A const & a )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r) = mat_traits<A>::template read_element<"<<i<<","<<j<<">(a);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
if( r==c && r>=3 )
|
|
{
|
|
g<<
|
|
NL
|
|
"template <class R,class A>" NL
|
|
"BOOST_QVM_INLINE" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "is_mat<R>::value && is_quat<A>::value &&" NL
|
|
TAB1 "mat_traits<R>::rows=="<<r<<" && mat_traits<R>::cols=="<<c<<"," NL
|
|
TAB1 "R>::type" NL
|
|
"convert_to( A const & q )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<R>::scalar_type T;" NL
|
|
TAB1 "T const a=quat_traits<A>::template read_element<0>(q);" NL
|
|
TAB1 "T const b=quat_traits<A>::template read_element<1>(q);" NL
|
|
TAB1 "T const c=quat_traits<A>::template read_element<2>(q);" NL
|
|
TAB1 "T const d=quat_traits<A>::template read_element<3>(q);" NL
|
|
TAB1 "T const bb = b*b;" NL
|
|
TAB1 "T const cc = c*c;" NL
|
|
TAB1 "T const dd = d*d;" NL
|
|
TAB1 "T const bc = b*c;" NL
|
|
TAB1 "T const bd = b*d;" NL
|
|
TAB1 "T const cd = c*d;" NL
|
|
TAB1 "T const ab = a*b;" NL
|
|
TAB1 "T const ac = a*c;" NL
|
|
TAB1 "T const ad = a*d;" NL<<
|
|
(r>3?TAB1 "T const zero = scalar_traits<T>::value(0);" NL:"")<<
|
|
TAB1 "T const one = scalar_traits<T>::value(1);" NL
|
|
TAB1 "T const two = one+one;" NL
|
|
TAB1 "R r;" NL
|
|
TAB1 "mat_traits<R>::template write_element<0,0>(r) = one - two*(cc+dd);" NL
|
|
TAB1 "mat_traits<R>::template write_element<0,1>(r) = two*(bc-ad);" NL
|
|
TAB1 "mat_traits<R>::template write_element<0,2>(r) = two*(bd+ac);" NL
|
|
;
|
|
for( int i=3; i!=c; ++i )
|
|
g<<TAB1 "mat_traits<R>::template write_element<0,"<<i<<">(r) = zero;" NL;
|
|
g<<
|
|
TAB1 "mat_traits<R>::template write_element<1,0>(r) = two*(bc+ad);" NL
|
|
TAB1 "mat_traits<R>::template write_element<1,1>(r) = one - two*(bb+dd);" NL
|
|
TAB1 "mat_traits<R>::template write_element<1,2>(r) = two*(cd-ab);" NL
|
|
;
|
|
for( int i=3; i!=c; ++i )
|
|
g<<TAB1 "mat_traits<R>::template write_element<1,"<<i<<">(r) = zero;" NL;
|
|
g<<
|
|
TAB1 "mat_traits<R>::template write_element<2,0>(r) = two*(bd-ac);" NL
|
|
TAB1 "mat_traits<R>::template write_element<2,1>(r) = two*(cd+ab);" NL
|
|
TAB1 "mat_traits<R>::template write_element<2,2>(r) = one - two*(bb+cc);" NL
|
|
;
|
|
for( int i=3; i!=c; ++i )
|
|
g<<TAB1 "mat_traits<R>::template write_element<2,"<<i<<">(r) = zero;" NL;
|
|
for( int i=3; i!=r; ++i )
|
|
for( int j=0; j!=c; ++j )
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<","<<j<<">(r) = "<<(i==j?"one":"zero")<<";" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
}
|
|
defined(g,r,c,"convert_to",suffix);
|
|
}
|
|
|
|
void
|
|
vr_convert_to_va( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_vr_va_same_size(out,d,"convert_to");
|
|
std::ostream & g=out.stream();
|
|
g<<TAB1 "{" NL<<
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"convert_to",suffix);
|
|
}
|
|
|
|
struct
|
|
del_row_col
|
|
{
|
|
del_row_col const * next;
|
|
int i, j;
|
|
char var;
|
|
explicit
|
|
del_row_col( char var ):
|
|
next(0),
|
|
i(std::numeric_limits<int>::max()),
|
|
j(std::numeric_limits<int>::max()),
|
|
var(var)
|
|
{
|
|
}
|
|
del_row_col( del_row_col const & next, int i, int j ):
|
|
next(&next),
|
|
i(i),
|
|
j(j),
|
|
var(next.var)
|
|
{
|
|
}
|
|
std::pair<int,int>
|
|
idx( std::pair<int,int> const & x ) const
|
|
{
|
|
std::pair<int,int> r(x.first+(x.first>=i),x.second+(x.second>=j));
|
|
if( next )
|
|
return next->idx(r);
|
|
else
|
|
return r;
|
|
|
|
}
|
|
void
|
|
operator()( std::ostream & g, int r, int c ) const
|
|
{
|
|
std::pair<int,int> p=idx(std::make_pair(r,c));
|
|
g << var << p.first << p.second;
|
|
}
|
|
};
|
|
|
|
void
|
|
determinant_impl( std::ostream & g, int n, del_row_col const & a )
|
|
{
|
|
if( n==1 )
|
|
return a(g,0,0);
|
|
g << "(";
|
|
char const * plus="";
|
|
for( int i=0; i!=n; ++i,plus="+" )
|
|
{
|
|
g<<((i&1)?"-":plus);
|
|
a(g,0,i);
|
|
g<<'*';
|
|
determinant_impl(g,n-1,del_row_col(a,0,i));
|
|
}
|
|
g << ")";
|
|
}
|
|
|
|
void
|
|
determinant( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_sr_ma(out,d,d,"determinant");
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<A>::scalar_type T;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
for( int j=0; j!=d; ++j )
|
|
g<<TAB1<<"T const a"<<i<<j<<"=mat_traits<A>::template read_element<"<<i<<','<<j<<">(a);" NL;
|
|
g<<TAB1 "T det=";
|
|
determinant_impl(g,d,del_row_col('a'));
|
|
g<<";" NL;
|
|
g<<
|
|
TAB1 "return det;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"determinant",suffix);
|
|
}
|
|
|
|
void
|
|
inverse_ma( output_file & out, int d, char const * suffix )
|
|
{
|
|
assert(d>1);
|
|
out.require_include(INCLUDE_DEDUCE_M);
|
|
out.require_include(INCLUDE_ASSERT);
|
|
out.require_include(INCLUDE_THROW_EXCEPTION);
|
|
out.require_include(INCLUDE_ERROR);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
"template <class A,class B>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<d<<" && mat_traits<A>::cols=="<<d<<" && is_scalar<B>::value," NL
|
|
TAB1 "deduce_mat2<A,B,mat_traits<A>::rows,mat_traits<A>::cols> >::type" NL
|
|
"inverse( A const & a, B det )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<A>::scalar_type T;" NL
|
|
TAB1 "BOOST_QVM_ASSERT(det!=scalar_traits<B>::value(0));" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
for( int j=0; j!=d; ++j )
|
|
g<<TAB1 "T const a"<<i<<j<<"=mat_traits<A>::template read_element<"<<i<<','<<j<<">(a);" NL;
|
|
g<<
|
|
TAB1 "T const f=scalar_traits<T>::value(1)/det;" NL
|
|
TAB1 "typedef typename deduce_mat2<A,B,mat_traits<A>::rows,mat_traits<A>::cols>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
for( int j=0; j!=d; ++j )
|
|
{
|
|
g<<TAB1 "mat_traits<R>::template write_element<"<<i<<','<<j<<">(r)="<<(((i+j)&1)?'-':' ')<<"f*";
|
|
determinant_impl(g,d-1,del_row_col(del_row_col('a'),j,i));
|
|
g<<";" NL;
|
|
}
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
"template <class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename lazy_enable_if_c<" NL
|
|
TAB1 "mat_traits<A>::rows=="<<d<<" && mat_traits<A>::cols=="<<d<<"," NL
|
|
TAB1 "deduce_mat<A> >::type" NL
|
|
"inverse( A const & a )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename mat_traits<A>::scalar_type T;" NL
|
|
TAB1 "T det=determinant(a);" NL
|
|
TAB1 "if( det==scalar_traits<T>::value(0) )" NL
|
|
TAB2"BOOST_QVM_THROW_EXCEPTION(zero_determinant_error());" NL
|
|
TAB1 "return inverse(a,det);" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"inverse",suffix);
|
|
}
|
|
|
|
void
|
|
mag_sqr( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_sr_va(out,d,"mag_sqr");
|
|
out.require_include(INCLUDE_MATH);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename vec_traits<A>::scalar_type T;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "T const a"<<i<<"=vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
g<<TAB1 "T const m2=";
|
|
for( int i=0; i!=d; ++i )
|
|
{
|
|
if( i )
|
|
g<<'+';
|
|
g<<'a'<<i<<"*a"<<i;
|
|
}
|
|
g<<
|
|
";" NL
|
|
TAB1 "return m2;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"mag_sqr",suffix);
|
|
}
|
|
|
|
void
|
|
mag( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_sr_va(out,d,"mag");
|
|
out.require_include(INCLUDE_MATH);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename vec_traits<A>::scalar_type T;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "T const a"<<i<<"=vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
g<<TAB1 "T const m2=";
|
|
for( int i=0; i!=d; ++i )
|
|
{
|
|
if( i )
|
|
g<<'+';
|
|
g<<'a'<<i<<"*a"<<i;
|
|
}
|
|
g<<
|
|
";" NL
|
|
TAB1 "T const mag=sqrt<T>(m2);" NL
|
|
TAB1 "return mag;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"mag",suffix);
|
|
}
|
|
|
|
void
|
|
normalize( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_vr_va(out,d,"normalized");
|
|
out.require_include(INCLUDE_MATH);
|
|
out.require_include(INCLUDE_THROW_EXCEPTION);
|
|
out.require_include(INCLUDE_ERROR);
|
|
out.require_include(INCLUDE_DEDUCE_V);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename vec_traits<A>::scalar_type T;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "T const a"<<i<<"=vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
g<<TAB1 "T const m2=";
|
|
for( int i=0; i!=d; ++i )
|
|
{
|
|
if( i )
|
|
g<<'+';
|
|
g<<'a'<<i<<"*a"<<i;
|
|
}
|
|
g<<
|
|
";" NL
|
|
TAB1 "if( m2==scalar_traits<typename vec_traits<A>::scalar_type>::value(0) )" NL
|
|
TAB2"BOOST_QVM_THROW_EXCEPTION(zero_magnitude_error());" NL
|
|
TAB1 "T const rm=scalar_traits<T>::value(1)/sqrt<T>(m2);" NL
|
|
TAB1 "typedef typename deduce_vec<A>::type R;" NL
|
|
TAB1 "R r;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<R>::template write_element<"<<i<<">(r)=a"<<i<<"*rm;" NL;
|
|
g<<
|
|
TAB1 "return r;" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
"namespace" NL
|
|
"sfinae" NL
|
|
TAB1 "{" NL
|
|
TAB1 "using ::boost::qvm::normalized;" NL
|
|
TAB1 "}" NL
|
|
NL
|
|
"template <class A>" NL
|
|
"BOOST_QVM_INLINE_OPERATIONS" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "vec_traits<A>::dim=="<<d<<"," NL
|
|
TAB1 "void>::type" NL
|
|
<<"normalize( A & a )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename vec_traits<A>::scalar_type T;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "T const a"<<i<<"=vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
g<<TAB1 "T const m2=";
|
|
for( int i=0; i!=d; ++i )
|
|
{
|
|
if( i )
|
|
g<<'+';
|
|
g<<'a'<<i<<"*a"<<i;
|
|
}
|
|
g<<
|
|
";" NL
|
|
TAB1 "if( m2==scalar_traits<typename vec_traits<A>::scalar_type>::value(0) )" NL
|
|
TAB2"BOOST_QVM_THROW_EXCEPTION(zero_magnitude_error());" NL
|
|
TAB1 "T const rm=scalar_traits<T>::value(1)/sqrt<T>(m2);" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "vec_traits<A>::template write_element<"<<i<<">(a)*=rm;" NL;
|
|
g<<TAB1 "}" NL;
|
|
defined(g,d,"normalize",suffix);
|
|
}
|
|
|
|
void
|
|
dot( output_file & out, int d, char const * suffix )
|
|
{
|
|
header_sr_va_vb(out,d,"dot");
|
|
out.require_include(INCLUDE_DEDUCE_S);
|
|
out.require_include(INCLUDE_STATIC_ASSERT);
|
|
std::ostream & g=out.stream();
|
|
g<<
|
|
TAB1 "{" NL
|
|
TAB1 "typedef typename vec_traits<A>::scalar_type Ta;" NL
|
|
TAB1 "typedef typename vec_traits<B>::scalar_type Tb;" NL
|
|
TAB1 "typedef typename deduce_scalar<Ta,Tb>::type Tr;" NL
|
|
;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "Ta const a"<<i<<"=vec_traits<A>::template read_element<"<<i<<">(a);" NL;
|
|
for( int i=0; i!=d; ++i )
|
|
g<<TAB1 "Tb const b"<<i<<"=vec_traits<B>::template read_element<"<<i<<">(b);" NL;
|
|
g<<TAB1 "Tr const dot=";
|
|
for( int i=0; i!=d; ++i )
|
|
{
|
|
if( i )
|
|
g<<'+';
|
|
g<<'a'<<i<<"*b"<<i;
|
|
}
|
|
g<<
|
|
";" NL
|
|
TAB1 "return dot;" NL
|
|
TAB1 "}" NL
|
|
;
|
|
defined(g,d,"dot",suffix);
|
|
}
|
|
|
|
struct
|
|
swizzle_pair
|
|
{
|
|
char ch;
|
|
int idx;
|
|
};
|
|
|
|
template <int N>
|
|
void
|
|
swizzle_impl( std::ostream & g, int d, swizzle_pair const (&ids)[N], std::vector<int> const & initial_count )
|
|
{
|
|
assert(d>=2);
|
|
std::vector<int> count(initial_count);
|
|
for( char const * const ref_id[2] = { " const &", " &" };; )
|
|
{
|
|
int max_dim=-100;
|
|
for( int i=0; i!=d; ++i )
|
|
max_dim=std::max(max_dim,ids[count[i]-1].idx);
|
|
if( max_dim<0 )
|
|
{
|
|
g<<
|
|
"BOOST_QVM_INLINE_TRIVIAL" NL
|
|
"qvm_detail::sw01_<";
|
|
for( int k=0; k!=d; ++k )
|
|
g<<(k?",":"")<<"qvm_detail::swizzle_idx<"<<ids[count[k]-1].idx;
|
|
for( int k=0; k!=d; ++k )
|
|
g<<" >";
|
|
g<<
|
|
" > const &" NL
|
|
"_";
|
|
for( int k=0; k!=d; ++k )
|
|
{
|
|
char f=ids[count[k]-1].ch;
|
|
assert(f>='0' && f<='9');
|
|
g<<f;
|
|
}
|
|
g<<
|
|
"()" NL
|
|
TAB1 "{" NL
|
|
TAB1 "return *reinterpret_cast<qvm_detail::sw01_<";
|
|
for( int k=0; k!=d; ++k )
|
|
g<<(k?",":"")<<"qvm_detail::swizzle_idx<"<<ids[count[k]-1].idx;
|
|
for( int k=0; k!=d; ++k )
|
|
g<<" >";
|
|
g<<
|
|
" > const *>(qvm_detail::get_null());" NL
|
|
TAB1 "}" NL;
|
|
}
|
|
else
|
|
for( int rfid=0; rfid<2; ++rfid )
|
|
{
|
|
for( int scalar=0; scalar!=2; ++scalar )
|
|
{
|
|
if( scalar && max_dim>0 )
|
|
break;
|
|
if( scalar )
|
|
g<<
|
|
"template <class S>" NL
|
|
"BOOST_QVM_INLINE_TRIVIAL" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "is_scalar<S>::value," NL
|
|
TAB1 "qvm_detail::sws_<S,";
|
|
else
|
|
g<<
|
|
"template <class V>" NL
|
|
"BOOST_QVM_INLINE_TRIVIAL" NL
|
|
"typename enable_if_c<" NL
|
|
TAB1 "is_vec<V>::value && vec_traits<V>::dim>="<<max_dim+1<<"," NL
|
|
TAB1 "qvm_detail::sw_<V,";
|
|
for( int k=0; k!=d; ++k )
|
|
g<<(k?",":"")<<"qvm_detail::swizzle_idx<"<<ids[count[k]-1].idx;
|
|
for( int k=0; k!=d; ++k )
|
|
g<<" >";
|
|
g<<" >"<<ref_id[rfid]<<">::type" NL;
|
|
for( int k=0; k!=d; ++k )
|
|
{
|
|
char f=ids[count[k]-1].ch;
|
|
if( !k && f>='0' && f<='9' )
|
|
g<<'_';
|
|
g<<f;
|
|
}
|
|
if( scalar )
|
|
g<<
|
|
"( S"<<ref_id[rfid]<<" a )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "return reinterpret_cast<qvm_detail::sws_<S,";
|
|
else
|
|
g<<
|
|
"( V"<<ref_id[rfid]<<" a )" NL
|
|
TAB1 "{" NL
|
|
TAB1 "return reinterpret_cast<qvm_detail::sw_<V,";
|
|
for( int k=0; k!=d; ++k )
|
|
g<<(k?",":"")<<"qvm_detail::swizzle_idx<"<<ids[count[k]-1].idx;
|
|
for( int k=0; k!=d; ++k )
|
|
g<<" >";
|
|
g<<
|
|
" >"<<ref_id[rfid]<<">(a);" NL
|
|
TAB1 "}" NL;
|
|
}
|
|
}
|
|
int j;
|
|
for( j=0; j!=d; ++j )
|
|
if( --count[j] )
|
|
break;
|
|
else
|
|
count[j]=initial_count[j];
|
|
if( j==d )
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
swizzle( output_file & out, int d )
|
|
{
|
|
assert(d>1);
|
|
out.require_include(INCLUDE_INLINE);
|
|
out.require_include(INCLUDE_SWIZZLE_TRAITS);
|
|
out.require_include(INCLUDE_ENABLE_IF);
|
|
std::ostream & g=out.stream();
|
|
swizzle_pair const swizzle_ids[6] =
|
|
{
|
|
{'X',0},
|
|
{'Y',1},
|
|
{'Z',2},
|
|
{'W',3},
|
|
{'0',-1},
|
|
{'1',-2}
|
|
};
|
|
std::vector<int> initial_count(d,6);
|
|
swizzle_impl(g,d,swizzle_ids,initial_count);
|
|
}
|
|
|
|
command_line_options
|
|
parse_command_line( int argc, char const * argv[] )
|
|
{
|
|
class
|
|
next
|
|
{
|
|
char const * const * const argv;
|
|
public:
|
|
int const argc;
|
|
next( int argc, char const * argv[] ):
|
|
argv(argv),
|
|
argc(argc)
|
|
{
|
|
}
|
|
std::string
|
|
operator()( int & i ) const
|
|
{
|
|
assert(i<argc);
|
|
if( ++i==argc )
|
|
BOOST_THROW_EXCEPTION(bad_command_line() << cmd_arg(argv[i-1]));
|
|
return argv[i];
|
|
}
|
|
} next_token(argc,argv);
|
|
command_line_options r;
|
|
for( int i=1; i!=argc; ++i )
|
|
if( argv[i][0]=='-' )
|
|
{
|
|
char const * arg=argv[i];
|
|
if( arg==std::string("-od") )
|
|
r.output_directory=next_token(i);
|
|
else if( arg==std::string("-con") )
|
|
r.con=true;
|
|
else
|
|
BOOST_THROW_EXCEPTION(bad_command_line() << cmd_arg(arg));
|
|
}
|
|
return r;
|
|
}
|
|
|
|
void
|
|
gen( int argc, char const * argv[] )
|
|
{
|
|
command_line_options opt=parse_command_line(argc,argv);
|
|
for( int d=2; d!=5; ++d )
|
|
{
|
|
output_file f(opt);
|
|
{
|
|
char buf[1024];
|
|
sprintf(buf,INCLUDE_MAT_ASSIGN,d);
|
|
f.require_include(buf);
|
|
}
|
|
mr_op_ma_mb_same_size(f,d,d,"operator+","+","mm");
|
|
mr_op_ma_mb_same_size(f,d,1,"operator+","+","mm");
|
|
mr_op_ma_mb_same_size(f,1,d,"operator+","+","mm");
|
|
mr_op_ma_mb_same_size(f,d,d,"operator-","-","mm");
|
|
mr_op_ma_mb_same_size(f,d,1,"operator-","-","mm");
|
|
mr_op_ma_mb_same_size(f,1,d,"operator-","-","mm");
|
|
ma_op_ma_mb_same_size(f,d,d,"operator+=","+=","mm");
|
|
ma_op_ma_mb_same_size(f,d,1,"operator+=","+=","mm");
|
|
ma_op_ma_mb_same_size(f,1,d,"operator+=","+=","mm");
|
|
ma_op_ma_mb_same_size(f,d,d,"operator-=","-=","mm");
|
|
ma_op_ma_mb_same_size(f,d,1,"operator-=","-=","mm");
|
|
ma_op_ma_mb_same_size(f,1,d,"operator-=","-=","mm");
|
|
mr_op_ma_sb(f,d,d,"operator*","*","ms");
|
|
mr_op_sa_mb(f,d,d,"operator*","*","sm");
|
|
mr_op_ma_sb(f,d,1,"operator*","*","ms");
|
|
mr_op_sa_mb(f,d,1,"operator*","*","sm");
|
|
mr_op_ma_sb(f,1,d,"operator*","*","ms");
|
|
mr_op_sa_mb(f,1,d,"operator*","*","sm");
|
|
ma_op_ma_sb(f,d,d,"operator*=","*=","ms");
|
|
ma_op_ma_sb(f,d,1,"operator*=","*=","ms");
|
|
ma_op_ma_sb(f,1,d,"operator*=","*=","ms");
|
|
mr_op_ma_sb(f,d,d,"operator/","/","ms");
|
|
mr_op_sa_mb(f,d,d,"operator/","/","sm");
|
|
mr_op_ma_sb(f,d,1,"operator/","/","ms");
|
|
mr_op_sa_mb(f,d,1,"operator/","/","sm");
|
|
mr_op_ma_sb(f,1,d,"operator/","/","ms");
|
|
ma_op_ma_sb(f,d,d,"operator/=","/=","ms");
|
|
ma_op_ma_sb(f,d,1,"operator/=","/=","ms");
|
|
ma_op_ma_sb(f,1,d,"operator/=","/=","ms");
|
|
mr_convert_to_ma(f,d,d,"m");
|
|
mr_convert_to_ma(f,d,1,"m");
|
|
mr_convert_to_ma(f,1,d,"m");
|
|
bool_eq_ma_mb(f,d,d,"mm");
|
|
bool_eq_ma_mb(f,d,1,"mm");
|
|
bool_eq_ma_mb(f,1,d,"mm");
|
|
bool_neq_ma_mb(f,d,d,"mm");
|
|
bool_neq_ma_mb(f,d,1,"mm");
|
|
bool_neq_ma_mb(f,1,d,"mm");
|
|
mr_op_ma(f,d,d,"operator-","-","m");
|
|
mr_op_ma(f,d,1,"operator-","-","m");
|
|
mr_op_ma(f,1,d,"operator-","-","m");
|
|
determinant(f,d,0);
|
|
inverse_ma(f,d,"m");
|
|
mr_mult_ma_mb(f,d,d,d,"mm");
|
|
ma_mult_ma_mb(f,d,"mm");
|
|
mr_mult_ma_mb(f,d,d,1,"mm");
|
|
mr_mult_ma_mb(f,1,d,d,"mm");
|
|
f.dump("mat_operations"+to_string(d)+".hpp");
|
|
}
|
|
|
|
for( int d=2; d!=5; ++d )
|
|
{
|
|
output_file f(opt);
|
|
ma_assign_ma_mb(f,d,d,"mm");
|
|
ma_assign_ma_mb(f,d,1,"mm");
|
|
ma_assign_ma_mb(f,1,d,"mm");
|
|
f.dump("mat_assign"+to_string(d)+".hpp");
|
|
}
|
|
|
|
for( int d=2; d!=5; ++d )
|
|
{
|
|
output_file f(opt);
|
|
{
|
|
char buf[1024];
|
|
sprintf(buf,INCLUDE_VEC_ASSIGN,d);
|
|
f.require_include(buf);
|
|
}
|
|
vr_op_va_vb_same_size(f,d,"operator+","+","vv");
|
|
vr_op_va_vb_same_size(f,d,"operator-","-","vv");
|
|
va_op_va_vb_same_size(f,d,"operator+=","+=","vv");
|
|
va_op_va_vb_same_size(f,d,"operator-=","-=","vv");
|
|
vr_op_va_sb(f,d,"operator*","*","vs");
|
|
vr_op_sa_vb(f,d,"operator*","*","sv");
|
|
va_op_va_sb(f,d,"operator*=","*=","vs");
|
|
vr_op_va_sb(f,d,"operator/","/","vs");
|
|
va_op_va_sb(f,d,"operator/=","/=","vs");
|
|
vr_convert_to_va(f,d,"v");
|
|
bool_eq_va_vb(f,d,"vv");
|
|
bool_neq_va_vb(f,d,"vv");
|
|
vr_op_va(f,d,"operator-","-","v");
|
|
mag(f,d,"v");
|
|
mag_sqr(f,d,"v");
|
|
normalize(f,d,"v");
|
|
dot(f,d,"vv");
|
|
f.dump("vec_operations"+to_string(d)+".hpp");
|
|
}
|
|
|
|
for( int d=2; d!=5; ++d )
|
|
{
|
|
output_file f(opt);
|
|
va_assign_va_vb(f,d,"vv");
|
|
f.dump("vec_assign"+to_string(d)+".hpp");
|
|
}
|
|
|
|
for( int d=2; d!=5; ++d )
|
|
{
|
|
output_file f(opt);
|
|
vr_mult_ma_vb(f,d,d,"mv");
|
|
vr_mult_va_mb(f,d,d,"vm");
|
|
f.dump("vec_mat_operations"+to_string(d)+".hpp");
|
|
}
|
|
|
|
{
|
|
output_file f(opt);
|
|
swizzle(f,2);
|
|
f.dump("swizzle2.hpp");
|
|
}
|
|
{
|
|
output_file f(opt);
|
|
swizzle(f,3);
|
|
f.dump("swizzle3.hpp");
|
|
}
|
|
{
|
|
output_file f(opt);
|
|
swizzle(f,4);
|
|
f.dump("swizzle4.hpp");
|
|
}
|
|
}
|
|
}
|
|
|
|
int
|
|
main( int argc, char const * argv[] )
|
|
{
|
|
try
|
|
{
|
|
gen(argc,argv);
|
|
}
|
|
catch(
|
|
std::ifstream::failure & )
|
|
{
|
|
std::cerr << "Failed to write generated output file" << std::endl;
|
|
}
|
|
catch(
|
|
... )
|
|
{
|
|
std::cerr << "Unexpected exception" << std::endl << boost::current_exception_diagnostic_information();
|
|
}
|
|
return 1;
|
|
}
|