2017-01-06 00:01:44 +01:00
|
|
|
// This file is part of Eigen, a lightweight C++ template library
|
|
|
|
|
// for linear algebra.
|
|
|
|
|
//
|
|
|
|
|
// Copyright (C) 2017 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_ARITHMETIC_SEQUENCE_H
|
|
|
|
|
#define EIGEN_ARITHMETIC_SEQUENCE_H
|
|
|
|
|
|
|
|
|
|
namespace Eigen {
|
|
|
|
|
|
2017-01-11 14:28:28 +01:00
|
|
|
/** \namespace Eigen::placeholders
|
|
|
|
|
* \ingroup Core_Module
|
2017-01-10 11:46:29 +01:00
|
|
|
*
|
2017-01-11 17:24:02 +01:00
|
|
|
* Namespace containing symbolic placeholder and identifiers
|
2017-01-10 22:58:58 +01:00
|
|
|
*/
|
2017-01-10 10:31:07 +01:00
|
|
|
namespace placeholders {
|
|
|
|
|
|
2017-01-10 10:57:08 +01:00
|
|
|
namespace internal {
|
|
|
|
|
struct symbolic_last_tag {};
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-11 14:28:28 +01:00
|
|
|
/** \var last
|
|
|
|
|
* \ingroup Core_Module
|
|
|
|
|
*
|
|
|
|
|
* Can be used as a parameter to Eigen::seq and Eigen::seqN functions to symbolically reference the last element/row/columns
|
2017-01-10 22:58:58 +01:00
|
|
|
* of the underlying vector or matrix once passed to DenseBase::operator()(const RowIndices&, const ColIndices&).
|
|
|
|
|
*
|
|
|
|
|
* This symbolic placeholder support standard arithmetic operation.
|
|
|
|
|
*
|
|
|
|
|
* A typical usage example would be:
|
|
|
|
|
* \code
|
|
|
|
|
* using namespace Eigen;
|
|
|
|
|
* using Eigen::placeholders::last;
|
|
|
|
|
* VectorXd v(n);
|
|
|
|
|
* v(seq(2,last-2)).setOnes();
|
|
|
|
|
* \endcode
|
|
|
|
|
*
|
|
|
|
|
* \sa end
|
|
|
|
|
*/
|
2017-01-10 10:57:08 +01:00
|
|
|
static const Symbolic::SymbolExpr<internal::symbolic_last_tag> last;
|
2017-01-10 22:58:58 +01:00
|
|
|
|
2017-01-11 14:28:28 +01:00
|
|
|
/** \var end
|
|
|
|
|
* \ingroup Core_Module
|
|
|
|
|
*
|
|
|
|
|
* Can be used as a parameter to Eigen::seq and Eigen::seqN functions to symbolically reference the last+1 element/row/columns
|
2017-01-10 22:58:58 +01:00
|
|
|
* of the underlying vector or matrix once passed to DenseBase::operator()(const RowIndices&, const ColIndices&).
|
|
|
|
|
*
|
|
|
|
|
* This symbolic placeholder support standard arithmetic operation.
|
|
|
|
|
* It is essentially an alias to last+1
|
|
|
|
|
*
|
|
|
|
|
* \sa last
|
|
|
|
|
*/
|
2017-01-11 14:28:28 +01:00
|
|
|
#ifdef EIGEN_PARSED_BY_DOXYGEN
|
|
|
|
|
static const auto end = last+1;
|
|
|
|
|
#else
|
2017-01-10 10:57:08 +01:00
|
|
|
static const Symbolic::AddExpr<Symbolic::SymbolExpr<internal::symbolic_last_tag>,Symbolic::ValueExpr> end(last+1);
|
2017-01-11 14:28:28 +01:00
|
|
|
#endif
|
2017-01-06 00:01:44 +01:00
|
|
|
|
2017-01-10 10:31:07 +01:00
|
|
|
} // end namespace placeholders
|
|
|
|
|
|
2017-01-06 00:01:44 +01:00
|
|
|
//--------------------------------------------------------------------------------
|
2017-01-09 17:35:21 +01:00
|
|
|
// seq(first,last,incr) and seqN(first,size,incr)
|
2017-01-06 00:01:44 +01:00
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
|
|
2017-01-16 16:20:16 +01:00
|
|
|
/** \class ArithmeticSequence
|
2017-01-11 14:28:28 +01:00
|
|
|
* \ingroup Core_Module
|
2017-01-10 22:58:58 +01:00
|
|
|
*
|
|
|
|
|
* This class represents an arithmetic progression \f$ a_0, a_1, a_2, ..., a_{n-1}\f$ defined by
|
|
|
|
|
* its \em first value \f$ a_0 \f$, its \em size (aka length) \em n, and the \em increment (aka stride)
|
|
|
|
|
* that is equal to \f$ a_{i+1}-a_{i}\f$ for any \em i.
|
|
|
|
|
*
|
2017-01-11 14:28:28 +01:00
|
|
|
* It is internally used as the return type of the Eigen::seq and Eigen::seqN functions, and as the input arguments
|
2017-01-10 22:58:58 +01:00
|
|
|
* of DenseBase::operator()(const RowIndices&, const ColIndices&), and most of the time this is the
|
|
|
|
|
* only way it is used.
|
|
|
|
|
*
|
|
|
|
|
* \tparam FirstType type of the first element, usually an Index,
|
|
|
|
|
* but internally it can be a symbolic expression
|
|
|
|
|
* \tparam SizeType type representing the size of the sequence, usually an Index
|
|
|
|
|
* or a compile time integral constant. Internally, it can also be a symbolic expression
|
|
|
|
|
* \tparam IncrType type of the increment, can be a runtime Index, or a compile time integral constant (default is compile-time 1)
|
|
|
|
|
*
|
2017-01-11 14:28:28 +01:00
|
|
|
* \sa Eigen::seq, Eigen::seqN, DenseBase::operator()(const RowIndices&, const ColIndices&), class IndexedView
|
2017-01-10 22:58:58 +01:00
|
|
|
*/
|
2017-01-11 14:28:28 +01:00
|
|
|
template<typename FirstType=Index,typename SizeType=Index,typename IncrType=internal::fix_t<1> >
|
2017-01-16 16:20:16 +01:00
|
|
|
class ArithmeticSequence
|
2017-01-09 17:35:21 +01:00
|
|
|
{
|
2017-01-06 00:01:44 +01:00
|
|
|
|
2017-01-09 17:35:21 +01:00
|
|
|
public:
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence(FirstType first, SizeType size) : m_first(first), m_size(size) {}
|
|
|
|
|
ArithmeticSequence(FirstType first, SizeType size, IncrType incr) : m_first(first), m_size(size), m_incr(incr) {}
|
2017-01-06 00:01:44 +01:00
|
|
|
|
2017-01-09 17:35:21 +01:00
|
|
|
enum {
|
2017-01-11 14:28:28 +01:00
|
|
|
SizeAtCompileTime = internal::get_compile_time<SizeType>::value,
|
|
|
|
|
IncrAtCompileTime = internal::get_compile_time<IncrType,DynamicIndex>::value
|
2017-01-09 17:35:21 +01:00
|
|
|
};
|
|
|
|
|
|
2017-01-10 22:58:58 +01:00
|
|
|
/** \returns the size, i.e., number of elements, of the sequence */
|
2017-01-09 17:35:21 +01:00
|
|
|
Index size() const { return m_size; }
|
2017-01-10 22:58:58 +01:00
|
|
|
|
|
|
|
|
/** \returns the first element \f$ a_0 \f$ in the sequence */
|
2017-01-10 14:25:30 +01:00
|
|
|
Index first() const { return m_first; }
|
2017-01-10 22:58:58 +01:00
|
|
|
|
|
|
|
|
/** \returns the value \f$ a_i \f$ at index \a i in the sequence. */
|
2017-01-09 17:35:21 +01:00
|
|
|
Index operator[](Index i) const { return m_first + i * m_incr; }
|
2017-01-06 00:01:44 +01:00
|
|
|
|
2017-01-09 17:35:21 +01:00
|
|
|
const FirstType& firstObject() const { return m_first; }
|
|
|
|
|
const SizeType& sizeObject() const { return m_size; }
|
|
|
|
|
const IncrType& incrObject() const { return m_incr; }
|
|
|
|
|
|
|
|
|
|
protected:
|
2017-01-06 00:01:44 +01:00
|
|
|
FirstType m_first;
|
|
|
|
|
SizeType m_size;
|
2017-01-06 13:10:10 +01:00
|
|
|
IncrType m_incr;
|
2017-01-06 00:01:44 +01:00
|
|
|
};
|
|
|
|
|
|
2017-01-10 11:06:02 +01:00
|
|
|
namespace internal {
|
|
|
|
|
|
2017-01-16 16:17:01 +01:00
|
|
|
// Cleanup return types:
|
|
|
|
|
|
|
|
|
|
// By default, no change:
|
|
|
|
|
template<typename T, int DynamicKey=Dynamic, typename EnableIf=void> struct cleanup_seq_type { typedef T type; };
|
|
|
|
|
|
|
|
|
|
// Convert short, int, unsigned int, etc. to Eigen::Index
|
|
|
|
|
template<typename T, int DynamicKey> struct cleanup_seq_type<T,DynamicKey,typename internal::enable_if<internal::is_integral<T>::value>::type> { typedef Index type; };
|
|
|
|
|
|
|
|
|
|
// In c++98/c++11, fix<N> is a pointer to function that we better cleanup to a true fix_t<N>:
|
|
|
|
|
template<int N, int DynamicKey> struct cleanup_seq_type<fix_t<N> (*)(), DynamicKey> { typedef fix_t<N> type; };
|
|
|
|
|
|
|
|
|
|
// If variable_or_fixed does not match DynamicKey, then we turn it to a pure compile-time value:
|
|
|
|
|
template<int N, int DynamicKey> struct cleanup_seq_type<variable_or_fixed<N>, DynamicKey> { typedef fix_t<N> type; };
|
|
|
|
|
// If variable_or_fixed matches DynamicKey, then we turn it to a pure runtime-value (aka Index):
|
|
|
|
|
template<int DynamicKey> struct cleanup_seq_type<variable_or_fixed<DynamicKey>, DynamicKey> { typedef Index type; };
|
|
|
|
|
|
|
|
|
|
// Helper to cleanup the type of the increment:
|
|
|
|
|
template<typename T> struct cleanup_seq_incr {
|
|
|
|
|
typedef typename cleanup_seq_type<T,DynamicIndex>::type type;
|
|
|
|
|
};
|
2017-01-09 17:35:21 +01:00
|
|
|
|
2017-01-10 11:06:02 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-16 16:20:16 +01:00
|
|
|
/** \returns an ArithmeticSequence starting at \a first, of length \a size, and increment \a incr
|
2017-01-11 14:28:28 +01:00
|
|
|
*
|
|
|
|
|
* \sa seqN(FirstType,SizeType), seq(FirstType,LastType,IncrType) */
|
2017-01-06 13:10:10 +01:00
|
|
|
template<typename FirstType,typename SizeType,typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,typename internal::cleanup_seq_type<SizeType>::type,typename internal::cleanup_seq_incr<IncrType>::type >
|
2017-01-09 17:35:21 +01:00
|
|
|
seqN(FirstType first, SizeType size, IncrType incr) {
|
2017-01-16 16:20:16 +01:00
|
|
|
return ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,typename internal::cleanup_seq_type<SizeType>::type,typename internal::cleanup_seq_incr<IncrType>::type>(first,size,incr);
|
2017-01-06 00:01:44 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-16 16:20:16 +01:00
|
|
|
/** \returns an ArithmeticSequence starting at \a first, of length \a size, and unit increment
|
2017-01-11 14:28:28 +01:00
|
|
|
*
|
2017-01-10 22:58:58 +01:00
|
|
|
* \sa seqN(FirstType,SizeType,IncrType), seq(FirstType,LastType) */
|
2017-01-06 00:01:44 +01:00
|
|
|
template<typename FirstType,typename SizeType>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,typename internal::cleanup_seq_type<SizeType>::type >
|
2017-01-09 17:35:21 +01:00
|
|
|
seqN(FirstType first, SizeType size) {
|
2017-01-16 16:20:16 +01:00
|
|
|
return ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,typename internal::cleanup_seq_type<SizeType>::type>(first,size);
|
2017-01-06 00:01:44 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-10 22:58:58 +01:00
|
|
|
#ifdef EIGEN_PARSED_BY_DOXYGEN
|
|
|
|
|
|
2017-01-16 16:20:16 +01:00
|
|
|
/** \returns an ArithmeticSequence starting at \a f, up (or down) to \a l, and with positive (or negative) increment \a incr
|
2017-01-10 22:58:58 +01:00
|
|
|
*
|
|
|
|
|
* It is essentially an alias to:
|
|
|
|
|
* \code
|
2017-01-11 14:28:28 +01:00
|
|
|
* seqN(f, (l-f+incr)/incr, incr);
|
2017-01-10 22:58:58 +01:00
|
|
|
* \endcode
|
|
|
|
|
*
|
2017-01-11 14:28:28 +01:00
|
|
|
* \sa seqN(FirstType,SizeType,IncrType), seq(FirstType,LastType)
|
2017-01-10 22:58:58 +01:00
|
|
|
*/
|
2017-01-11 14:28:28 +01:00
|
|
|
template<typename FirstType,typename LastType, typename IncrType>
|
|
|
|
|
auto seq(FirstType f, LastType l, IncrType incr);
|
2017-01-10 22:58:58 +01:00
|
|
|
|
2017-01-16 16:20:16 +01:00
|
|
|
/** \returns an ArithmeticSequence starting at \a f, up (or down) to \a l, and unit increment
|
2017-01-10 22:58:58 +01:00
|
|
|
*
|
|
|
|
|
* It is essentially an alias to:
|
|
|
|
|
* \code
|
2017-01-11 14:28:28 +01:00
|
|
|
* seqN(f,l-f+1);
|
2017-01-10 22:58:58 +01:00
|
|
|
* \endcode
|
|
|
|
|
*
|
2017-01-11 14:28:28 +01:00
|
|
|
* \sa seqN(FirstType,SizeType), seq(FirstType,LastType,IncrType)
|
2017-01-10 22:58:58 +01:00
|
|
|
*/
|
2017-01-11 14:28:28 +01:00
|
|
|
template<typename FirstType,typename LastType>
|
|
|
|
|
auto seq(FirstType f, LastType l);
|
2017-01-10 22:58:58 +01:00
|
|
|
|
|
|
|
|
#else // EIGEN_PARSED_BY_DOXYGEN
|
|
|
|
|
|
2017-01-10 10:28:45 +01:00
|
|
|
#if EIGEN_HAS_CXX11
|
2017-01-09 23:42:16 +01:00
|
|
|
template<typename FirstType,typename LastType>
|
2017-01-10 10:28:45 +01:00
|
|
|
auto seq(FirstType f, LastType l) -> decltype(seqN(f,(l-f+fix<1>())))
|
2017-01-09 23:42:16 +01:00
|
|
|
{
|
2017-01-10 10:28:45 +01:00
|
|
|
return seqN(f,(l-f+fix<1>()));
|
2017-01-09 23:42:16 +01:00
|
|
|
}
|
2017-01-06 13:10:10 +01:00
|
|
|
|
2017-01-09 23:42:16 +01:00
|
|
|
template<typename FirstType,typename LastType, typename IncrType>
|
|
|
|
|
auto seq(FirstType f, LastType l, IncrType incr)
|
2017-01-16 16:17:01 +01:00
|
|
|
-> decltype(seqN(f, (l-f+typename internal::cleanup_seq_incr<IncrType>::type(incr))
|
|
|
|
|
/ typename internal::cleanup_seq_incr<IncrType>::type(incr),typename internal::cleanup_seq_incr<IncrType>::type(incr)))
|
2017-01-09 23:42:16 +01:00
|
|
|
{
|
2017-01-16 16:17:01 +01:00
|
|
|
typedef typename internal::cleanup_seq_incr<IncrType>::type CleanedIncrType;
|
2017-01-09 23:42:16 +01:00
|
|
|
return seqN(f,(l-f+CleanedIncrType(incr))/CleanedIncrType(incr),CleanedIncrType(incr));
|
|
|
|
|
}
|
2017-01-10 10:28:45 +01:00
|
|
|
#else
|
|
|
|
|
template<typename FirstType,typename LastType>
|
2017-01-10 10:57:08 +01:00
|
|
|
typename internal::enable_if<!(Symbolic::is_symbolic<FirstType>::value || Symbolic::is_symbolic<LastType>::value),
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,Index> >::type
|
2017-01-10 10:28:45 +01:00
|
|
|
seq(FirstType f, LastType l)
|
|
|
|
|
{
|
|
|
|
|
return seqN(f,(l-f+1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstTypeDerived,typename LastType>
|
2017-01-10 10:57:08 +01:00
|
|
|
typename internal::enable_if<!Symbolic::is_symbolic<LastType>::value,
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<FirstTypeDerived, Symbolic::AddExpr<Symbolic::AddExpr<Symbolic::NegateExpr<FirstTypeDerived>,Symbolic::ValueExpr>,
|
2017-01-10 10:57:08 +01:00
|
|
|
Symbolic::ValueExpr> > >::type
|
|
|
|
|
seq(const Symbolic::BaseExpr<FirstTypeDerived> &f, LastType l)
|
2017-01-10 10:28:45 +01:00
|
|
|
{
|
|
|
|
|
return seqN(f.derived(),(l-f.derived()+1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastTypeDerived>
|
2017-01-10 10:57:08 +01:00
|
|
|
typename internal::enable_if<!Symbolic::is_symbolic<FirstType>::value,
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,
|
2017-01-10 10:57:08 +01:00
|
|
|
Symbolic::AddExpr<Symbolic::AddExpr<LastTypeDerived,Symbolic::ValueExpr>,Symbolic::ValueExpr> > >::type
|
|
|
|
|
seq(FirstType f, const Symbolic::BaseExpr<LastTypeDerived> &l)
|
2017-01-10 10:28:45 +01:00
|
|
|
{
|
|
|
|
|
return seqN(f,(l.derived()-f+1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstTypeDerived,typename LastTypeDerived>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<FirstTypeDerived,
|
2017-01-10 10:57:08 +01:00
|
|
|
Symbolic::AddExpr<Symbolic::AddExpr<LastTypeDerived,Symbolic::NegateExpr<FirstTypeDerived> >,Symbolic::ValueExpr> >
|
|
|
|
|
seq(const Symbolic::BaseExpr<FirstTypeDerived> &f, const Symbolic::BaseExpr<LastTypeDerived> &l)
|
2017-01-10 10:28:45 +01:00
|
|
|
{
|
|
|
|
|
return seqN(f.derived(),(l.derived()-f.derived()+1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastType, typename IncrType>
|
2017-01-10 10:57:08 +01:00
|
|
|
typename internal::enable_if<!(Symbolic::is_symbolic<FirstType>::value || Symbolic::is_symbolic<LastType>::value),
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,Index,typename internal::cleanup_seq_incr<IncrType>::type> >::type
|
2017-01-10 10:28:45 +01:00
|
|
|
seq(FirstType f, LastType l, IncrType incr)
|
|
|
|
|
{
|
2017-01-16 16:17:01 +01:00
|
|
|
typedef typename internal::cleanup_seq_incr<IncrType>::type CleanedIncrType;
|
2017-01-10 10:28:45 +01:00
|
|
|
return seqN(f,(l-f+CleanedIncrType(incr))/CleanedIncrType(incr), incr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstTypeDerived,typename LastType, typename IncrType>
|
2017-01-10 10:57:08 +01:00
|
|
|
typename internal::enable_if<!Symbolic::is_symbolic<LastType>::value,
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<FirstTypeDerived,
|
2017-01-10 10:57:08 +01:00
|
|
|
Symbolic::QuotientExpr<Symbolic::AddExpr<Symbolic::AddExpr<Symbolic::NegateExpr<FirstTypeDerived>,
|
|
|
|
|
Symbolic::ValueExpr>,
|
|
|
|
|
Symbolic::ValueExpr>,
|
|
|
|
|
Symbolic::ValueExpr>,
|
2017-01-16 16:17:01 +01:00
|
|
|
typename internal::cleanup_seq_incr<IncrType>::type> >::type
|
2017-01-10 10:57:08 +01:00
|
|
|
seq(const Symbolic::BaseExpr<FirstTypeDerived> &f, LastType l, IncrType incr)
|
2017-01-10 10:28:45 +01:00
|
|
|
{
|
2017-01-16 16:17:01 +01:00
|
|
|
typedef typename internal::cleanup_seq_incr<IncrType>::type CleanedIncrType;
|
2017-01-10 10:28:45 +01:00
|
|
|
return seqN(f.derived(),(l-f.derived()+CleanedIncrType(incr))/CleanedIncrType(incr), incr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastTypeDerived, typename IncrType>
|
2017-01-10 10:57:08 +01:00
|
|
|
typename internal::enable_if<!Symbolic::is_symbolic<FirstType>::value,
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<typename internal::cleanup_seq_type<FirstType>::type,
|
2017-01-10 10:57:08 +01:00
|
|
|
Symbolic::QuotientExpr<Symbolic::AddExpr<Symbolic::AddExpr<LastTypeDerived,Symbolic::ValueExpr>,
|
|
|
|
|
Symbolic::ValueExpr>,
|
|
|
|
|
Symbolic::ValueExpr>,
|
2017-01-16 16:17:01 +01:00
|
|
|
typename internal::cleanup_seq_incr<IncrType>::type> >::type
|
2017-01-10 10:57:08 +01:00
|
|
|
seq(FirstType f, const Symbolic::BaseExpr<LastTypeDerived> &l, IncrType incr)
|
2017-01-10 10:28:45 +01:00
|
|
|
{
|
2017-01-16 16:17:01 +01:00
|
|
|
typedef typename internal::cleanup_seq_incr<IncrType>::type CleanedIncrType;
|
2017-01-10 10:28:45 +01:00
|
|
|
return seqN(f,(l.derived()-f+CleanedIncrType(incr))/CleanedIncrType(incr), incr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstTypeDerived,typename LastTypeDerived, typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<FirstTypeDerived,
|
2017-01-10 10:57:08 +01:00
|
|
|
Symbolic::QuotientExpr<Symbolic::AddExpr<Symbolic::AddExpr<LastTypeDerived,
|
|
|
|
|
Symbolic::NegateExpr<FirstTypeDerived> >,
|
|
|
|
|
Symbolic::ValueExpr>,
|
|
|
|
|
Symbolic::ValueExpr>,
|
2017-01-16 16:17:01 +01:00
|
|
|
typename internal::cleanup_seq_incr<IncrType>::type>
|
2017-01-10 10:57:08 +01:00
|
|
|
seq(const Symbolic::BaseExpr<FirstTypeDerived> &f, const Symbolic::BaseExpr<LastTypeDerived> &l, IncrType incr)
|
2017-01-10 10:28:45 +01:00
|
|
|
{
|
2017-01-16 16:17:01 +01:00
|
|
|
typedef typename internal::cleanup_seq_incr<IncrType>::type CleanedIncrType;
|
2017-01-10 10:28:45 +01:00
|
|
|
return seqN(f.derived(),(l.derived()-f.derived()+CleanedIncrType(incr))/CleanedIncrType(incr), incr);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2017-01-06 00:01:44 +01:00
|
|
|
|
2017-01-10 22:58:58 +01:00
|
|
|
#endif // EIGEN_PARSED_BY_DOXYGEN
|
|
|
|
|
|
2017-01-06 00:01:44 +01:00
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
|
|
// Replace symbolic last/end "keywords" by their true runtime value
|
2017-01-10 16:32:06 +01:00
|
|
|
inline Index eval_expr_given_size(Index x, Index /* size */) { return x; }
|
2017-01-06 00:01:44 +01:00
|
|
|
|
2017-01-09 23:42:16 +01:00
|
|
|
template<int N>
|
2017-01-10 11:06:02 +01:00
|
|
|
fix_t<N> eval_expr_given_size(fix_t<N> x, Index /*size*/) { return x; }
|
2017-01-09 23:42:16 +01:00
|
|
|
|
|
|
|
|
template<typename Derived>
|
2017-01-10 11:06:02 +01:00
|
|
|
Index eval_expr_given_size(const Symbolic::BaseExpr<Derived> &x, Index size)
|
2017-01-09 23:42:16 +01:00
|
|
|
{
|
2017-01-10 16:45:32 +01:00
|
|
|
return x.derived().eval(placeholders::last=size-1);
|
2017-01-09 23:42:16 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-06 00:01:44 +01:00
|
|
|
// Convert a symbolic span into a usable one (i.e., remove last/end "keywords")
|
2017-01-09 23:42:16 +01:00
|
|
|
template<typename T>
|
|
|
|
|
struct make_size_type {
|
2017-01-10 10:57:08 +01:00
|
|
|
typedef typename internal::conditional<Symbolic::is_symbolic<T>::value, Index, T>::type type;
|
2017-01-09 23:42:16 +01:00
|
|
|
};
|
|
|
|
|
|
2017-01-11 17:24:02 +01:00
|
|
|
template<typename FirstType,typename SizeType,typename IncrType,int XprSize>
|
2017-01-16 16:20:16 +01:00
|
|
|
struct IndexedViewCompatibleType<ArithmeticSequence<FirstType,SizeType,IncrType>, XprSize> {
|
|
|
|
|
typedef ArithmeticSequence<Index,typename make_size_type<SizeType>::type,IncrType> type;
|
2017-01-06 00:01:44 +01:00
|
|
|
};
|
|
|
|
|
|
2017-01-06 13:10:10 +01:00
|
|
|
template<typename FirstType,typename SizeType,typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequence<Index,typename make_size_type<SizeType>::type,IncrType>
|
|
|
|
|
makeIndexedViewCompatible(const ArithmeticSequence<FirstType,SizeType,IncrType>& ids, Index size) {
|
|
|
|
|
return ArithmeticSequence<Index,typename make_size_type<SizeType>::type,IncrType>(
|
2017-01-10 11:06:02 +01:00
|
|
|
eval_expr_given_size(ids.firstObject(),size),eval_expr_given_size(ids.sizeObject(),size),ids.incrObject());
|
2017-01-06 00:01:44 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-11 17:24:02 +01:00
|
|
|
template<typename FirstType,typename SizeType,typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
struct get_compile_time_incr<ArithmeticSequence<FirstType,SizeType,IncrType> > {
|
2017-01-11 17:24:02 +01:00
|
|
|
enum { value = get_compile_time<IncrType,DynamicIndex>::value };
|
2017-01-06 15:54:55 +01:00
|
|
|
};
|
|
|
|
|
|
2017-01-06 00:01:44 +01:00
|
|
|
} // end namespace internal
|
|
|
|
|
|
2017-01-10 09:34:25 +01:00
|
|
|
//--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
namespace legacy {
|
|
|
|
|
// Here are some initial code that I keep here for now to compare the quality of the code generated by the compilers
|
2017-01-11 17:24:02 +01:00
|
|
|
// This part will be removed once we have checked everything is right.
|
2017-01-10 09:34:25 +01:00
|
|
|
|
|
|
|
|
struct shifted_last {
|
|
|
|
|
explicit shifted_last(int o) : offset(o) {}
|
|
|
|
|
int offset;
|
|
|
|
|
shifted_last operator+ (int x) const { return shifted_last(offset+x); }
|
|
|
|
|
shifted_last operator- (int x) const { return shifted_last(offset-x); }
|
|
|
|
|
int operator- (shifted_last x) const { return offset-x.offset; }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct last_t {
|
|
|
|
|
last_t() {}
|
|
|
|
|
shifted_last operator- (int offset) const { return shifted_last(-offset); }
|
|
|
|
|
shifted_last operator+ (int offset) const { return shifted_last(+offset); }
|
|
|
|
|
int operator- (last_t) const { return 0; }
|
|
|
|
|
int operator- (shifted_last x) const { return -x.offset; }
|
|
|
|
|
};
|
|
|
|
|
static const last_t last;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct shifted_end {
|
|
|
|
|
explicit shifted_end(int o) : offset(o) {}
|
|
|
|
|
int offset;
|
|
|
|
|
shifted_end operator+ (int x) const { return shifted_end(offset+x); }
|
|
|
|
|
shifted_end operator- (int x) const { return shifted_end(offset-x); }
|
|
|
|
|
int operator- (shifted_end x) const { return offset-x.offset; }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct end_t {
|
|
|
|
|
end_t() {}
|
|
|
|
|
shifted_end operator- (int offset) const { return shifted_end (-offset); }
|
|
|
|
|
shifted_end operator+ (int offset) const { return shifted_end ( offset); }
|
|
|
|
|
int operator- (end_t) const { return 0; }
|
|
|
|
|
int operator- (shifted_end x) const { return -x.offset; }
|
|
|
|
|
};
|
|
|
|
|
static const end_t end;
|
|
|
|
|
|
2017-01-10 16:32:06 +01:00
|
|
|
inline Index eval_expr_given_size(last_t, Index size) { return size-1; }
|
|
|
|
|
inline Index eval_expr_given_size(shifted_last x, Index size) { return size+x.offset-1; }
|
|
|
|
|
inline Index eval_expr_given_size(end_t, Index size) { return size; }
|
|
|
|
|
inline Index eval_expr_given_size(shifted_end x, Index size) { return size+x.offset; }
|
2017-01-10 09:34:25 +01:00
|
|
|
|
2017-01-11 14:28:28 +01:00
|
|
|
template<typename FirstType=Index,typename LastType=Index,typename IncrType=internal::fix_t<1> >
|
2017-01-16 16:20:16 +01:00
|
|
|
class ArithmeticSequenceProxyWithBounds
|
2017-01-10 09:34:25 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequenceProxyWithBounds(FirstType f, LastType l) : m_first(f), m_last(l) {}
|
|
|
|
|
ArithmeticSequenceProxyWithBounds(FirstType f, LastType l, IncrType s) : m_first(f), m_last(l), m_incr(s) {}
|
2017-01-10 09:34:25 +01:00
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
SizeAtCompileTime = -1,
|
2017-01-11 14:28:28 +01:00
|
|
|
IncrAtCompileTime = internal::get_compile_time<IncrType,DynamicIndex>::value
|
2017-01-10 09:34:25 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Index size() const { return (m_last-m_first+m_incr)/m_incr; }
|
|
|
|
|
Index operator[](Index i) const { return m_first + i * m_incr; }
|
|
|
|
|
|
|
|
|
|
const FirstType& firstObject() const { return m_first; }
|
|
|
|
|
const LastType& lastObject() const { return m_last; }
|
|
|
|
|
const IncrType& incrObject() const { return m_incr; }
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
FirstType m_first;
|
|
|
|
|
LastType m_last;
|
|
|
|
|
IncrType m_incr;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastType>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequenceProxyWithBounds<typename internal::cleanup_seq_type<FirstType>::type,typename internal::cleanup_seq_type<LastType>::type >
|
2017-01-10 09:34:25 +01:00
|
|
|
seq(FirstType f, LastType l) {
|
2017-01-16 16:20:16 +01:00
|
|
|
return ArithmeticSequenceProxyWithBounds<typename internal::cleanup_seq_type<FirstType>::type,typename internal::cleanup_seq_type<LastType>::type>(f,l);
|
2017-01-10 09:34:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastType,typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
ArithmeticSequenceProxyWithBounds< typename internal::cleanup_seq_type<FirstType>::type,
|
2017-01-10 11:06:02 +01:00
|
|
|
typename internal::cleanup_seq_type<LastType>::type,
|
2017-01-16 16:17:01 +01:00
|
|
|
typename internal::cleanup_seq_incr<IncrType>::type >
|
2017-01-10 11:06:02 +01:00
|
|
|
seq(FirstType f, LastType l, IncrType s)
|
|
|
|
|
{
|
2017-01-16 16:20:16 +01:00
|
|
|
return ArithmeticSequenceProxyWithBounds<typename internal::cleanup_seq_type<FirstType>::type,
|
2017-01-10 11:06:02 +01:00
|
|
|
typename internal::cleanup_seq_type<LastType>::type,
|
2017-01-16 16:17:01 +01:00
|
|
|
typename internal::cleanup_seq_incr<IncrType>::type>
|
|
|
|
|
(f,l,typename internal::cleanup_seq_incr<IncrType>::type(s));
|
2017-01-10 09:34:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastType,typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
struct get_compile_time_incr<legacy::ArithmeticSequenceProxyWithBounds<FirstType,LastType,IncrType> > {
|
2017-01-10 09:34:25 +01:00
|
|
|
enum { value = get_compile_time<IncrType,DynamicIndex>::value };
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Convert a symbolic range into a usable one (i.e., remove last/end "keywords")
|
2017-01-11 17:24:02 +01:00
|
|
|
template<typename FirstType,typename LastType,typename IncrType,int XprSize>
|
2017-01-16 16:20:16 +01:00
|
|
|
struct IndexedViewCompatibleType<legacy::ArithmeticSequenceProxyWithBounds<FirstType,LastType,IncrType>,XprSize> {
|
|
|
|
|
typedef legacy::ArithmeticSequenceProxyWithBounds<Index,Index,IncrType> type;
|
2017-01-10 09:34:25 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename FirstType,typename LastType,typename IncrType>
|
2017-01-16 16:20:16 +01:00
|
|
|
legacy::ArithmeticSequenceProxyWithBounds<Index,Index,IncrType>
|
|
|
|
|
makeIndexedViewCompatible(const legacy::ArithmeticSequenceProxyWithBounds<FirstType,LastType,IncrType>& ids, Index size) {
|
|
|
|
|
return legacy::ArithmeticSequenceProxyWithBounds<Index,Index,IncrType>(
|
2017-01-10 11:06:02 +01:00
|
|
|
eval_expr_given_size(ids.firstObject(),size),eval_expr_given_size(ids.lastObject(),size),ids.incrObject());
|
2017-01-10 09:34:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-06 00:01:44 +01:00
|
|
|
} // end namespace Eigen
|
|
|
|
|
|
|
|
|
|
#endif // EIGEN_ARITHMETIC_SEQUENCE_H
|