SparseTriangularView.h 6.29 KB
Newer Older
Don Gagne's avatar
Don Gagne committed
1 2 3
// This file is part of Eigen, a lightweight C++ template library
// for linear algebra.
//
4
// Copyright (C) 2009-2015 Gael Guennebaud <gael.guennebaud@inria.fr>
Don Gagne's avatar
Don Gagne committed
5 6 7 8 9 10 11 12 13
// Copyright (C) 2012 Désiré Nuentsa-Wakam <desire.nuentsa_wakam@inria.fr>
//
// 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_SPARSE_TRIANGULARVIEW_H
#define EIGEN_SPARSE_TRIANGULARVIEW_H

14 15 16 17 18 19 20 21 22 23 24 25 26
namespace Eigen {

/** \ingroup SparseCore_Module
  *
  * \brief Base class for a triangular part in a \b sparse matrix
  *
  * This class is an abstract base class of class TriangularView, and objects of type TriangularViewImpl cannot be instantiated.
  * It extends class TriangularView with additional methods which are available for sparse expressions only.
  *
  * \sa class TriangularView, SparseMatrixBase::triangularView()
  */
template<typename MatrixType, unsigned int Mode> class TriangularViewImpl<MatrixType,Mode,Sparse>
  : public SparseMatrixBase<TriangularView<MatrixType,Mode> >
Don Gagne's avatar
Don Gagne committed
27 28 29 30 31 32 33
{
    enum { SkipFirst = ((Mode&Lower) && !(MatrixType::Flags&RowMajorBit))
                    || ((Mode&Upper) &&  (MatrixType::Flags&RowMajorBit)),
           SkipLast = !SkipFirst,
           SkipDiag = (Mode&ZeroDiag) ? 1 : 0,
           HasUnitDiag = (Mode&UnitDiag) ? 1 : 0
    };
34 35 36 37 38 39
    
    typedef TriangularView<MatrixType,Mode> TriangularViewType;
    
  protected:
    // dummy solve function to make TriangularView happy.
    void solve() const;
Don Gagne's avatar
Don Gagne committed
40

41
    typedef SparseMatrixBase<TriangularViewType> Base;
Don Gagne's avatar
Don Gagne committed
42 43
  public:
    
44 45
    EIGEN_SPARSE_PUBLIC_INTERFACE(TriangularViewType)
    
Don Gagne's avatar
Don Gagne committed
46 47 48 49
    typedef typename MatrixType::Nested MatrixTypeNested;
    typedef typename internal::remove_reference<MatrixTypeNested>::type MatrixTypeNestedNonRef;
    typedef typename internal::remove_all<MatrixTypeNested>::type MatrixTypeNestedCleaned;

50 51 52 53 54 55 56
    template<typename RhsType, typename DstType>
    EIGEN_DEVICE_FUNC
    EIGEN_STRONG_INLINE void _solve_impl(const RhsType &rhs, DstType &dst) const {
      if(!(internal::is_same<RhsType,DstType>::value && internal::extract_data(dst) == internal::extract_data(rhs)))
        dst = rhs;
      this->solveInPlace(dst);
    }
Don Gagne's avatar
Don Gagne committed
57

58
    /** Applies the inverse of \c *this to the dense vector or matrix \a other, "in-place" */
Don Gagne's avatar
Don Gagne committed
59 60
    template<typename OtherDerived> void solveInPlace(MatrixBase<OtherDerived>& other) const;

61 62 63
    /** Applies the inverse of \c *this to the sparse vector or matrix \a other, "in-place" */
    template<typename OtherDerived> void solveInPlace(SparseMatrixBase<OtherDerived>& other) const;
  
Don Gagne's avatar
Don Gagne committed
64 65
};

66 67 68 69 70
namespace internal {

template<typename ArgType, unsigned int Mode>
struct unary_evaluator<TriangularView<ArgType,Mode>, IteratorBased>
 : evaluator_base<TriangularView<ArgType,Mode> >
Don Gagne's avatar
Don Gagne committed
71
{
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
  typedef TriangularView<ArgType,Mode> XprType;
  
protected:
  
  typedef typename XprType::Scalar Scalar;
  typedef typename XprType::StorageIndex StorageIndex;
  typedef typename evaluator<ArgType>::InnerIterator EvalIterator;
  
  enum { SkipFirst = ((Mode&Lower) && !(ArgType::Flags&RowMajorBit))
                    || ((Mode&Upper) &&  (ArgType::Flags&RowMajorBit)),
         SkipLast = !SkipFirst,
         SkipDiag = (Mode&ZeroDiag) ? 1 : 0,
         HasUnitDiag = (Mode&UnitDiag) ? 1 : 0
  };
  
public:
  
  enum {
    CoeffReadCost = evaluator<ArgType>::CoeffReadCost,
    Flags = XprType::Flags
  };
    
  explicit unary_evaluator(const XprType &xpr) : m_argImpl(xpr.nestedExpression()), m_arg(xpr.nestedExpression()) {}
  
  inline Index nonZerosEstimate() const {
    return m_argImpl.nonZerosEstimate();
  }
  
  class InnerIterator : public EvalIterator
  {
      typedef EvalIterator Base;
    public:
Don Gagne's avatar
Don Gagne committed
104

105 106
      EIGEN_STRONG_INLINE InnerIterator(const unary_evaluator& xprEval, Index outer)
        : Base(xprEval.m_argImpl,outer), m_returnOne(false), m_containsDiag(Base::outer()<xprEval.m_arg.innerSize())
Don Gagne's avatar
Don Gagne committed
107
      {
108 109 110 111 112 113 114 115 116 117 118 119 120
        if(SkipFirst)
        {
          while((*this) && ((HasUnitDiag||SkipDiag)  ? this->index()<=outer : this->index()<outer))
            Base::operator++();
          if(HasUnitDiag)
            m_returnOne = m_containsDiag;
        }
        else if(HasUnitDiag && ((!Base::operator bool()) || Base::index()>=Base::outer()))
        {
          if((!SkipFirst) && Base::operator bool())
            Base::operator++();
          m_returnOne = m_containsDiag;
        }
Don Gagne's avatar
Don Gagne committed
121
      }
122 123

      EIGEN_STRONG_INLINE InnerIterator& operator++()
Don Gagne's avatar
Don Gagne committed
124
      {
125 126 127 128
        if(HasUnitDiag && m_returnOne)
          m_returnOne = false;
        else
        {
Don Gagne's avatar
Don Gagne committed
129
          Base::operator++();
130 131 132 133 134 135 136 137
          if(HasUnitDiag && (!SkipFirst) && ((!Base::operator bool()) || Base::index()>=Base::outer()))
          {
            if((!SkipFirst) && Base::operator bool())
              Base::operator++();
            m_returnOne = m_containsDiag;
          }
        }
        return *this;
Don Gagne's avatar
Don Gagne committed
138
      }
139 140
      
      EIGEN_STRONG_INLINE operator bool() const
Don Gagne's avatar
Don Gagne committed
141
      {
142 143 144 145
        if(HasUnitDiag && m_returnOne)
          return true;
        if(SkipFirst) return  Base::operator bool();
        else
Don Gagne's avatar
Don Gagne committed
146
        {
147 148
          if (SkipDiag) return (Base::operator bool() && this->index() < this->outer());
          else return (Base::operator bool() && this->index() <= this->outer());
Don Gagne's avatar
Don Gagne committed
149 150 151
        }
      }

152 153 154
//       inline Index row() const { return (ArgType::Flags&RowMajorBit ? Base::outer() : this->index()); }
//       inline Index col() const { return (ArgType::Flags&RowMajorBit ? this->index() : Base::outer()); }
      inline StorageIndex index() const
Don Gagne's avatar
Don Gagne committed
155
      {
156 157
        if(HasUnitDiag && m_returnOne)  return internal::convert_index<StorageIndex>(Base::outer());
        else                            return Base::index();
Don Gagne's avatar
Don Gagne committed
158
      }
159
      inline Scalar value() const
Don Gagne's avatar
Don Gagne committed
160
      {
161 162
        if(HasUnitDiag && m_returnOne)  return Scalar(1);
        else                            return Base::value();
Don Gagne's avatar
Don Gagne committed
163
      }
164 165 166 167 168 169 170 171 172 173 174

    protected:
      bool m_returnOne;
      bool m_containsDiag;
    private:
      Scalar& valueRef();
  };
  
protected:
  evaluator<ArgType> m_argImpl;
  const ArgType& m_arg;
Don Gagne's avatar
Don Gagne committed
175 176
};

177 178
} // end namespace internal

Don Gagne's avatar
Don Gagne committed
179 180
template<typename Derived>
template<int Mode>
181
inline const TriangularView<const Derived, Mode>
Don Gagne's avatar
Don Gagne committed
182 183
SparseMatrixBase<Derived>::triangularView() const
{
184
  return TriangularView<const Derived, Mode>(derived());
Don Gagne's avatar
Don Gagne committed
185 186 187 188 189
}

} // end namespace Eigen

#endif // EIGEN_SPARSE_TRIANGULARVIEW_H