| // This file is part of Eigen, a lightweight C++ template library | 
 | // for linear algebra. | 
 | // | 
 | // Copyright (C) 2015 Gael Guennebaud <gael.guennebaud@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_MAP_H | 
 | #define EIGEN_SPARSE_MAP_H | 
 |  | 
 | // IWYU pragma: private | 
 | #include "./InternalHeaderCheck.h" | 
 |  | 
 | namespace Eigen { | 
 |  | 
 | namespace internal { | 
 |  | 
 | template <typename MatScalar, int MatOptions, typename MatIndex, int Options, typename StrideType> | 
 | struct traits<Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > | 
 |     : public traits<SparseMatrix<MatScalar, MatOptions, MatIndex> > { | 
 |   typedef SparseMatrix<MatScalar, MatOptions, MatIndex> PlainObjectType; | 
 |   typedef traits<PlainObjectType> TraitsBase; | 
 |   enum { Flags = TraitsBase::Flags & (~NestByRefBit) }; | 
 | }; | 
 |  | 
 | template <typename MatScalar, int MatOptions, typename MatIndex, int Options, typename StrideType> | 
 | struct traits<Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > | 
 |     : public traits<SparseMatrix<MatScalar, MatOptions, MatIndex> > { | 
 |   typedef SparseMatrix<MatScalar, MatOptions, MatIndex> PlainObjectType; | 
 |   typedef traits<PlainObjectType> TraitsBase; | 
 |   enum { Flags = TraitsBase::Flags & (~(NestByRefBit | LvalueBit)) }; | 
 | }; | 
 |  | 
 | }  // end namespace internal | 
 |  | 
 | template <typename Derived, | 
 |           int Level = internal::accessors_level<Derived>::has_write_access ? WriteAccessors : ReadOnlyAccessors> | 
 | class SparseMapBase; | 
 |  | 
 | /** \ingroup SparseCore_Module | 
 |  * class SparseMapBase | 
 |  * \brief Common base class for Map and Ref instance of sparse matrix and vector. | 
 |  */ | 
 | template <typename Derived> | 
 | class SparseMapBase<Derived, ReadOnlyAccessors> : public SparseCompressedBase<Derived> { | 
 |  public: | 
 |   typedef SparseCompressedBase<Derived> Base; | 
 |   typedef typename Base::Scalar Scalar; | 
 |   typedef typename Base::StorageIndex StorageIndex; | 
 |   enum { IsRowMajor = Base::IsRowMajor }; | 
 |   using Base::operator=; | 
 |  | 
 |  protected: | 
 |   typedef std::conditional_t<bool(internal::is_lvalue<Derived>::value), Scalar*, const Scalar*> ScalarPointer; | 
 |   typedef std::conditional_t<bool(internal::is_lvalue<Derived>::value), StorageIndex*, const StorageIndex*> | 
 |       IndexPointer; | 
 |  | 
 |   Index m_outerSize; | 
 |   Index m_innerSize; | 
 |   Array<StorageIndex, 2, 1> m_zero_nnz; | 
 |   IndexPointer m_outerIndex; | 
 |   IndexPointer m_innerIndices; | 
 |   ScalarPointer m_values; | 
 |   IndexPointer m_innerNonZeros; | 
 |  | 
 |  public: | 
 |   /** \copydoc SparseMatrixBase::rows() */ | 
 |   inline Index rows() const { return IsRowMajor ? m_outerSize : m_innerSize; } | 
 |   /** \copydoc SparseMatrixBase::cols() */ | 
 |   inline Index cols() const { return IsRowMajor ? m_innerSize : m_outerSize; } | 
 |   /** \copydoc SparseMatrixBase::innerSize() */ | 
 |   inline Index innerSize() const { return m_innerSize; } | 
 |   /** \copydoc SparseMatrixBase::outerSize() */ | 
 |   inline Index outerSize() const { return m_outerSize; } | 
 |   /** \copydoc SparseCompressedBase::nonZeros */ | 
 |   inline Index nonZeros() const { return m_zero_nnz[1]; } | 
 |  | 
 |   /** \copydoc SparseCompressedBase::isCompressed */ | 
 |   bool isCompressed() const { return m_innerNonZeros == 0; } | 
 |  | 
 |   //---------------------------------------- | 
 |   // direct access interface | 
 |   /** \copydoc SparseMatrix::valuePtr */ | 
 |   inline const Scalar* valuePtr() const { return m_values; } | 
 |   /** \copydoc SparseMatrix::innerIndexPtr */ | 
 |   inline const StorageIndex* innerIndexPtr() const { return m_innerIndices; } | 
 |   /** \copydoc SparseMatrix::outerIndexPtr */ | 
 |   inline const StorageIndex* outerIndexPtr() const { return m_outerIndex; } | 
 |   /** \copydoc SparseMatrix::innerNonZeroPtr */ | 
 |   inline const StorageIndex* innerNonZeroPtr() const { return m_innerNonZeros; } | 
 |   //---------------------------------------- | 
 |  | 
 |   /** \copydoc SparseMatrix::coeff */ | 
 |   inline Scalar coeff(Index row, Index col) const { | 
 |     const Index outer = IsRowMajor ? row : col; | 
 |     const Index inner = IsRowMajor ? col : row; | 
 |  | 
 |     Index start = m_outerIndex[outer]; | 
 |     Index end = isCompressed() ? m_outerIndex[outer + 1] : start + m_innerNonZeros[outer]; | 
 |     if (start == end) | 
 |       return Scalar(0); | 
 |     else if (end > 0 && inner == m_innerIndices[end - 1]) | 
 |       return m_values[end - 1]; | 
 |     // ^^  optimization: let's first check if it is the last coefficient | 
 |     // (very common in high level algorithms) | 
 |  | 
 |     const StorageIndex* r = std::lower_bound(&m_innerIndices[start], &m_innerIndices[end - 1], inner); | 
 |     const Index id = r - &m_innerIndices[0]; | 
 |     return ((*r == inner) && (id < end)) ? m_values[id] : Scalar(0); | 
 |   } | 
 |  | 
 |   inline SparseMapBase(Index rows, Index cols, Index nnz, IndexPointer outerIndexPtr, IndexPointer innerIndexPtr, | 
 |                        ScalarPointer valuePtr, IndexPointer innerNonZerosPtr = 0) | 
 |       : m_outerSize(IsRowMajor ? rows : cols), | 
 |         m_innerSize(IsRowMajor ? cols : rows), | 
 |         m_zero_nnz(0, internal::convert_index<StorageIndex>(nnz)), | 
 |         m_outerIndex(outerIndexPtr), | 
 |         m_innerIndices(innerIndexPtr), | 
 |         m_values(valuePtr), | 
 |         m_innerNonZeros(innerNonZerosPtr) {} | 
 |  | 
 |   // for vectors | 
 |   inline SparseMapBase(Index size, Index nnz, IndexPointer innerIndexPtr, ScalarPointer valuePtr) | 
 |       : m_outerSize(1), | 
 |         m_innerSize(size), | 
 |         m_zero_nnz(0, internal::convert_index<StorageIndex>(nnz)), | 
 |         m_outerIndex(m_zero_nnz.data()), | 
 |         m_innerIndices(innerIndexPtr), | 
 |         m_values(valuePtr), | 
 |         m_innerNonZeros(0) {} | 
 |  | 
 |   /** Empty destructor */ | 
 |   inline ~SparseMapBase() {} | 
 |  | 
 |  protected: | 
 |   inline SparseMapBase() {} | 
 | }; | 
 |  | 
 | /** \ingroup SparseCore_Module | 
 |  * class SparseMapBase | 
 |  * \brief Common base class for writable Map and Ref instance of sparse matrix and vector. | 
 |  */ | 
 | template <typename Derived> | 
 | class SparseMapBase<Derived, WriteAccessors> : public SparseMapBase<Derived, ReadOnlyAccessors> { | 
 |   typedef MapBase<Derived, ReadOnlyAccessors> ReadOnlyMapBase; | 
 |  | 
 |  public: | 
 |   typedef SparseMapBase<Derived, ReadOnlyAccessors> Base; | 
 |   typedef typename Base::Scalar Scalar; | 
 |   typedef typename Base::StorageIndex StorageIndex; | 
 |   enum { IsRowMajor = Base::IsRowMajor }; | 
 |  | 
 |   using Base::operator=; | 
 |  | 
 |  public: | 
 |   //---------------------------------------- | 
 |   // direct access interface | 
 |   using Base::innerIndexPtr; | 
 |   using Base::innerNonZeroPtr; | 
 |   using Base::outerIndexPtr; | 
 |   using Base::valuePtr; | 
 |   /** \copydoc SparseMatrix::valuePtr */ | 
 |   inline Scalar* valuePtr() { return Base::m_values; } | 
 |   /** \copydoc SparseMatrix::innerIndexPtr */ | 
 |   inline StorageIndex* innerIndexPtr() { return Base::m_innerIndices; } | 
 |   /** \copydoc SparseMatrix::outerIndexPtr */ | 
 |   inline StorageIndex* outerIndexPtr() { return Base::m_outerIndex; } | 
 |   /** \copydoc SparseMatrix::innerNonZeroPtr */ | 
 |   inline StorageIndex* innerNonZeroPtr() { return Base::m_innerNonZeros; } | 
 |   //---------------------------------------- | 
 |  | 
 |   /** \copydoc SparseMatrix::coeffRef */ | 
 |   inline Scalar& coeffRef(Index row, Index col) { | 
 |     const Index outer = IsRowMajor ? row : col; | 
 |     const Index inner = IsRowMajor ? col : row; | 
 |  | 
 |     Index start = Base::m_outerIndex[outer]; | 
 |     Index end = Base::isCompressed() ? Base::m_outerIndex[outer + 1] : start + Base::m_innerNonZeros[outer]; | 
 |     eigen_assert(end >= start && "you probably called coeffRef on a non finalized matrix"); | 
 |     eigen_assert(end > start && "coeffRef cannot be called on a zero coefficient"); | 
 |     StorageIndex* r = std::lower_bound(&Base::m_innerIndices[start], &Base::m_innerIndices[end], inner); | 
 |     const Index id = r - &Base::m_innerIndices[0]; | 
 |     eigen_assert((*r == inner) && (id < end) && "coeffRef cannot be called on a zero coefficient"); | 
 |     return const_cast<Scalar*>(Base::m_values)[id]; | 
 |   } | 
 |  | 
 |   inline SparseMapBase(Index rows, Index cols, Index nnz, StorageIndex* outerIndexPtr, StorageIndex* innerIndexPtr, | 
 |                        Scalar* valuePtr, StorageIndex* innerNonZerosPtr = 0) | 
 |       : Base(rows, cols, nnz, outerIndexPtr, innerIndexPtr, valuePtr, innerNonZerosPtr) {} | 
 |  | 
 |   // for vectors | 
 |   inline SparseMapBase(Index size, Index nnz, StorageIndex* innerIndexPtr, Scalar* valuePtr) | 
 |       : Base(size, nnz, innerIndexPtr, valuePtr) {} | 
 |  | 
 |   /** Empty destructor */ | 
 |   inline ~SparseMapBase() {} | 
 |  | 
 |  protected: | 
 |   inline SparseMapBase() {} | 
 | }; | 
 |  | 
 | /** \ingroup SparseCore_Module | 
 |  * | 
 |  * \brief Specialization of class Map for SparseMatrix-like storage. | 
 |  * | 
 |  * \tparam SparseMatrixType the equivalent sparse matrix type of the referenced data, it must be a template instance of | 
 |  * class SparseMatrix. | 
 |  * | 
 |  * \sa class Map, class SparseMatrix, class Ref<SparseMatrixType,Options> | 
 |  */ | 
 | #ifndef EIGEN_PARSED_BY_DOXYGEN | 
 | template <typename MatScalar, int MatOptions, typename MatIndex, int Options, typename StrideType> | 
 | class Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> | 
 |     : public SparseMapBase<Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > | 
 | #else | 
 | template <typename SparseMatrixType> | 
 | class Map<SparseMatrixType> : public SparseMapBase<Derived, WriteAccessors> | 
 | #endif | 
 | { | 
 |  public: | 
 |   typedef SparseMapBase<Map> Base; | 
 |   EIGEN_SPARSE_PUBLIC_INTERFACE(Map) | 
 |   enum { IsRowMajor = Base::IsRowMajor }; | 
 |  | 
 |  public: | 
 |   /** Constructs a read-write Map to a sparse matrix of size \a rows x \a cols, containing \a nnz non-zero coefficients, | 
 |    * stored as a sparse format as defined by the pointers \a outerIndexPtr, \a innerIndexPtr, and \a valuePtr. | 
 |    * If the optional parameter \a innerNonZerosPtr is the null pointer, then a standard compressed format is assumed. | 
 |    * The inner indices must be sorted appropriately. | 
 |    * | 
 |    * This constructor is available only if \c SparseMatrixType is non-const. | 
 |    * | 
 |    * More details on the expected storage schemes are given in the \ref TutorialSparse "manual pages". | 
 |    */ | 
 |   inline Map(Index rows, Index cols, Index nnz, StorageIndex* outerIndexPtr, StorageIndex* innerIndexPtr, | 
 |              Scalar* valuePtr, StorageIndex* innerNonZerosPtr = 0) | 
 |       : Base(rows, cols, nnz, outerIndexPtr, innerIndexPtr, valuePtr, innerNonZerosPtr) {} | 
 | #ifndef EIGEN_PARSED_BY_DOXYGEN | 
 |   /** Empty destructor */ | 
 |   inline ~Map() {} | 
 | }; | 
 |  | 
 | template <typename MatScalar, int MatOptions, typename MatIndex, int Options, typename StrideType> | 
 | class Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> | 
 |     : public SparseMapBase<Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > { | 
 |  public: | 
 |   typedef SparseMapBase<Map> Base; | 
 |   EIGEN_SPARSE_PUBLIC_INTERFACE(Map) | 
 |   enum { IsRowMajor = Base::IsRowMajor }; | 
 |  | 
 |  public: | 
 | #endif | 
 |   /** This is the const version of the above constructor. | 
 |    * | 
 |    * This constructor is available only if \c SparseMatrixType is const, e.g.: | 
 |    * \code Map<const SparseMatrix<double> >  \endcode | 
 |    */ | 
 |   inline Map(Index rows, Index cols, Index nnz, const StorageIndex* outerIndexPtr, const StorageIndex* innerIndexPtr, | 
 |              const Scalar* valuePtr, const StorageIndex* innerNonZerosPtr = 0) | 
 |       : Base(rows, cols, nnz, outerIndexPtr, innerIndexPtr, valuePtr, innerNonZerosPtr) {} | 
 |  | 
 |   /** Empty destructor */ | 
 |   inline ~Map() {} | 
 | }; | 
 |  | 
 | namespace internal { | 
 |  | 
 | template <typename MatScalar, int MatOptions, typename MatIndex, int Options, typename StrideType> | 
 | struct evaluator<Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > | 
 |     : evaluator<SparseCompressedBase<Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > > { | 
 |   typedef evaluator<SparseCompressedBase<Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > > | 
 |       Base; | 
 |   typedef Map<SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> XprType; | 
 |   evaluator() : Base() {} | 
 |   explicit evaluator(const XprType& mat) : Base(mat) {} | 
 | }; | 
 |  | 
 | template <typename MatScalar, int MatOptions, typename MatIndex, int Options, typename StrideType> | 
 | struct evaluator<Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > | 
 |     : evaluator<SparseCompressedBase<Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > > { | 
 |   typedef evaluator< | 
 |       SparseCompressedBase<Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> > > | 
 |       Base; | 
 |   typedef Map<const SparseMatrix<MatScalar, MatOptions, MatIndex>, Options, StrideType> XprType; | 
 |   evaluator() : Base() {} | 
 |   explicit evaluator(const XprType& mat) : Base(mat) {} | 
 | }; | 
 |  | 
 | }  // namespace internal | 
 |  | 
 | }  // end namespace Eigen | 
 |  | 
 | #endif  // EIGEN_SPARSE_MAP_H |