/* LA: linear algebra C++ interface library Copyright (C) 2008 Jiri Pittner or This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ // //for autotools // //#include "config.h" //this would force the user of the library to have config.h //////////////////////////////////////////////////////////////////////////// //LA traits classes and generally needed includes #ifndef _LA_TRAITS_INCL #define _LA_TRAITS_INCL //avoid unsupported register directive in new C++ versions #if __cplusplus > 201402L #define register #endif #include #include #include #include #include #include #include //using namespace std; #include "laerror.h" #include "la_random.h" #include "cuda_la.h" #ifdef NONCBLAS #include "noncblas.h" #else extern "C" { #ifdef HAS_MKL #include "mkl_cblas.h" #else #include "cblas.h" #endif } #endif #ifdef NONCLAPACK #include "noncblas.h" #else extern "C" { #include "clapack.h" } #endif //forward declarations namespace LA_Vecmat3 { template class Vec3; template class Mat3; } namespace LA { //forward declarations template class NRVec; template class NRVec_from1; template class NRMat; template class NRMat_from1; template class NRSMat; template class NRSMat_from1; template class SparseMat; template class SparseSMat; template class CSRMat; template class NRPerm; template class CyclePerm; template class Partition; template class CompressedPartition; //trick to allow real and imag part of complex as l-values template T &real(std::complex &c) { return reinterpret_cast(&c)[0]; } template T &imag(std::complex &c) { return reinterpret_cast(&c)[1]; } // typedef class {} Dummy_type; typedef class {} Dummy_type2; //for components of complex numbers // template struct LA_traits_complex { typedef Dummy_type Component_type; typedef Dummy_type NRVec_Noncomplex_type; typedef Dummy_type NRMat_Noncomplex_type; typedef Dummy_type2 NRSMat_Noncomplex_type; }; #define SPECIALIZE_COMPLEX(T) \ template<> \ struct LA_traits_complex > \ { \ typedef T Component_type; \ typedef NRVec NRVec_Noncomplex_type; \ typedef NRMat NRMat_Noncomplex_type; \ typedef NRSMat NRSMat_Noncomplex_type; \ }; SPECIALIZE_COMPLEX(double) SPECIALIZE_COMPLEX(std::complex) SPECIALIZE_COMPLEX(float) SPECIALIZE_COMPLEX(std::complex) SPECIALIZE_COMPLEX(char) SPECIALIZE_COMPLEX(unsigned char) SPECIALIZE_COMPLEX(short) SPECIALIZE_COMPLEX(unsigned short) SPECIALIZE_COMPLEX(int) SPECIALIZE_COMPLEX(unsigned int) SPECIALIZE_COMPLEX(long) SPECIALIZE_COMPLEX(unsigned long) SPECIALIZE_COMPLEX(long long) SPECIALIZE_COMPLEX(unsigned long long) //for general sortable classes template struct LA_sort_traits; template struct LA_sort_traits { static inline bool compare(const T &object, I i, I j) {return object.bigger(i,j);}; static inline bool comparestable(const T &object, I i, I j, I *auxkey) { bool r= object.bigger(i,j); if(!r && object[i]==object[j]) r= auxkey[i]>auxkey[j]; return r; }; }; template struct LA_sort_traits { static inline bool compare(const T &object, I i, I j) {return object.smaller(i,j);}; static inline bool comparestable(const T &object, I i, I j, I *auxkey) { bool r= object.smaller(i,j); if(!r && object[i]==object[j]) r= auxkey[i]> I/O operators template struct LA_traits_io { typedef C IOtype; }; template<> struct LA_traits_io { typedef int IOtype; }; template<> struct LA_traits_io { typedef unsigned int IOtype; }; //let's do some simple template metaprogramming and preprocessing //to keep the thing general and compact class scalar_false {}; class scalar_true {}; //default is non-scalar template class isscalar { public: typedef scalar_false scalar_type;}; //specializations #define SCALAR(X) \ template<>\ class isscalar {public: typedef scalar_true scalar_type;};\ template<>\ class isscalar > {public: typedef scalar_true scalar_type;};\ template<>\ class isscalar > > {public: typedef scalar_true scalar_type;};\ //declare what is scalar SCALAR(bool) SCALAR(char) SCALAR(short) SCALAR(int) SCALAR(long) SCALAR(long long) SCALAR(unsigned char) SCALAR(unsigned short) SCALAR(unsigned int) SCALAR(unsigned long) SCALAR(unsigned long long) SCALAR(float) SCALAR(double) SCALAR(void *) //#undef SCALAR //declare this generically as traits for any unknown class template struct LA_traits_aux { typedef Dummy_type normtype; }; //TRAITS SPECIALIZATIONS ////now declare the traits for scalars and for composed classes ////NOTE! methods in traits classes have to be declared static, ////since the class itself is never instantiated. ////for performance, it can be also inlined at the same time // //complex scalars template struct LA_traits_aux, scalar_true> { typedef std::complex elementtype; typedef std::complex producttype; typedef C normtype; typedef C realtype; typedef std::complex complextype; static inline C sqrabs(const std::complex x) { return x.real()*x.real()+x.imag()*x.imag();} static inline bool gencmp(const std::complex *x, const std::complex *y, size_t n) {return memcmp(x,y,n*sizeof(std::complex));} static bool bigger(const std::complex &x, const std::complex &y) {laerror("std::complex comparison undefined"); return false;} static bool smaller(const std::complex &x, const std::complex &y) {laerror("std::complex comparison undefined"); return false;} static inline normtype norm (const std::complex &x) {return std::abs(x);} static inline normtype abs2(const std::complex &x) {return std::abs(x.real())+std::abs(x.imag());} //faster without sqrt static inline void axpy (std::complex &s, const std::complex &x, const std::complex &c) {s+=x*c;} static inline void get(int fd, std::complex &x, bool dimensions=0, bool transp=0) {if(sizeof(std::complex)!=read(fd,&x,sizeof(std::complex))) laerror("read error");} static inline void put(int fd, const std::complex &x, bool dimensions=0, bool transp=0) {if(sizeof(std::complex)!=write(fd,&x,sizeof(std::complex))) laerror("write error");} static void multiget(size_t n,int fd, std::complex *x, bool dimensions=0) { size_t total=0; size_t system_limit = (1L<<30)/sizeof(std::complex); //do not expect too much from the system and read at most 1GB at once ssize_t r; size_t nn; do{ r=read(fd,x+total,nn=(n-total > system_limit ? system_limit : n-total)*sizeof(std::complex)); if(r<0 || r==0 && nn!=0 ) {std::cout<<"read returned "<); if(r%sizeof(std::complex)) laerror("read error 2"); } while(total < n); } static void multiput(size_t n, int fd, const std::complex *x, bool dimensions=0) { size_t total=0; size_t system_limit = (1L<<30)/sizeof(std::complex); //do not expect too much from the system and write at most 1GB at once ssize_t r; size_t nn; do{ r=write(fd,x+total,nn=(n-total > system_limit ? system_limit : n-total)*sizeof(std::complex)); if(r<0 || r==0 && nn!=0 ) {std::cout<<"write returned "<); if(r%sizeof(std::complex)) laerror("write error 2"); } while(total < n); } static void copy(std::complex *dest, std::complex *src, size_t n) {memcpy(dest,src,n*sizeof(std::complex));} static void clear(std::complex *dest, size_t n) {memset(dest,0,n*sizeof(std::complex));} static void copyonwrite(std::complex &x) {}; static bool is_plaindata() {return true;} static void clearme(std::complex &x) {x=0;}; static void deallocate(std::complex &x) {}; static inline std::complex conjugate(const std::complex &x) {return std::complex(x.real(),-x.imag());}; static inline C realpart(const std::complex &x) {return x.real();} static inline C imagpart(const std::complex &x) {return x.imag();} }; //non-complex scalars template struct LA_traits_aux { typedef C elementtype; typedef C producttype; typedef C normtype; typedef C realtype; typedef std::complex complextype; static inline C sqrabs(const C x) { return x*x;} static inline bool gencmp(const C *x, const C *y, size_t n) {return memcmp(x,y,n*sizeof(C));} static inline bool bigger(const C &x, const C &y) {return x>y;} static inline bool smaller(const C &x, const C &y) {return x system_limit ? system_limit : n-total)*sizeof(C)); if(r<0 || r==0 && nn!=0 ) {std::cout<<"read returned "< system_limit ? system_limit : n-total)*sizeof(C)); if(r<0 || r==0 && nn!=0 ) {std::cout<<"write returned "< struct LA_traits; //forward declaration needed for template recursion #define generate_traits_nonscalar(X) \ template \ struct LA_traits_aux, scalar_false> { \ typedef C elementtype; \ typedef X producttype; \ typedef C PAIRTYPE[2]; \ typedef C QUADRUPLETYPE[4]; \ typedef X SUBMATRIXTYPE; \ typedef typename LA_traits::normtype normtype; \ typedef X::realtype> realtype; \ typedef X::complextype> complextype; \ static bool gencmp(const X *x, const X *y, size_t n) {for(size_t i=0; i &x, const X &y) {return x>y;} \ static inline bool smaller(const X &x, const X &y) {return x &x) {return x.norm();} \ static inline void axpy (X&s, const X &x, const C c) {s.axpy(c,x);} \ static void put(int fd, const X &x, bool dimensions=1, bool transp=0) {x.put(fd,dimensions,transp);} \ static void get(int fd, X &x, bool dimensions=1, bool transp=0) {x.get(fd,dimensions,transp);} \ static void multiput(size_t n,int fd, const X *x, bool dimensions=1) {for(size_t i=0; i *x, bool dimensions=1) {for(size_t i=0; i *dest, X *src, size_t n) {for(size_t i=0; i *dest, size_t n) {for(size_t i=0; i &x) {x.copyonwrite();}\ static bool is_plaindata() {return false;}\ static void clearme(X &x) {x.clear();}\ static void deallocate(X &x) {x.dealloc();}\ }; //non-scalar types defined in this library generate_traits_nonscalar(NRMat) generate_traits_nonscalar(NRMat_from1) generate_traits_nonscalar(NRVec) generate_traits_nonscalar(NRVec_from1) generate_traits_nonscalar(SparseMat) generate_traits_nonscalar(SparseSMat) //product leading to non-symmetric result not implemented generate_traits_nonscalar(CSRMat) generate_traits_nonscalar(NRPerm) generate_traits_nonscalar(CyclePerm) generate_traits_nonscalar(Partition) generate_traits_nonscalar(CompressedPartition) //smat #define generate_traits_smat(X) \ template \ struct LA_traits_aux, scalar_false> { \ typedef C elementtype; \ typedef NRMat producttype; \ typedef typename LA_traits::normtype normtype; \ typedef X::realtype> realtype; \ typedef X::complextype> complextype; \ static bool gencmp(const C *x, const C *y, size_t n) {for(size_t i=0; iy;} \ static inline bool smaller(const C &x, const C &y) {return x &x) {return x.norm();} \ static inline void axpy (X&s, const X &x, const C c) {s.axpy(c,x);} \ static void put(int fd, const X &x, bool dimensions=1, bool transp=0) {x.put(fd,dimensions);} \ static void get(int fd, X &x, bool dimensions=1, bool transp=0) {x.get(fd,dimensions);} \ static void multiput(size_t n,int fd, const X *x, bool dimensions=1) {for(size_t i=0; i *x, bool dimensions=1) {for(size_t i=0; i &x) {x.copyonwrite();} \ static bool is_plaindata() {return false;}\ static void clearme(X &x) {x.clear();} \ static void deallocate(X &x) {x.dealloc();} \ }; generate_traits_smat(NRSMat) generate_traits_smat(NRSMat_from1) //the final traits class template struct LA_traits : LA_traits_aux::scalar_type> {}; }//namespace #endif