From eaebb7ea729057f7f8553bd9e6513964e9295590 Mon Sep 17 00:00:00 2001 From: ManifoldFR <wilson.jallet@polytechnique.org> Date: Fri, 25 Nov 2022 16:36:03 +0100 Subject: [PATCH] eigen-allocator: refactor check of pyArray memory layout vs desired output layout eigen-allocator: try a fix for mutable Eigen::Ref from Python array with "wrong" layout * trick transposes pyArray to get the right layout :) --- include/eigenpy/eigen-allocator.hpp | 41 ++++++++++++++++++----------- unittest/python/test_std_vector.py | 3 ++- 2 files changed, 27 insertions(+), 17 deletions(-) diff --git a/include/eigenpy/eigen-allocator.hpp b/include/eigenpy/eigen-allocator.hpp index 3cde4dd4..1c4f8a64 100644 --- a/include/eigenpy/eigen-allocator.hpp +++ b/include/eigenpy/eigen-allocator.hpp @@ -232,6 +232,22 @@ struct EigenAllocator { }; #if EIGEN_VERSION_AT_LEAST(3, 2, 0) +/// @brief Check if we need to allocate @tparam MatType to convert @param +/// pyArray. +/// @details do not allocate if: +/// want row-major & data C-contiguous OR +/// want col-major & data F-contiguous OR +/// you want a compile-time vector +/// in these cases, data layout fits desired view layout +template <typename MatType> +inline bool is_arr_layout_compatible_with_mat_type(PyArrayObject *pyArray) { + bool is_array_C_cont = PyArray_IS_C_CONTIGUOUS(pyArray); + bool is_array_F_cont = PyArray_IS_F_CONTIGUOUS(pyArray); + return (MatType::IsRowMajor && is_array_C_cont) || + (!MatType::IsRowMajor && is_array_F_cont) || + MatType::IsVectorAtCompileTime; +} + template <typename MatType, int Options, typename Stride> struct EigenAllocator<Eigen::Ref<MatType, Options, Stride> > { typedef Eigen::Ref<MatType, Options, Stride> RefType; @@ -255,14 +271,8 @@ struct EigenAllocator<Eigen::Ref<MatType, Options, Stride> > { const int pyArray_type_code = EIGENPY_GET_PY_ARRAY_TYPE(pyArray); const int Scalar_type_code = Register::getTypeCode<Scalar>(); if (pyArray_type_code != Scalar_type_code) need_to_allocate |= true; - if ((MatType::IsRowMajor && (PyArray_IS_C_CONTIGUOUS(pyArray) && - !PyArray_IS_F_CONTIGUOUS(pyArray))) || - (!MatType::IsRowMajor && (PyArray_IS_F_CONTIGUOUS(pyArray) && - !PyArray_IS_C_CONTIGUOUS(pyArray))) || - MatType::IsVectorAtCompileTime || - (PyArray_IS_F_CONTIGUOUS(pyArray) && - PyArray_IS_C_CONTIGUOUS(pyArray))) // no need to allocate - need_to_allocate |= false; + if (is_arr_layout_compatible_with_mat_type<MatType>(pyArray)) + need_to_allocate |= false; // no need to allocate else need_to_allocate |= true; if (Options != @@ -276,6 +286,10 @@ struct EigenAllocator<Eigen::Ref<MatType, Options, Stride> > { void *raw_ptr = storage->storage.bytes; if (need_to_allocate) { + PyObject *pyarr_transposed = PyArray_Transpose(pyArray, NULL); + allocate((PyArrayObject *)pyarr_transposed, storage); + /* + std::cout << " doing allocate\n"; MatType *mat_ptr; mat_ptr = details::init_matrix_or_array<MatType>::run(pyArray); RefType mat_ref(*mat_ptr); @@ -326,6 +340,7 @@ struct EigenAllocator<Eigen::Ref<MatType, Options, Stride> > { throw Exception( "You asked for a conversion which is not implemented."); } + */ } else { assert(pyArray_type_code == Scalar_type_code); typename NumpyMap<MatType, Scalar, Options, NumpyMapStride>::EigenMap @@ -365,14 +380,8 @@ struct EigenAllocator<const Eigen::Ref<const MatType, Options, Stride> > { const int Scalar_type_code = Register::getTypeCode<Scalar>(); if (pyArray_type_code != Scalar_type_code) need_to_allocate |= true; - if ((MatType::IsRowMajor && (PyArray_IS_C_CONTIGUOUS(pyArray) && - !PyArray_IS_F_CONTIGUOUS(pyArray))) || - (!MatType::IsRowMajor && (PyArray_IS_F_CONTIGUOUS(pyArray) && - !PyArray_IS_C_CONTIGUOUS(pyArray))) || - MatType::IsVectorAtCompileTime || - (PyArray_IS_F_CONTIGUOUS(pyArray) && - PyArray_IS_C_CONTIGUOUS(pyArray))) // no need to allocate - need_to_allocate |= false; + if (is_arr_layout_compatible_with_mat_type<MatType>(pyArray)) + need_to_allocate |= false; // no need to allocate else need_to_allocate |= true; if (Options != diff --git a/unittest/python/test_std_vector.py b/unittest/python/test_std_vector.py index 7f7c99a5..53121ec5 100644 --- a/unittest/python/test_std_vector.py +++ b/unittest/python/test_std_vector.py @@ -76,9 +76,10 @@ print("-----------------") print("l3:") vector.setZero(l3) pprint.pp(list(l3)) -# checkZero(l3) +checkZero(l3) print("-----------------") print("l4:") vector.setZero(l4) pprint.pp(list(l4)) +checkZero(l4) -- GitLab