Loading [MathJax]/extensions/tex2jax.js
IODA
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
Eigen_Compat.h
Go to the documentation of this file.
1 #pragma once
2 /*
3  * (C) Copyright 2020-2021 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  */
8 /*! \addtogroup ioda_cxx_api
9  *
10  * @{
11  * \file Eigen_Compat.h
12  * \brief Convenience functions to work with Eigen objects.
13  */
14 #include <type_traits>
15 #include <vector>
16 
17 #include "ioda/defs.h"
18 
19 // Eigen causes static analysis tools to emit lots of warnings.
20 #ifdef _MSC_FULL_VER
21 # pragma warning(push)
22 # pragma warning(disable : 26450)
23 # pragma warning(disable : 26495)
24 # pragma warning(disable : 26812)
25 #endif
26 #include "Eigen/Dense"
27 #include "unsupported/Eigen/CXX11/Tensor"
28 #ifdef _MSC_FULL_VER
29 # pragma warning(pop)
30 #endif
31 #include "ioda/Misc/Dimensions.h"
32 
33 /// See the [Eigen3 project](http://eigen.tuxfamily.org/) for stuff in this namespace.
34 namespace Eigen {
35 // Forward declaration in case Eigen/Dense can not be included.
36 template <class T>
38 } // namespace Eigen
39 namespace ioda {
40 /// \brief Do we want to auto-resize the Eigen object on read to fit the data being read?
41 enum class ioda_Eigen_Resize {
42  Resize, ///< Yes
43  No_Resize ///< No
44 };
45 
46 namespace detail {
47 /// Functions to work with Eigen.
48 namespace EigenCompat {
49 // This can resize
50 // template<typename Derived>
51 // class Eigen::PlainObjectBase< Derived >;
52 // Other objects can not resize
53 template <class EigenClass, class ResizeableBase = ::Eigen::PlainObjectBase<EigenClass>>
54 using CanResize = ::std::is_base_of<ResizeableBase, EigenClass>;
55 
56 template <class EigenClass>
57 typename ::std::enable_if<CanResize<EigenClass>::value>::type DoEigenResize(EigenClass& e,
58  ::Eigen::Index rows,
59  ::Eigen::Index cols) {
60  e.resize(rows, cols);
61 }
62 
63 /// \todo Make a static_assert!
64 template <class EigenClass>
65 typename ::std::enable_if<!CanResize<EigenClass>::value>::type DoEigenResize(EigenClass&,
66  ::Eigen::Index,
67  ::Eigen::Index) {
68  throw;
69 }
70 
71 template <class EigenClass>
73  const int numDims = e.NumDimensions;
74  const auto& dims = e.dimensions();
75 
76  Dimensions res;
77  std::vector<Dimensions_t> hdims;
78  Dimensions_t sz = (numDims > 0) ? 1 : 0;
79  for (int i = 0; i < numDims; ++i) {
80  auto val = gsl::narrow<Dimensions_t>(dims[i]);
81  hdims.push_back(val);
82  sz *= val;
83  }
84  res.dimsCur = hdims;
85  res.dimsMax = hdims;
86  res.numElements = sz;
87  res.dimensionality = gsl::narrow<Dimensions_t>(numDims);
88  return res;
89 }
90 } // namespace EigenCompat
91 } // namespace detail
92 } // namespace ioda
93 
94 /// @}
Describe the dimensions of a ioda::Attribute or ioda::Variable.
Common preprocessor definitions used throughout IODA.
See the Eigen3 project for stuff in this namespace.
Definition: Eigen_Compat.h:34
::std::is_base_of< ResizeableBase, EigenClass > CanResize
Definition: Eigen_Compat.h:54
Dimensions getTensorDimensions(EigenClass &e)
Definition: Eigen_Compat.h:72
typename ::std::enable_if< CanResize< EigenClass >::value >::type DoEigenResize(EigenClass &e, ::Eigen::Index rows, ::Eigen::Index cols)
Definition: Eigen_Compat.h:57
ioda_Eigen_Resize
Do we want to auto-resize the Eigen object on read to fit the data being read?
Definition: Eigen_Compat.h:41
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Definition: Dimensions.h:23
Dimensions_t numElements
Definition: Dimensions.h:26
Dimensions_t dimensionality
The dimensionality (rank) of the data.
Definition: Dimensions.h:25
std::vector< Dimensions_t > dimsMax
This must always equal dimsCur for Attribute.
Definition: Dimensions.h:24