Reverse.h 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. // This file is part of Eigen, a lightweight C++ template library
  2. // for linear algebra.
  3. //
  4. // Copyright (C) 2006-2008 Benoit Jacob <jacob.benoit.1@gmail.com>
  5. // Copyright (C) 2009 Ricard Marxer <email@ricardmarxer.com>
  6. // Copyright (C) 2009-2010 Gael Guennebaud <gael.guennebaud@inria.fr>
  7. //
  8. // This Source Code Form is subject to the terms of the Mozilla
  9. // Public License v. 2.0. If a copy of the MPL was not distributed
  10. // with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
  11. #ifndef EIGEN_REVERSE_H
  12. #define EIGEN_REVERSE_H
  13. namespace Eigen {
  14. namespace internal {
  15. template<typename MatrixType, int Direction>
  16. struct traits<Reverse<MatrixType, Direction> >
  17. : traits<MatrixType>
  18. {
  19. typedef typename MatrixType::Scalar Scalar;
  20. typedef typename traits<MatrixType>::StorageKind StorageKind;
  21. typedef typename traits<MatrixType>::XprKind XprKind;
  22. typedef typename ref_selector<MatrixType>::type MatrixTypeNested;
  23. typedef typename remove_reference<MatrixTypeNested>::type _MatrixTypeNested;
  24. enum {
  25. RowsAtCompileTime = MatrixType::RowsAtCompileTime,
  26. ColsAtCompileTime = MatrixType::ColsAtCompileTime,
  27. MaxRowsAtCompileTime = MatrixType::MaxRowsAtCompileTime,
  28. MaxColsAtCompileTime = MatrixType::MaxColsAtCompileTime,
  29. Flags = _MatrixTypeNested::Flags & (RowMajorBit | LvalueBit)
  30. };
  31. };
  32. template<typename PacketType, bool ReversePacket> struct reverse_packet_cond
  33. {
  34. static inline PacketType run(const PacketType& x) { return preverse(x); }
  35. };
  36. template<typename PacketType> struct reverse_packet_cond<PacketType,false>
  37. {
  38. static inline PacketType run(const PacketType& x) { return x; }
  39. };
  40. } // end namespace internal
  41. /** \class Reverse
  42. * \ingroup Core_Module
  43. *
  44. * \brief Expression of the reverse of a vector or matrix
  45. *
  46. * \tparam MatrixType the type of the object of which we are taking the reverse
  47. * \tparam Direction defines the direction of the reverse operation, can be Vertical, Horizontal, or BothDirections
  48. *
  49. * This class represents an expression of the reverse of a vector.
  50. * It is the return type of MatrixBase::reverse() and VectorwiseOp::reverse()
  51. * and most of the time this is the only way it is used.
  52. *
  53. * \sa MatrixBase::reverse(), VectorwiseOp::reverse()
  54. */
  55. template<typename MatrixType, int Direction> class Reverse
  56. : public internal::dense_xpr_base< Reverse<MatrixType, Direction> >::type
  57. {
  58. public:
  59. typedef typename internal::dense_xpr_base<Reverse>::type Base;
  60. EIGEN_DENSE_PUBLIC_INTERFACE(Reverse)
  61. typedef typename internal::remove_all<MatrixType>::type NestedExpression;
  62. using Base::IsRowMajor;
  63. protected:
  64. enum {
  65. PacketSize = internal::packet_traits<Scalar>::size,
  66. IsColMajor = !IsRowMajor,
  67. ReverseRow = (Direction == Vertical) || (Direction == BothDirections),
  68. ReverseCol = (Direction == Horizontal) || (Direction == BothDirections),
  69. OffsetRow = ReverseRow && IsColMajor ? PacketSize : 1,
  70. OffsetCol = ReverseCol && IsRowMajor ? PacketSize : 1,
  71. ReversePacket = (Direction == BothDirections)
  72. || ((Direction == Vertical) && IsColMajor)
  73. || ((Direction == Horizontal) && IsRowMajor)
  74. };
  75. typedef internal::reverse_packet_cond<PacketScalar,ReversePacket> reverse_packet;
  76. public:
  77. EIGEN_DEVICE_FUNC explicit inline Reverse(const MatrixType& matrix) : m_matrix(matrix) { }
  78. EIGEN_INHERIT_ASSIGNMENT_OPERATORS(Reverse)
  79. EIGEN_DEVICE_FUNC inline Index rows() const { return m_matrix.rows(); }
  80. EIGEN_DEVICE_FUNC inline Index cols() const { return m_matrix.cols(); }
  81. EIGEN_DEVICE_FUNC inline Index innerStride() const
  82. {
  83. return -m_matrix.innerStride();
  84. }
  85. EIGEN_DEVICE_FUNC const typename internal::remove_all<typename MatrixType::Nested>::type&
  86. nestedExpression() const
  87. {
  88. return m_matrix;
  89. }
  90. protected:
  91. typename MatrixType::Nested m_matrix;
  92. };
  93. /** \returns an expression of the reverse of *this.
  94. *
  95. * Example: \include MatrixBase_reverse.cpp
  96. * Output: \verbinclude MatrixBase_reverse.out
  97. *
  98. */
  99. template<typename Derived>
  100. inline typename DenseBase<Derived>::ReverseReturnType
  101. DenseBase<Derived>::reverse()
  102. {
  103. return ReverseReturnType(derived());
  104. }
  105. //reverse const overload moved DenseBase.h due to a CUDA compiler bug
  106. /** This is the "in place" version of reverse: it reverses \c *this.
  107. *
  108. * In most cases it is probably better to simply use the reversed expression
  109. * of a matrix. However, when reversing the matrix data itself is really needed,
  110. * then this "in-place" version is probably the right choice because it provides
  111. * the following additional benefits:
  112. * - less error prone: doing the same operation with .reverse() requires special care:
  113. * \code m = m.reverse().eval(); \endcode
  114. * - this API enables reverse operations without the need for a temporary
  115. * - it allows future optimizations (cache friendliness, etc.)
  116. *
  117. * \sa VectorwiseOp::reverseInPlace(), reverse() */
  118. template<typename Derived>
  119. inline void DenseBase<Derived>::reverseInPlace()
  120. {
  121. if(cols()>rows())
  122. {
  123. Index half = cols()/2;
  124. leftCols(half).swap(rightCols(half).reverse());
  125. if((cols()%2)==1)
  126. {
  127. Index half2 = rows()/2;
  128. col(half).head(half2).swap(col(half).tail(half2).reverse());
  129. }
  130. }
  131. else
  132. {
  133. Index half = rows()/2;
  134. topRows(half).swap(bottomRows(half).reverse());
  135. if((rows()%2)==1)
  136. {
  137. Index half2 = cols()/2;
  138. row(half).head(half2).swap(row(half).tail(half2).reverse());
  139. }
  140. }
  141. }
  142. namespace internal {
  143. template<int Direction>
  144. struct vectorwise_reverse_inplace_impl;
  145. template<>
  146. struct vectorwise_reverse_inplace_impl<Vertical>
  147. {
  148. template<typename ExpressionType>
  149. static void run(ExpressionType &xpr)
  150. {
  151. Index half = xpr.rows()/2;
  152. xpr.topRows(half).swap(xpr.bottomRows(half).colwise().reverse());
  153. }
  154. };
  155. template<>
  156. struct vectorwise_reverse_inplace_impl<Horizontal>
  157. {
  158. template<typename ExpressionType>
  159. static void run(ExpressionType &xpr)
  160. {
  161. Index half = xpr.cols()/2;
  162. xpr.leftCols(half).swap(xpr.rightCols(half).rowwise().reverse());
  163. }
  164. };
  165. } // end namespace internal
  166. /** This is the "in place" version of VectorwiseOp::reverse: it reverses each column or row of \c *this.
  167. *
  168. * In most cases it is probably better to simply use the reversed expression
  169. * of a matrix. However, when reversing the matrix data itself is really needed,
  170. * then this "in-place" version is probably the right choice because it provides
  171. * the following additional benefits:
  172. * - less error prone: doing the same operation with .reverse() requires special care:
  173. * \code m = m.reverse().eval(); \endcode
  174. * - this API enables reverse operations without the need for a temporary
  175. *
  176. * \sa DenseBase::reverseInPlace(), reverse() */
  177. template<typename ExpressionType, int Direction>
  178. void VectorwiseOp<ExpressionType,Direction>::reverseInPlace()
  179. {
  180. internal::vectorwise_reverse_inplace_impl<Direction>::run(_expression().const_cast_derived());
  181. }
  182. } // end namespace Eigen
  183. #endif // EIGEN_REVERSE_H