/******************************************************************************* * * TRIQS: a Toolbox for Research in Interacting Quantum Systems * * Copyright (C) 2012 by O. Parcollet * * TRIQS 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. * * TRIQS 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 * TRIQS. If not, see . * ******************************************************************************/ #ifndef TRIQS_ARRAYS_BLAS_LAPACK_GEMV_H #define TRIQS_ARRAYS_BLAS_LAPACK_GEMV_H #include "./tools.hpp" #include "./qcache.hpp" namespace triqs { namespace arrays { namespace blas { using namespace blas_lapack_tools; namespace f77 { // overload typedef std::complex dcomplex; extern "C" { void TRIQS_FORTRAN_MANGLING(dgemv)(const char* trans, const int & m, const int & n, const double & alpha, const double A[], int & lda, const double x[], const int & incx, const double & beta, double y[], const int & incy); void TRIQS_FORTRAN_MANGLING(zgemv)(const char* trans, const int & m, const int & n, const std::complex & alpha, const std::complex A[], int & lda, const std::complex x[], const int & incx, const std::complex & beta, std::complex y[], const int & incy); } inline void gemv (char * trans, const int & M, const int & N, double & alpha, const double* A, int & LDA, const double* x, const int & incx, double & beta, double* Y, const int & incy) { TRIQS_FORTRAN_MANGLING(dgemv)(trans, M, N, alpha, A, LDA,x, incx , beta, Y, incy); } inline void gemv (char * trans, const int & M, const int & N, dcomplex & alpha, const dcomplex* A, int & LDA, const dcomplex* x, const int & incx, dcomplex & beta, dcomplex* Y, const int & incy) { TRIQS_FORTRAN_MANGLING(zgemv)(trans, M, N, alpha, A, LDA,x, incx , beta, Y, incy); } } template struct use_blas_gemv { static_assert(is_amv_value_or_view_class::value, "output of matrix product must be a matrix or matrix_view"); //static constexpr bool are_both_value_view = is_amv_value_or_view_class::value && is_amv_value_or_view_class::value; //static constexpr bool value = are_both_value_view && is_blas_lapack_type::value && have_same_value_type< MT, VT, VTOut>::value; static constexpr bool value = is_blas_lapack_type::value && have_same_value_type< MT, VT, VTOut>::value; // cf gemm comment }; /** * Calls gemv * Takes care of making temporary copies if necessary */ template typename std::enable_if< use_blas_gemv::value >::type gemv (typename MT::value_type alpha, MT const & A, VT const & X, typename MT::value_type beta, VTOut & Y) { //std::cerr << "gemm: blas call "<< std::endl ; resize_or_check_if_view(Y,make_shape(first_dim(A)));// first resize if necessary and possible const_qcache Ca(A); const_qcache Cx(X); // mettre la condition a la main if (!(second_dim(Ca()) == Cx().size())) TRIQS_RUNTIME_ERROR << "Dimension mismatch in gemv : A : "<< get_shape(Ca()) <<" while X : "< void gemv_generic (typename MT::value_type alpha, MT const & A, VT const & X, typename MT::value_type beta, VTOut & C) { //std::cerr << "gemm: generic call "<< std::endl ; // first resize if necessary and possible resize_or_check_if_view(C,make_shape(first_dim(A))); if (second_dim(A) != X.size()) TRIQS_RUNTIME_ERROR << "gemm generic : dimension mismatch "<< second_dim(A) << " vs " << X.size(); C() = 0; for (int i=0; i typename std::enable_if< !use_blas_gemv::value >::type gemv (typename MT::value_type alpha, MT const & A, VT const & X, typename MT::value_type beta, VTOut & Y) { gemv_generic(alpha,A,X,beta,Y); } // to allow gem (alpha, a, b, beta, M(..., ...)) i.e. a temporary view, which is not matched by previos templates // which require an lvalue. template void gemv (A alpha, MT const & a, VT const & b, B beta, vector_view && c) { gemv(alpha,a,b,beta,c);} }}}// namespace #endif