matio-cpp v0.3.0
A C++ wrapper of the matio library, with memory ownership handling, to read and write .mat files.
Loading...
Searching...
No Matches
EigenConversions.tpp
Go to the documentation of this file.
1/*
2 * Copyright (C) 2022 Fondazione Istituto Italiano di Tecnologia
3 *
4 * This software may be modified and distributed under the terms of the
5 * BSD-2-Clause license (https://opensource.org/licenses/BSD-2-Clause).
6 */
7
8#ifndef EIGENCONVERSIONS_TPP
9#define EIGENCONVERSIONS_TPP
10
11#include <cassert>
12
13namespace matioCpp
14{
16 {
18 size_t offset{ 0 };
19 size_t innerStride{ 0 };
20 size_t outerStride{ 0 };
21 bool valid{ false };
22 };
23
24 template <typename type>
26 {
27 std::string errorPrefix = "[ERROR][matioCpp::to_eigen] ";
28
29 using index_type = typename matioCpp::MultiDimensionalArray<type>::index_type;
31
32 const auto& dimensions = input.dimensions();
33
34 if (slice.size() != dimensions.size())
35 {
36 std::cerr << errorPrefix << "The number of slices must be equal to the number of dimensions of the input MultiDimensionalArray" << std::endl;
37 assert(false);
38 return info;
39 }
40
42 index_type previousDimensionsFactorial = 1;
43
44 for (size_t i = 0; i < slice.size(); ++i)
45 {
46 if (slice[i] >= 0)
47 {
48 if (slice[i] >= dimensions(i))
49 {
50 std::cerr << errorPrefix << "The slice is larger than the dimension of the input MultiDimensionalArray" << std::endl;
51 assert(false);
52 return SlicingInfo();
53 }
54 startingIndex[i] = static_cast<size_t>(slice[i]);
55 }
56 else
57 {
58 if (info.dimensions.first == 0)
59 {
60 info.dimensions.first = dimensions(i);
62 }
63 else if (info.dimensions.second == 0)
64 {
65 info.dimensions.second = dimensions(i);
67 }
68 else
69 {
70 std::cerr << errorPrefix << "Only at most two free dimensions are allowed" << std::endl;
71 assert(false);
72 return SlicingInfo();
73 }
74 }
75 previousDimensionsFactorial *= dimensions(i);
76 }
77
78 if (info.dimensions.first == 0) //Single element
79 {
80 info.dimensions.first = 1;
81 info.innerStride = 1;
82 }
83
84 if (info.dimensions.second == 0) //Vector case
85 {
86 info.dimensions.second = 1;
87 info.outerStride = info.dimensions.first * info.innerStride;
88 }
89
91
92 info.valid = true;
93 return info;
94
95 }
96}
97
98template <typename type>
99inline Eigen::Map<Eigen::Matrix<type, Eigen::Dynamic, Eigen::Dynamic>> matioCpp::to_eigen(matioCpp::MultiDimensionalArray<type>& input)
100{
101 assert(input.isValid());
102 assert(input.dimensions().size() == 2);
103 return Eigen::Map<Eigen::Matrix<type, Eigen::Dynamic, Eigen::Dynamic>>(input.data(), input.dimensions()(0), input.dimensions()(1));
104}
105
106template <typename type>
107inline Eigen::Map<const Eigen::Matrix<type, Eigen::Dynamic, Eigen::Dynamic>> matioCpp::to_eigen(const matioCpp::MultiDimensionalArray<type>& input)
108{
109 assert(input.isValid());
110 assert(input.dimensions().size() == 2);
111 return Eigen::Map<const Eigen::Matrix<type, Eigen::Dynamic, Eigen::Dynamic>>(input.data(), input.dimensions()(0), input.dimensions()(1));
112}
113
114template <typename type>
116{
117 assert(input.isValid());
118
119 auto slicingInfo = computeSlicingInfo(input, slice);
120 Eigen::Stride<Eigen::Dynamic, Eigen::Dynamic> stride(slicingInfo.outerStride, slicingInfo.innerStride);
121 if (!slicingInfo.valid)
122 {
123 return matioCpp::EigenMapWithStride<type>(nullptr, 0, 0, stride);
124 }
125 return matioCpp::EigenMapWithStride<type>(input.data() + slicingInfo.offset, slicingInfo.dimensions.first, slicingInfo.dimensions.second, stride);
126}
127
128template <typename type>
130{
131 assert(input.isValid());
132
133 auto slicingInfo = computeSlicingInfo(input, slice);
134 Eigen::Stride<Eigen::Dynamic, Eigen::Dynamic> stride(slicingInfo.outerStride, slicingInfo.innerStride);
135
136
137 if (!slicingInfo.valid)
138 {
139 return ConstEigenMapWithStride<type>(nullptr, 0, 0, stride);
140 }
141
142 return ConstEigenMapWithStride<type>(input.data() + slicingInfo.offset, slicingInfo.dimensions.first, slicingInfo.dimensions.second, stride);
143}
144
145template <typename type>
146inline Eigen::Map<Eigen::Matrix<type, Eigen::Dynamic, 1>> matioCpp::to_eigen(matioCpp::Vector<type>& input)
147{
148 assert(input.isValid());
149 return Eigen::Map<Eigen::Matrix<type, Eigen::Dynamic, 1>>(input.data(), input.size());
150}
151
152template <typename type>
153inline const Eigen::Map<Eigen::Matrix<type, Eigen::Dynamic, 1>> matioCpp::to_eigen(const matioCpp::Vector<type>& input)
154{
155 assert(input.isValid());
156 return Eigen::Map<const Eigen::Matrix<type, Eigen::Dynamic, 1>>(input.data(), input.size());
157}
158
159template <typename EigenDerived, typename>
160inline matioCpp::MultiDimensionalArray<typename EigenDerived::Scalar> matioCpp::make_variable(const std::string& name, const Eigen::MatrixBase<EigenDerived>& input)
161{
162 matioCpp::MultiDimensionalArray<typename EigenDerived::Scalar> matio(name, {static_cast<size_t>(input.rows()), static_cast<size_t>(input.cols())});
163 matioCpp::to_eigen(matio) = input;
164 return matio;
165}
166
167#endif // EIGENCONVERSIONS_TPP
MultiDimensionalArray is a particular type of Variable specialized for multidimensional arrays of a g...
pointer data()
Direct access to the underlying array.
size_t index_type
Defines how to allocate T.
index_type rawIndexFromIndices(const std::vector< index_type > &el) const
Get the index in the vectorized array corresponding to the provided indices.
matioCpp::Span< const size_t > dimensions() const
Get the dimensions of this object.
Definition Variable.cpp:532
bool isValid() const
Check if the variable is valid.
Definition Variable.cpp:546
T endl(T... args)
SlicingInfo computeSlicingInfo(const matioCpp::MultiDimensionalArray< type > &input, const std::vector< int > &slice)
matioCpp::Vector< typename std::remove_cv_t< typename matioCpp::SpanUtils::container_data< Vector >::type > > make_variable(const std::string &name, const Vector &input)
Conversion from a generic vector to a matioCpp::Vector.
std::pair< size_t, size_t > dimensions