| // This file is part of Eigen, a lightweight C++ template library | 
 | // for linear algebra. | 
 | // | 
 | // Copyright (C) 2012 Chen-Pang He <jdh8@ms63.hinet.net> | 
 | // | 
 | // This Source Code Form is subject to the terms of the Mozilla | 
 | // Public License v. 2.0. If a copy of the MPL was not distributed | 
 | // with this file, You can obtain one at http://mozilla.org/MPL/2.0/. | 
 |  | 
 | #ifndef EIGEN_PACKED_TRIANGULAR_MATRIX_VECTOR_H | 
 | #define EIGEN_PACKED_TRIANGULAR_MATRIX_VECTOR_H | 
 |  | 
 | namespace Eigen { | 
 | namespace internal { | 
 |  | 
 | template <typename Index, int Mode, typename LhsScalar, bool ConjLhs, typename RhsScalar, bool ConjRhs, | 
 |           int StorageOrder> | 
 | struct packed_triangular_matrix_vector_product; | 
 |  | 
 | template <typename Index, int Mode, typename LhsScalar, bool ConjLhs, typename RhsScalar, bool ConjRhs> | 
 | struct packed_triangular_matrix_vector_product<Index, Mode, LhsScalar, ConjLhs, RhsScalar, ConjRhs, ColMajor> { | 
 |   typedef typename ScalarBinaryOpTraits<LhsScalar, RhsScalar>::ReturnType ResScalar; | 
 |   enum { | 
 |     IsLower = (Mode & Lower) == Lower, | 
 |     HasUnitDiag = (Mode & UnitDiag) == UnitDiag, | 
 |     HasZeroDiag = (Mode & ZeroDiag) == ZeroDiag | 
 |   }; | 
 |   static void run(Index size, const LhsScalar* lhs, const RhsScalar* rhs, ResScalar* res, ResScalar alpha) { | 
 |     internal::conj_if<ConjRhs> cj; | 
 |     typedef Map<const Matrix<LhsScalar, Dynamic, 1> > LhsMap; | 
 |     typedef typename conj_expr_if<ConjLhs, LhsMap>::type ConjLhsType; | 
 |     typedef Map<Matrix<ResScalar, Dynamic, 1> > ResMap; | 
 |  | 
 |     for (Index i = 0; i < size; ++i) { | 
 |       Index s = IsLower && (HasUnitDiag || HasZeroDiag) ? 1 : 0; | 
 |       Index r = IsLower ? size - i : i + 1; | 
 |       if (!(HasUnitDiag || HasZeroDiag) || (--r > 0)) { | 
 |         ResMap(res + (IsLower ? s + i : 0), r) += alpha * cj(rhs[i]) * ConjLhsType(LhsMap(lhs + s, r)); | 
 |       } | 
 |       if (HasUnitDiag) { | 
 |         res[i] += alpha * cj(rhs[i]); | 
 |       } | 
 |       lhs += IsLower ? size - i : i + 1; | 
 |     } | 
 |   }; | 
 | }; | 
 |  | 
 | template <typename Index, int Mode, typename LhsScalar, bool ConjLhs, typename RhsScalar, bool ConjRhs> | 
 | struct packed_triangular_matrix_vector_product<Index, Mode, LhsScalar, ConjLhs, RhsScalar, ConjRhs, RowMajor> { | 
 |   typedef typename ScalarBinaryOpTraits<LhsScalar, RhsScalar>::ReturnType ResScalar; | 
 |   enum { | 
 |     IsLower = (Mode & Lower) == Lower, | 
 |     HasUnitDiag = (Mode & UnitDiag) == UnitDiag, | 
 |     HasZeroDiag = (Mode & ZeroDiag) == ZeroDiag | 
 |   }; | 
 |   static void run(Index size, const LhsScalar* lhs, const RhsScalar* rhs, ResScalar* res, ResScalar alpha) { | 
 |     internal::conj_if<ConjRhs> cj; | 
 |     typedef Map<const Matrix<LhsScalar, Dynamic, 1> > LhsMap; | 
 |     typedef typename conj_expr_if<ConjLhs, LhsMap>::type ConjLhsType; | 
 |     typedef Map<const Matrix<RhsScalar, Dynamic, 1> > RhsMap; | 
 |     typedef typename conj_expr_if<ConjRhs, RhsMap>::type ConjRhsType; | 
 |  | 
 |     for (Index i = 0; i < size; ++i) { | 
 |       Index s = !IsLower && (HasUnitDiag || HasZeroDiag) ? 1 : 0; | 
 |       Index r = IsLower ? i + 1 : size - i; | 
 |       if (!(HasUnitDiag || HasZeroDiag) || (--r > 0)) { | 
 |         res[i] += | 
 |             alpha * | 
 |             (ConjLhsType(LhsMap(lhs + s, r)).cwiseProduct(ConjRhsType(RhsMap(rhs + (IsLower ? 0 : s + i), r)))).sum(); | 
 |       } | 
 |       if (HasUnitDiag) { | 
 |         res[i] += alpha * cj(rhs[i]); | 
 |       } | 
 |       lhs += IsLower ? i + 1 : size - i; | 
 |     } | 
 |   }; | 
 | }; | 
 |  | 
 | }  // namespace internal | 
 | }  // namespace Eigen | 
 |  | 
 | #endif  // EIGEN_PACKED_TRIANGULAR_MATRIX_VECTOR_H |