diff --git a/dune/xt/common/fmatrix.hh b/dune/xt/common/fmatrix.hh
index 65c21efbb17f0eb96ace6752fc43d77ea6e13435..92efd3b3e28c3c46c104303223622249b2abc306 100644
--- a/dune/xt/common/fmatrix.hh
+++ b/dune/xt/common/fmatrix.hh
@@ -386,6 +386,16 @@ struct MatrixAbstraction<Dune::XT::Common::FieldMatrix<K, N, M>>
     return MatrixType(rows, cols, val);
   }
 
+  static inline std::unique_ptr<MatrixType> create_dynamic(const size_t rows, const size_t cols)
+  {
+    return std::make_unique<MatrixType>(rows, cols);
+  }
+
+  static inline std::unique_ptr<MatrixType> create_dynamic(const size_t rows, const size_t cols, const ScalarType& val)
+  {
+    return std::make_unique<MatrixType>(rows, cols, val);
+  }
+
   static inline size_t rows(const MatrixType& /*mat*/)
   {
     return N;
@@ -575,6 +585,19 @@ Dune::XT::Common::FieldMatrix<K, L_ROWS, R_COLS> operator*(const Dune::FieldMatr
   return left.rightmultiplyany(right);
 }
 
+template <class K, int L_ROWS, int L_COLS, int R_COLS>
+void rightmultiply(Dune::FieldMatrix<K, L_ROWS, R_COLS>& ret,
+                   const Dune::FieldMatrix<K, L_ROWS, L_COLS>& left,
+                   const Dune::FieldMatrix<K, L_COLS, R_COLS>& right)
+{
+  for (size_t ii = 0; ii < L_ROWS; ++ii) {
+    for (size_t jj = 0; jj < R_COLS; ++jj) {
+      ret[ii][jj] = 0.;
+      for (size_t kk = 0; kk < L_COLS; ++kk)
+        ret[ii][jj] += left[ii][kk] * right[kk][jj];
+    }
+  }
+}
 
 template <class L, int L_ROWS, int L_COLS, class R, int R_COLS>
 typename std::enable_if<!std::is_same<L, R>::value,
@@ -587,6 +610,35 @@ typename std::enable_if<!std::is_same<L, R>::value,
   return convert_to<Promoted>(left).rightmultiplyany(convert_to<Promoted>(right));
 }
 
+// versions that do not allocate matrices on the stack (for large matrices)
+template <class K, int L_ROWS, int L_COLS, int R_COLS>
+std::unique_ptr<Dune::XT::Common::FieldMatrix<K, L_ROWS, R_COLS>>
+operator*(const std::unique_ptr<Dune::FieldMatrix<K, L_ROWS, L_COLS>>& left,
+          const Dune::FieldMatrix<K, L_COLS, R_COLS>& right)
+{
+  auto ret = std::make_unique<Dune::XT::Common::FieldMatrix<K, L_ROWS, R_COLS>>();
+  rightmultiply(*ret, *left, right);
+  return ret;
+}
+
+template <class K, int L_ROWS, int L_COLS, int R_COLS>
+std::unique_ptr<Dune::XT::Common::FieldMatrix<K, L_ROWS, R_COLS>>
+operator*(const Dune::FieldMatrix<K, L_ROWS, L_COLS>& left,
+          const std::unique_ptr<Dune::FieldMatrix<K, L_COLS, R_COLS>>& right)
+{
+  auto ret = std::make_unique<Dune::XT::Common::FieldMatrix<K, L_ROWS, R_COLS>>();
+  rightmultiply(*ret, left, *right);
+  return ret;
+}
+
+template <class K, int L_ROWS, int L_COLS, int R_COLS>
+std::unique_ptr<Dune::XT::Common::FieldMatrix<K, L_ROWS, R_COLS>>
+operator*(const std::unique_ptr<Dune::FieldMatrix<K, L_ROWS, L_COLS>>& left,
+          const std::unique_ptr<Dune::FieldMatrix<K, L_COLS, R_COLS>>& right)
+{
+  return left * *right;
+}
+
 
 } // namespace Dune
 
diff --git a/dune/xt/common/lapacke.cc b/dune/xt/common/lapacke.cc
index ffc053a7a2392414b4166316f9f2bfbc5cf12154..7cd9c8710fc26a55da32aad574ac155dd1213f8d 100644
--- a/dune/xt/common/lapacke.cc
+++ b/dune/xt/common/lapacke.cc
@@ -320,21 +320,25 @@ void dtrsm(const int layout,
            const int ldb)
 {
 #if HAVE_MKL
-  return cblas_dtrsm(static_cast<CBLAS_LAYOUT>(layout),
-                     static_cast<CBLAS_SIDE>(side),
-                     static_cast<CBLAS_UPLO>(uplo),
-                     static_cast<CBLAS_TRANSPOSE>(transa),
-                     static_cast<CBLAS_DIAG>(diag),
-                     m,
-                     n,
-                     alpha,
-                     a,
-                     lda,
-                     b,
-                     ldb);
+  cblas_dtrsm(static_cast<CBLAS_LAYOUT>(layout),
+              static_cast<CBLAS_SIDE>(side),
+              static_cast<CBLAS_UPLO>(uplo),
+              static_cast<CBLAS_TRANSPOSE>(transa),
+              static_cast<CBLAS_DIAG>(diag),
+              m,
+              n,
+              alpha,
+              a,
+              lda,
+              b,
+              ldb);
+#ifndef NDEBUG
+  for (size_t ii = 0; ii < m; ++ii)
+    if (std::isnan(b[ii]) || std::isinf(b[ii]))
+      DUNE_THROW(Dune::MathError, "Triangular solve using cblas_dtrsm failed!");
+#endif
 #else
   DUNE_THROW(Exceptions::dependency_missing, "You are missing CBLAS or the intel mkl, check available() first!");
-  return 1;
 #endif
 }
 
diff --git a/dune/xt/common/matrix.hh b/dune/xt/common/matrix.hh
index 71252c19eadfad219ffe0595f962f8dde8dcc3ef..5aee8b0e1478d5d7a6393ef7ce03ec5490896f15 100644
--- a/dune/xt/common/matrix.hh
+++ b/dune/xt/common/matrix.hh
@@ -61,6 +61,12 @@ struct MatrixAbstraction
     static_assert(AlwaysFalse<MatType>::value, "Do not call me if is_matrix is false!");
   }
 
+  static inline /*std::unique_ptr<MatrixType>*/ void
+  create_dynamic(const size_t /*rows*/, const size_t /*cols*/, const ScalarType& /*val*/)
+  {
+    static_assert(AlwaysFalse<MatType>::value, "Do not call me if is_matrix is false!");
+  }
+
   static inline /*size_t*/ void rows(const MatrixType& /*mat*/)
   {
     static_assert(AlwaysFalse<MatType>::value, "Do not call me if is_matrix is false!");
@@ -119,6 +125,16 @@ struct MatrixAbstraction<Dune::DynamicMatrix<K>>
     return MatrixType(rows, cols, val);
   }
 
+  static inline std::unique_ptr<MatrixType> create_dynamic(const size_t rows, const size_t cols)
+  {
+    return std::make_unique<MatrixType>(rows, cols);
+  }
+
+  static inline std::unique_ptr<MatrixType> create_dynamic(const size_t rows, const size_t cols, const ScalarType& val)
+  {
+    return std::make_unique<MatrixType>(rows, cols, val);
+  }
+
   static inline size_t rows(const MatrixType& mat)
   {
     return mat.rows();
@@ -175,6 +191,24 @@ struct MatrixAbstraction<Dune::FieldMatrix<K, N, M>>
     return MatrixType(val);
   }
 
+  static inline std::unique_ptr<MatrixType> create_dynamic(const size_t rows, const size_t cols)
+  {
+    if (rows != N)
+      DUNE_THROW(Exceptions::shapes_do_not_match, "rows = " << rows << "\nN = " << int(N));
+    if (cols != M)
+      DUNE_THROW(Exceptions::shapes_do_not_match, "cols = " << cols << "\nM = " << int(M));
+    return std::make_unique<MatrixType>();
+  }
+
+  static inline std::unique_ptr<MatrixType> create_dynamic(const size_t rows, const size_t cols, const ScalarType& val)
+  {
+    if (rows != N)
+      DUNE_THROW(Exceptions::shapes_do_not_match, "rows = " << rows << "\nN = " << int(N));
+    if (cols != M)
+      DUNE_THROW(Exceptions::shapes_do_not_match, "cols = " << cols << "\nM = " << int(M));
+    return std::make_unique<MatrixType>(val);
+  }
+
   static inline size_t rows(const MatrixType& /*mat*/)
   {
     return numeric_cast<size_t>(N);