Home | History | Annotate | Download | only in Core
      1 // This file is part of Eigen, a lightweight C++ template library
      2 // for linear algebra.
      3 //
      4 // Copyright (C) 2007 Michael Olbrich <michael.olbrich (at) gmx.net>
      5 // Copyright (C) 2006-2010 Benoit Jacob <jacob.benoit.1 (at) gmail.com>
      6 // Copyright (C) 2008 Gael Guennebaud <gael.guennebaud (at) 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 
     12 #ifndef EIGEN_ASSIGN_H
     13 #define EIGEN_ASSIGN_H
     14 
     15 namespace Eigen {
     16 
     17 namespace internal {
     18 
     19 /***************************************************************************
     20 * Part 1 : the logic deciding a strategy for traversal and unrolling       *
     21 ***************************************************************************/
     22 
     23 template <typename Derived, typename OtherDerived>
     24 struct assign_traits
     25 {
     26 public:
     27   enum {
     28     DstIsAligned = Derived::Flags & AlignedBit,
     29     DstHasDirectAccess = Derived::Flags & DirectAccessBit,
     30     SrcIsAligned = OtherDerived::Flags & AlignedBit,
     31     JointAlignment = bool(DstIsAligned) && bool(SrcIsAligned) ? Aligned : Unaligned
     32   };
     33 
     34 private:
     35   enum {
     36     InnerSize = int(Derived::IsVectorAtCompileTime) ? int(Derived::SizeAtCompileTime)
     37               : int(Derived::Flags)&RowMajorBit ? int(Derived::ColsAtCompileTime)
     38               : int(Derived::RowsAtCompileTime),
     39     InnerMaxSize = int(Derived::IsVectorAtCompileTime) ? int(Derived::MaxSizeAtCompileTime)
     40               : int(Derived::Flags)&RowMajorBit ? int(Derived::MaxColsAtCompileTime)
     41               : int(Derived::MaxRowsAtCompileTime),
     42     MaxSizeAtCompileTime = Derived::SizeAtCompileTime,
     43     PacketSize = packet_traits<typename Derived::Scalar>::size
     44   };
     45 
     46   enum {
     47     StorageOrdersAgree = (int(Derived::IsRowMajor) == int(OtherDerived::IsRowMajor)),
     48     MightVectorize = StorageOrdersAgree
     49                   && (int(Derived::Flags) & int(OtherDerived::Flags) & ActualPacketAccessBit),
     50     MayInnerVectorize  = MightVectorize && int(InnerSize)!=Dynamic && int(InnerSize)%int(PacketSize)==0
     51                        && int(DstIsAligned) && int(SrcIsAligned),
     52     MayLinearize = StorageOrdersAgree && (int(Derived::Flags) & int(OtherDerived::Flags) & LinearAccessBit),
     53     MayLinearVectorize = MightVectorize && MayLinearize && DstHasDirectAccess
     54                        && (DstIsAligned || MaxSizeAtCompileTime == Dynamic),
     55       /* If the destination isn't aligned, we have to do runtime checks and we don't unroll,
     56          so it's only good for large enough sizes. */
     57     MaySliceVectorize  = MightVectorize && DstHasDirectAccess
     58                        && (int(InnerMaxSize)==Dynamic || int(InnerMaxSize)>=3*PacketSize)
     59       /* slice vectorization can be slow, so we only want it if the slices are big, which is
     60          indicated by InnerMaxSize rather than InnerSize, think of the case of a dynamic block
     61          in a fixed-size matrix */
     62   };
     63 
     64 public:
     65   enum {
     66     Traversal = int(MayInnerVectorize)  ? int(InnerVectorizedTraversal)
     67               : int(MayLinearVectorize) ? int(LinearVectorizedTraversal)
     68               : int(MaySliceVectorize)  ? int(SliceVectorizedTraversal)
     69               : int(MayLinearize)       ? int(LinearTraversal)
     70                                         : int(DefaultTraversal),
     71     Vectorized = int(Traversal) == InnerVectorizedTraversal
     72               || int(Traversal) == LinearVectorizedTraversal
     73               || int(Traversal) == SliceVectorizedTraversal
     74   };
     75 
     76 private:
     77   enum {
     78     UnrollingLimit      = EIGEN_UNROLLING_LIMIT * (Vectorized ? int(PacketSize) : 1),
     79     MayUnrollCompletely = int(Derived::SizeAtCompileTime) != Dynamic
     80                        && int(OtherDerived::CoeffReadCost) != Dynamic
     81                        && int(Derived::SizeAtCompileTime) * int(OtherDerived::CoeffReadCost) <= int(UnrollingLimit),
     82     MayUnrollInner      = int(InnerSize) != Dynamic
     83                        && int(OtherDerived::CoeffReadCost) != Dynamic
     84                        && int(InnerSize) * int(OtherDerived::CoeffReadCost) <= int(UnrollingLimit)
     85   };
     86 
     87 public:
     88   enum {
     89     Unrolling = (int(Traversal) == int(InnerVectorizedTraversal) || int(Traversal) == int(DefaultTraversal))
     90                 ? (
     91                     int(MayUnrollCompletely) ? int(CompleteUnrolling)
     92                   : int(MayUnrollInner)      ? int(InnerUnrolling)
     93                                              : int(NoUnrolling)
     94                   )
     95               : int(Traversal) == int(LinearVectorizedTraversal)
     96                 ? ( bool(MayUnrollCompletely) && bool(DstIsAligned) ? int(CompleteUnrolling) : int(NoUnrolling) )
     97               : int(Traversal) == int(LinearTraversal)
     98                 ? ( bool(MayUnrollCompletely) ? int(CompleteUnrolling) : int(NoUnrolling) )
     99               : int(NoUnrolling)
    100   };
    101 
    102 #ifdef EIGEN_DEBUG_ASSIGN
    103   static void debug()
    104   {
    105     EIGEN_DEBUG_VAR(DstIsAligned)
    106     EIGEN_DEBUG_VAR(SrcIsAligned)
    107     EIGEN_DEBUG_VAR(JointAlignment)
    108     EIGEN_DEBUG_VAR(InnerSize)
    109     EIGEN_DEBUG_VAR(InnerMaxSize)
    110     EIGEN_DEBUG_VAR(PacketSize)
    111     EIGEN_DEBUG_VAR(StorageOrdersAgree)
    112     EIGEN_DEBUG_VAR(MightVectorize)
    113     EIGEN_DEBUG_VAR(MayLinearize)
    114     EIGEN_DEBUG_VAR(MayInnerVectorize)
    115     EIGEN_DEBUG_VAR(MayLinearVectorize)
    116     EIGEN_DEBUG_VAR(MaySliceVectorize)
    117     EIGEN_DEBUG_VAR(Traversal)
    118     EIGEN_DEBUG_VAR(UnrollingLimit)
    119     EIGEN_DEBUG_VAR(MayUnrollCompletely)
    120     EIGEN_DEBUG_VAR(MayUnrollInner)
    121     EIGEN_DEBUG_VAR(Unrolling)
    122   }
    123 #endif
    124 };
    125 
    126 /***************************************************************************
    127 * Part 2 : meta-unrollers
    128 ***************************************************************************/
    129 
    130 /************************
    131 *** Default traversal ***
    132 ************************/
    133 
    134 template<typename Derived1, typename Derived2, int Index, int Stop>
    135 struct assign_DefaultTraversal_CompleteUnrolling
    136 {
    137   enum {
    138     outer = Index / Derived1::InnerSizeAtCompileTime,
    139     inner = Index % Derived1::InnerSizeAtCompileTime
    140   };
    141 
    142   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    143   {
    144     dst.copyCoeffByOuterInner(outer, inner, src);
    145     assign_DefaultTraversal_CompleteUnrolling<Derived1, Derived2, Index+1, Stop>::run(dst, src);
    146   }
    147 };
    148 
    149 template<typename Derived1, typename Derived2, int Stop>
    150 struct assign_DefaultTraversal_CompleteUnrolling<Derived1, Derived2, Stop, Stop>
    151 {
    152   static EIGEN_STRONG_INLINE void run(Derived1 &, const Derived2 &) {}
    153 };
    154 
    155 template<typename Derived1, typename Derived2, int Index, int Stop>
    156 struct assign_DefaultTraversal_InnerUnrolling
    157 {
    158   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src, typename Derived1::Index outer)
    159   {
    160     dst.copyCoeffByOuterInner(outer, Index, src);
    161     assign_DefaultTraversal_InnerUnrolling<Derived1, Derived2, Index+1, Stop>::run(dst, src, outer);
    162   }
    163 };
    164 
    165 template<typename Derived1, typename Derived2, int Stop>
    166 struct assign_DefaultTraversal_InnerUnrolling<Derived1, Derived2, Stop, Stop>
    167 {
    168   static EIGEN_STRONG_INLINE void run(Derived1 &, const Derived2 &, typename Derived1::Index) {}
    169 };
    170 
    171 /***********************
    172 *** Linear traversal ***
    173 ***********************/
    174 
    175 template<typename Derived1, typename Derived2, int Index, int Stop>
    176 struct assign_LinearTraversal_CompleteUnrolling
    177 {
    178   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    179   {
    180     dst.copyCoeff(Index, src);
    181     assign_LinearTraversal_CompleteUnrolling<Derived1, Derived2, Index+1, Stop>::run(dst, src);
    182   }
    183 };
    184 
    185 template<typename Derived1, typename Derived2, int Stop>
    186 struct assign_LinearTraversal_CompleteUnrolling<Derived1, Derived2, Stop, Stop>
    187 {
    188   static EIGEN_STRONG_INLINE void run(Derived1 &, const Derived2 &) {}
    189 };
    190 
    191 /**************************
    192 *** Inner vectorization ***
    193 **************************/
    194 
    195 template<typename Derived1, typename Derived2, int Index, int Stop>
    196 struct assign_innervec_CompleteUnrolling
    197 {
    198   enum {
    199     outer = Index / Derived1::InnerSizeAtCompileTime,
    200     inner = Index % Derived1::InnerSizeAtCompileTime,
    201     JointAlignment = assign_traits<Derived1,Derived2>::JointAlignment
    202   };
    203 
    204   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    205   {
    206     dst.template copyPacketByOuterInner<Derived2, Aligned, JointAlignment>(outer, inner, src);
    207     assign_innervec_CompleteUnrolling<Derived1, Derived2,
    208       Index+packet_traits<typename Derived1::Scalar>::size, Stop>::run(dst, src);
    209   }
    210 };
    211 
    212 template<typename Derived1, typename Derived2, int Stop>
    213 struct assign_innervec_CompleteUnrolling<Derived1, Derived2, Stop, Stop>
    214 {
    215   static EIGEN_STRONG_INLINE void run(Derived1 &, const Derived2 &) {}
    216 };
    217 
    218 template<typename Derived1, typename Derived2, int Index, int Stop>
    219 struct assign_innervec_InnerUnrolling
    220 {
    221   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src, typename Derived1::Index outer)
    222   {
    223     dst.template copyPacketByOuterInner<Derived2, Aligned, Aligned>(outer, Index, src);
    224     assign_innervec_InnerUnrolling<Derived1, Derived2,
    225       Index+packet_traits<typename Derived1::Scalar>::size, Stop>::run(dst, src, outer);
    226   }
    227 };
    228 
    229 template<typename Derived1, typename Derived2, int Stop>
    230 struct assign_innervec_InnerUnrolling<Derived1, Derived2, Stop, Stop>
    231 {
    232   static EIGEN_STRONG_INLINE void run(Derived1 &, const Derived2 &, typename Derived1::Index) {}
    233 };
    234 
    235 /***************************************************************************
    236 * Part 3 : implementation of all cases
    237 ***************************************************************************/
    238 
    239 template<typename Derived1, typename Derived2,
    240          int Traversal = assign_traits<Derived1, Derived2>::Traversal,
    241          int Unrolling = assign_traits<Derived1, Derived2>::Unrolling,
    242          int Version = Specialized>
    243 struct assign_impl;
    244 
    245 /************************
    246 *** Default traversal ***
    247 ************************/
    248 
    249 template<typename Derived1, typename Derived2, int Unrolling, int Version>
    250 struct assign_impl<Derived1, Derived2, InvalidTraversal, Unrolling, Version>
    251 {
    252   static inline void run(Derived1 &, const Derived2 &) { }
    253 };
    254 
    255 template<typename Derived1, typename Derived2, int Version>
    256 struct assign_impl<Derived1, Derived2, DefaultTraversal, NoUnrolling, Version>
    257 {
    258   typedef typename Derived1::Index Index;
    259   static inline void run(Derived1 &dst, const Derived2 &src)
    260   {
    261     const Index innerSize = dst.innerSize();
    262     const Index outerSize = dst.outerSize();
    263     for(Index outer = 0; outer < outerSize; ++outer)
    264       for(Index inner = 0; inner < innerSize; ++inner)
    265         dst.copyCoeffByOuterInner(outer, inner, src);
    266   }
    267 };
    268 
    269 template<typename Derived1, typename Derived2, int Version>
    270 struct assign_impl<Derived1, Derived2, DefaultTraversal, CompleteUnrolling, Version>
    271 {
    272   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    273   {
    274     assign_DefaultTraversal_CompleteUnrolling<Derived1, Derived2, 0, Derived1::SizeAtCompileTime>
    275       ::run(dst, src);
    276   }
    277 };
    278 
    279 template<typename Derived1, typename Derived2, int Version>
    280 struct assign_impl<Derived1, Derived2, DefaultTraversal, InnerUnrolling, Version>
    281 {
    282   typedef typename Derived1::Index Index;
    283   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    284   {
    285     const Index outerSize = dst.outerSize();
    286     for(Index outer = 0; outer < outerSize; ++outer)
    287       assign_DefaultTraversal_InnerUnrolling<Derived1, Derived2, 0, Derived1::InnerSizeAtCompileTime>
    288         ::run(dst, src, outer);
    289   }
    290 };
    291 
    292 /***********************
    293 *** Linear traversal ***
    294 ***********************/
    295 
    296 template<typename Derived1, typename Derived2, int Version>
    297 struct assign_impl<Derived1, Derived2, LinearTraversal, NoUnrolling, Version>
    298 {
    299   typedef typename Derived1::Index Index;
    300   static inline void run(Derived1 &dst, const Derived2 &src)
    301   {
    302     const Index size = dst.size();
    303     for(Index i = 0; i < size; ++i)
    304       dst.copyCoeff(i, src);
    305   }
    306 };
    307 
    308 template<typename Derived1, typename Derived2, int Version>
    309 struct assign_impl<Derived1, Derived2, LinearTraversal, CompleteUnrolling, Version>
    310 {
    311   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    312   {
    313     assign_LinearTraversal_CompleteUnrolling<Derived1, Derived2, 0, Derived1::SizeAtCompileTime>
    314       ::run(dst, src);
    315   }
    316 };
    317 
    318 /**************************
    319 *** Inner vectorization ***
    320 **************************/
    321 
    322 template<typename Derived1, typename Derived2, int Version>
    323 struct assign_impl<Derived1, Derived2, InnerVectorizedTraversal, NoUnrolling, Version>
    324 {
    325   typedef typename Derived1::Index Index;
    326   static inline void run(Derived1 &dst, const Derived2 &src)
    327   {
    328     const Index innerSize = dst.innerSize();
    329     const Index outerSize = dst.outerSize();
    330     const Index packetSize = packet_traits<typename Derived1::Scalar>::size;
    331     for(Index outer = 0; outer < outerSize; ++outer)
    332       for(Index inner = 0; inner < innerSize; inner+=packetSize)
    333         dst.template copyPacketByOuterInner<Derived2, Aligned, Aligned>(outer, inner, src);
    334   }
    335 };
    336 
    337 template<typename Derived1, typename Derived2, int Version>
    338 struct assign_impl<Derived1, Derived2, InnerVectorizedTraversal, CompleteUnrolling, Version>
    339 {
    340   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    341   {
    342     assign_innervec_CompleteUnrolling<Derived1, Derived2, 0, Derived1::SizeAtCompileTime>
    343       ::run(dst, src);
    344   }
    345 };
    346 
    347 template<typename Derived1, typename Derived2, int Version>
    348 struct assign_impl<Derived1, Derived2, InnerVectorizedTraversal, InnerUnrolling, Version>
    349 {
    350   typedef typename Derived1::Index Index;
    351   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    352   {
    353     const Index outerSize = dst.outerSize();
    354     for(Index outer = 0; outer < outerSize; ++outer)
    355       assign_innervec_InnerUnrolling<Derived1, Derived2, 0, Derived1::InnerSizeAtCompileTime>
    356         ::run(dst, src, outer);
    357   }
    358 };
    359 
    360 /***************************
    361 *** Linear vectorization ***
    362 ***************************/
    363 
    364 template <bool IsAligned = false>
    365 struct unaligned_assign_impl
    366 {
    367   template <typename Derived, typename OtherDerived>
    368   static EIGEN_STRONG_INLINE void run(const Derived&, OtherDerived&, typename Derived::Index, typename Derived::Index) {}
    369 };
    370 
    371 template <>
    372 struct unaligned_assign_impl<false>
    373 {
    374   // MSVC must not inline this functions. If it does, it fails to optimize the
    375   // packet access path.
    376 #ifdef _MSC_VER
    377   template <typename Derived, typename OtherDerived>
    378   static EIGEN_DONT_INLINE void run(const Derived& src, OtherDerived& dst, typename Derived::Index start, typename Derived::Index end)
    379 #else
    380   template <typename Derived, typename OtherDerived>
    381   static EIGEN_STRONG_INLINE void run(const Derived& src, OtherDerived& dst, typename Derived::Index start, typename Derived::Index end)
    382 #endif
    383   {
    384     for (typename Derived::Index index = start; index < end; ++index)
    385       dst.copyCoeff(index, src);
    386   }
    387 };
    388 
    389 template<typename Derived1, typename Derived2, int Version>
    390 struct assign_impl<Derived1, Derived2, LinearVectorizedTraversal, NoUnrolling, Version>
    391 {
    392   typedef typename Derived1::Index Index;
    393   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    394   {
    395     const Index size = dst.size();
    396     typedef packet_traits<typename Derived1::Scalar> PacketTraits;
    397     enum {
    398       packetSize = PacketTraits::size,
    399       dstAlignment = PacketTraits::AlignedOnScalar ? Aligned : int(assign_traits<Derived1,Derived2>::DstIsAligned) ,
    400       srcAlignment = assign_traits<Derived1,Derived2>::JointAlignment
    401     };
    402     const Index alignedStart = assign_traits<Derived1,Derived2>::DstIsAligned ? 0
    403                              : internal::first_aligned(&dst.coeffRef(0), size);
    404     const Index alignedEnd = alignedStart + ((size-alignedStart)/packetSize)*packetSize;
    405 
    406     unaligned_assign_impl<assign_traits<Derived1,Derived2>::DstIsAligned!=0>::run(src,dst,0,alignedStart);
    407 
    408     for(Index index = alignedStart; index < alignedEnd; index += packetSize)
    409     {
    410       dst.template copyPacket<Derived2, dstAlignment, srcAlignment>(index, src);
    411     }
    412 
    413     unaligned_assign_impl<>::run(src,dst,alignedEnd,size);
    414   }
    415 };
    416 
    417 template<typename Derived1, typename Derived2, int Version>
    418 struct assign_impl<Derived1, Derived2, LinearVectorizedTraversal, CompleteUnrolling, Version>
    419 {
    420   typedef typename Derived1::Index Index;
    421   static EIGEN_STRONG_INLINE void run(Derived1 &dst, const Derived2 &src)
    422   {
    423     enum { size = Derived1::SizeAtCompileTime,
    424            packetSize = packet_traits<typename Derived1::Scalar>::size,
    425            alignedSize = (size/packetSize)*packetSize };
    426 
    427     assign_innervec_CompleteUnrolling<Derived1, Derived2, 0, alignedSize>::run(dst, src);
    428     assign_DefaultTraversal_CompleteUnrolling<Derived1, Derived2, alignedSize, size>::run(dst, src);
    429   }
    430 };
    431 
    432 /**************************
    433 *** Slice vectorization ***
    434 ***************************/
    435 
    436 template<typename Derived1, typename Derived2, int Version>
    437 struct assign_impl<Derived1, Derived2, SliceVectorizedTraversal, NoUnrolling, Version>
    438 {
    439   typedef typename Derived1::Index Index;
    440   static inline void run(Derived1 &dst, const Derived2 &src)
    441   {
    442     typedef typename Derived1::Scalar Scalar;
    443     typedef packet_traits<Scalar> PacketTraits;
    444     enum {
    445       packetSize = PacketTraits::size,
    446       alignable = PacketTraits::AlignedOnScalar,
    447       dstIsAligned = assign_traits<Derived1,Derived2>::DstIsAligned,
    448       dstAlignment = alignable ? Aligned : int(dstIsAligned),
    449       srcAlignment = assign_traits<Derived1,Derived2>::JointAlignment
    450     };
    451     const Scalar *dst_ptr = &dst.coeffRef(0,0);
    452     if((!bool(dstIsAligned)) && (Index(dst_ptr) % sizeof(Scalar))>0)
    453     {
    454       // the pointer is not aligend-on scalar, so alignment is not possible
    455       return assign_impl<Derived1,Derived2,DefaultTraversal,NoUnrolling>::run(dst, src);
    456     }
    457     const Index packetAlignedMask = packetSize - 1;
    458     const Index innerSize = dst.innerSize();
    459     const Index outerSize = dst.outerSize();
    460     const Index alignedStep = alignable ? (packetSize - dst.outerStride() % packetSize) & packetAlignedMask : 0;
    461     Index alignedStart = ((!alignable) || bool(dstIsAligned)) ? 0 : internal::first_aligned(dst_ptr, innerSize);
    462 
    463     for(Index outer = 0; outer < outerSize; ++outer)
    464     {
    465       const Index alignedEnd = alignedStart + ((innerSize-alignedStart) & ~packetAlignedMask);
    466       // do the non-vectorizable part of the assignment
    467       for(Index inner = 0; inner<alignedStart ; ++inner)
    468         dst.copyCoeffByOuterInner(outer, inner, src);
    469 
    470       // do the vectorizable part of the assignment
    471       for(Index inner = alignedStart; inner<alignedEnd; inner+=packetSize)
    472         dst.template copyPacketByOuterInner<Derived2, dstAlignment, Unaligned>(outer, inner, src);
    473 
    474       // do the non-vectorizable part of the assignment
    475       for(Index inner = alignedEnd; inner<innerSize ; ++inner)
    476         dst.copyCoeffByOuterInner(outer, inner, src);
    477 
    478       alignedStart = std::min<Index>((alignedStart+alignedStep)%packetSize, innerSize);
    479     }
    480   }
    481 };
    482 
    483 } // end namespace internal
    484 
    485 /***************************************************************************
    486 * Part 4 : implementation of DenseBase methods
    487 ***************************************************************************/
    488 
    489 template<typename Derived>
    490 template<typename OtherDerived>
    491 EIGEN_STRONG_INLINE Derived& DenseBase<Derived>
    492   ::lazyAssign(const DenseBase<OtherDerived>& other)
    493 {
    494   enum{
    495     SameType = internal::is_same<typename Derived::Scalar,typename OtherDerived::Scalar>::value
    496   };
    497 
    498   EIGEN_STATIC_ASSERT_LVALUE(Derived)
    499   EIGEN_STATIC_ASSERT_SAME_MATRIX_SIZE(Derived,OtherDerived)
    500   EIGEN_STATIC_ASSERT(SameType,YOU_MIXED_DIFFERENT_NUMERIC_TYPES__YOU_NEED_TO_USE_THE_CAST_METHOD_OF_MATRIXBASE_TO_CAST_NUMERIC_TYPES_EXPLICITLY)
    501 
    502 #ifdef EIGEN_DEBUG_ASSIGN
    503   internal::assign_traits<Derived, OtherDerived>::debug();
    504 #endif
    505   eigen_assert(rows() == other.rows() && cols() == other.cols());
    506   internal::assign_impl<Derived, OtherDerived, int(SameType) ? int(internal::assign_traits<Derived, OtherDerived>::Traversal)
    507                                                        : int(InvalidTraversal)>::run(derived(),other.derived());
    508 #ifndef EIGEN_NO_DEBUG
    509   checkTransposeAliasing(other.derived());
    510 #endif
    511   return derived();
    512 }
    513 
    514 namespace internal {
    515 
    516 template<typename Derived, typename OtherDerived,
    517          bool EvalBeforeAssigning = (int(internal::traits<OtherDerived>::Flags) & EvalBeforeAssigningBit) != 0,
    518          bool NeedToTranspose = ((int(Derived::RowsAtCompileTime) == 1 && int(OtherDerived::ColsAtCompileTime) == 1)
    519                               |   // FIXME | instead of || to please GCC 4.4.0 stupid warning "suggest parentheses around &&".
    520                                   // revert to || as soon as not needed anymore.
    521                                   (int(Derived::ColsAtCompileTime) == 1 && int(OtherDerived::RowsAtCompileTime) == 1))
    522                               && int(Derived::SizeAtCompileTime) != 1>
    523 struct assign_selector;
    524 
    525 template<typename Derived, typename OtherDerived>
    526 struct assign_selector<Derived,OtherDerived,false,false> {
    527   static EIGEN_STRONG_INLINE Derived& run(Derived& dst, const OtherDerived& other) { return dst.lazyAssign(other.derived()); }
    528   template<typename ActualDerived, typename ActualOtherDerived>
    529   static EIGEN_STRONG_INLINE Derived& evalTo(ActualDerived& dst, const ActualOtherDerived& other) { other.evalTo(dst); return dst; }
    530 };
    531 template<typename Derived, typename OtherDerived>
    532 struct assign_selector<Derived,OtherDerived,true,false> {
    533   static EIGEN_STRONG_INLINE Derived& run(Derived& dst, const OtherDerived& other) { return dst.lazyAssign(other.eval()); }
    534 };
    535 template<typename Derived, typename OtherDerived>
    536 struct assign_selector<Derived,OtherDerived,false,true> {
    537   static EIGEN_STRONG_INLINE Derived& run(Derived& dst, const OtherDerived& other) { return dst.lazyAssign(other.transpose()); }
    538   template<typename ActualDerived, typename ActualOtherDerived>
    539   static EIGEN_STRONG_INLINE Derived& evalTo(ActualDerived& dst, const ActualOtherDerived& other) { Transpose<ActualDerived> dstTrans(dst); other.evalTo(dstTrans); return dst; }
    540 };
    541 template<typename Derived, typename OtherDerived>
    542 struct assign_selector<Derived,OtherDerived,true,true> {
    543   static EIGEN_STRONG_INLINE Derived& run(Derived& dst, const OtherDerived& other) { return dst.lazyAssign(other.transpose().eval()); }
    544 };
    545 
    546 } // end namespace internal
    547 
    548 template<typename Derived>
    549 template<typename OtherDerived>
    550 EIGEN_STRONG_INLINE Derived& DenseBase<Derived>::operator=(const DenseBase<OtherDerived>& other)
    551 {
    552   return internal::assign_selector<Derived,OtherDerived>::run(derived(), other.derived());
    553 }
    554 
    555 template<typename Derived>
    556 EIGEN_STRONG_INLINE Derived& DenseBase<Derived>::operator=(const DenseBase& other)
    557 {
    558   return internal::assign_selector<Derived,Derived>::run(derived(), other.derived());
    559 }
    560 
    561 template<typename Derived>
    562 EIGEN_STRONG_INLINE Derived& MatrixBase<Derived>::operator=(const MatrixBase& other)
    563 {
    564   return internal::assign_selector<Derived,Derived>::run(derived(), other.derived());
    565 }
    566 
    567 template<typename Derived>
    568 template <typename OtherDerived>
    569 EIGEN_STRONG_INLINE Derived& MatrixBase<Derived>::operator=(const DenseBase<OtherDerived>& other)
    570 {
    571   return internal::assign_selector<Derived,OtherDerived>::run(derived(), other.derived());
    572 }
    573 
    574 template<typename Derived>
    575 template <typename OtherDerived>
    576 EIGEN_STRONG_INLINE Derived& MatrixBase<Derived>::operator=(const EigenBase<OtherDerived>& other)
    577 {
    578   return internal::assign_selector<Derived,OtherDerived,false>::evalTo(derived(), other.derived());
    579 }
    580 
    581 template<typename Derived>
    582 template<typename OtherDerived>
    583 EIGEN_STRONG_INLINE Derived& MatrixBase<Derived>::operator=(const ReturnByValue<OtherDerived>& other)
    584 {
    585   return internal::assign_selector<Derived,OtherDerived,false>::evalTo(derived(), other.derived());
    586 }
    587 
    588 } // end namespace Eigen
    589 
    590 #endif // EIGEN_ASSIGN_H
    591