From 5b636ee921775dd0dad362321a1c9b4207511fa3 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Carsten=20Gr=C3=A4ser?= <graeser@dune-project.org>
Date: Mon, 12 Dec 2022 12:22:53 +0100
Subject: [PATCH] Improve specializations of Hybrid::size()

The interfaces tested subsequently for `Hybrid::size(const T& t)` are now

* standard tuple-like interface `std::tuple_size<T>::value` (e.g. needed for `std::tuple`),
* static constexpr `T::size()` (e.g. needed for `Dune::TupleVector`),
* static constexpr `T::N()` (e.g. needed for `Dune::MultiTypeBlockMatrix`),
* dynamic member function `t.N()` (e.g. needed for `Dune::Matrix`),
* dynamic member function `t.size()` (e.g. needed for `std::vector`).

Unfortunately the matrix interface is rather inconsistent.
While `Dune::DenseMatrix` provides a dyanmic `size()` method, the dynamic dense matrix
`Dune::Matrix` does not. The only consistent interface to get the number of rows is the
`N()` method which exists in dynamic (e.g. `Dune::Matrix`) and constexpr static
(e.g. `Dune::MultiTypeBlockMatrix`) flavour.
Hence to support matrices consistently, we must add support for those cases, too.

For the static constexpr `N()`, this was made neccessary, because of the removal of
static consexpr `Dune::MultiTypeBlockMatrix::size()`. The dynamic case should be
added for consistency to also support `Dune::Matrix` and `Dune::BCRSMatrix`.
As a by-product, this allows to get rid of the `FieldMatrix` specialization.
---
 dune/common/hybridutilities.hh | 26 +++++++++++++++++---------
 1 file changed, 17 insertions(+), 9 deletions(-)

diff --git a/dune/common/hybridutilities.hh b/dune/common/hybridutilities.hh
index b40d6e1a2..e0cc906d6 100644
--- a/dune/common/hybridutilities.hh
+++ b/dune/common/hybridutilities.hh
@@ -23,30 +23,38 @@ namespace Hybrid {
 namespace Impl {
 
   // Try if tuple_size is implemented for class
-  template<class T, int i>
-  constexpr auto size(const Dune::FieldVector<T, i>&, const PriorityTag<5>&)
-    -> decltype(std::integral_constant<std::size_t,i>())
+  template<class T>
+  constexpr auto size(const T&, const PriorityTag<4>&)
+    -> decltype(std::integral_constant<std::size_t,std::tuple_size<T>::value>())
   {
     return {};
   }
 
-  // Try if tuple_size is implemented for class
+  // Try if there's a static constexpr size()
   template<class T>
   constexpr auto size(const T&, const PriorityTag<3>&)
-    -> decltype(std::integral_constant<std::size_t,std::tuple_size<T>::value>())
+    -> decltype(std::integral_constant<std::size_t,T::size()>())
   {
     return {};
   }
 
-  // Try if there's a static constexpr size()
+  // Try if there's a static constexpr N()
   template<class T>
-  constexpr auto size(const T&, const PriorityTag<1>&)
-    -> decltype(std::integral_constant<std::size_t,T::size()>())
+  constexpr auto size(const T&, const PriorityTag<2>&)
+    -> decltype(std::integral_constant<std::size_t,T::N()>())
   {
     return {};
   }
 
-  // As a last resort try if there's a static constexpr size()
+  // Try if there's a non-static N()
+  template<class T>
+  constexpr auto size(const T& t, const PriorityTag<1>&)
+    -> decltype(t.N())
+  {
+    return t.N();
+  }
+
+  // As a last resort try if there's a non-static size()
   template<class T>
   constexpr auto size(const T& t, const PriorityTag<0>&)
   {
-- 
GitLab