| // This file is part of Eigen, a lightweight C++ template library |
| // for linear algebra. |
| // |
| // Copyright (C) 2024 Charles Schlosser <cs.schlosser@gmail.com> |
| // |
| // 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_FILL_H |
| #define EIGEN_FILL_H |
| |
| // IWYU pragma: private |
| #include "./InternalHeaderCheck.h" |
| |
| namespace Eigen { |
| |
| namespace internal { |
| |
| template <typename Xpr> |
| struct eigen_fill_helper : std::false_type {}; |
| |
| template <typename Scalar, int Rows, int Cols, int Options, int MaxRows, int MaxCols> |
| struct eigen_fill_helper<Matrix<Scalar, Rows, Cols, Options, MaxRows, MaxCols>> : std::true_type {}; |
| |
| template <typename Scalar, int Rows, int Cols, int Options, int MaxRows, int MaxCols> |
| struct eigen_fill_helper<Array<Scalar, Rows, Cols, Options, MaxRows, MaxCols>> : std::true_type {}; |
| |
| template <typename Xpr, int BlockRows, int BlockCols> |
| struct eigen_fill_helper<Block<Xpr, BlockRows, BlockCols, /*InnerPanel*/ true>> : eigen_fill_helper<Xpr> {}; |
| |
| template <typename Xpr, int BlockRows, int BlockCols> |
| struct eigen_fill_helper<Block<Xpr, BlockRows, BlockCols, /*InnerPanel*/ false>> |
| : std::integral_constant<bool, eigen_fill_helper<Xpr>::value && |
| (Xpr::IsRowMajor ? (BlockRows == 1) : (BlockCols == 1))> {}; |
| |
| template <typename Xpr, int Options, typename StrideType> |
| struct eigen_fill_helper<Map<Xpr, Options, StrideType>> |
| : std::integral_constant<bool, eigen_fill_helper<Xpr>::value && |
| (evaluator<Map<Xpr, Options, StrideType>>::Flags & LinearAccessBit)> {}; |
| |
| template <typename Xpr, bool use_fill = eigen_fill_helper<Xpr>::value> |
| struct eigen_fill_impl { |
| using Scalar = typename Xpr::Scalar; |
| using Func = scalar_constant_op<Scalar>; |
| using PlainObject = typename Xpr::PlainObject; |
| using Constant = CwiseNullaryOp<Func, PlainObject>; |
| static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE void run(Xpr& dst, const Scalar& val) { |
| dst = Constant(dst.rows(), dst.cols(), Func(val)); |
| } |
| }; |
| |
| #if !EIGEN_COMP_MSVC |
| #ifndef EIGEN_GPU_COMPILE_PHASE |
| template <typename Xpr> |
| struct eigen_fill_impl<Xpr, /*use_fill*/ true> { |
| using Scalar = typename Xpr::Scalar; |
| static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE void run(Xpr& dst, const Scalar& val) { |
| EIGEN_USING_STD(fill_n); |
| fill_n(dst.data(), dst.size(), val); |
| } |
| }; |
| #endif |
| #endif |
| |
| template <typename Xpr> |
| struct eigen_memset_helper { |
| static constexpr bool value = std::is_trivial<typename Xpr::Scalar>::value && eigen_fill_helper<Xpr>::value; |
| }; |
| |
| template <typename Xpr, bool use_memset = eigen_memset_helper<Xpr>::value> |
| struct eigen_zero_impl { |
| using Scalar = typename Xpr::Scalar; |
| static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE void run(Xpr& dst) { eigen_fill_impl<Xpr, false>::run(dst, Scalar(0)); } |
| }; |
| |
| template <typename Xpr> |
| struct eigen_zero_impl<Xpr, /*use_memset*/ true> { |
| using Scalar = typename Xpr::Scalar; |
| static constexpr size_t max_bytes = (std::numeric_limits<std::ptrdiff_t>::max)(); |
| static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE void run(Xpr& dst) { |
| const size_t num_bytes = dst.size() * sizeof(Scalar); |
| #ifndef EIGEN_NO_DEBUG |
| if (num_bytes > max_bytes) throw_std_bad_alloc(); |
| #endif |
| EIGEN_USING_STD(memset); |
| memset(dst.data(), 0, num_bytes); |
| } |
| }; |
| |
| } // namespace internal |
| } // namespace Eigen |
| |
| #endif // EIGEN_FILL_H |