Program Listing for File EigenDiagonalization.h

Return to documentation for file (include/gamer/EigenDiagonalization.h)

// This file is part of the GAMer software.
// Copyright (C) 2016-2021
// by Christopher T. Lee and contributors
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2.1 of the License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, see <http://www.gnu.org/licenses/>
// or write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
// Boston, MA 02111-1307 USA

#pragma once

#include <Eigen/Dense>
#include <Eigen/Eigenvalues>

#include "gamer.h"

namespace gamer {

template <typename T = REAL, std::size_t dim = 3> class EigenDiagonalizeTraits {
public:
  using CovarianceMatrix = std::array<T, (dim * (dim + 1) / 2)>;

private:
  using _EigenMatrix = Eigen::Matrix<T, dim, dim>;
  using _EigenVector = Eigen::Matrix<T, dim, 1>;

  static _EigenMatrix construct_covariance_matrix(const CovarianceMatrix &cov) {
    _EigenMatrix m;

    for (std::size_t i = 0; i < dim; ++i) {
      for (std::size_t j = i; j < dim; ++j) {
        m(i, j) = static_cast<REAL>(cov[(dim * i) + j - ((i * (i + 1)) / 2)]);

        if (i != j)
          m(j, i) = m(i, j);
      }
    }
    return m;
  }

public:
  static bool diagonalizeSelfAdjointMatrix(const _EigenMatrix &mat,
                                           _EigenVector &eigenvalues,
                                           _EigenMatrix &eigenvectors) {
    Eigen::SelfAdjointEigenSolver<_EigenMatrix> eigensolver;

    if (dim == 2 || dim == 3)
      eigensolver.computeDirect(mat);
    else
      eigensolver.compute(mat); // More accurate but slower

    if (eigensolver.info() != Eigen::Success) {
      return false;
      // std::stringstream ss;
      // ss << "getEigenvalues has encountered Eigen error " <<
      // eigensolver.info();
      // gamer_runtime_error(ss.str());
    }

    eigenvalues = eigensolver.eigenvalues();
    eigenvectors = eigensolver.eigenvectors();
    return true;
  }

  static bool diagonalizeSelfAdjointCovMatrix(const CovarianceMatrix &cov,
                                              _EigenVector &eigenvalues,
                                              _EigenMatrix &eigenvectors) {
    _EigenMatrix m = construct_covariance_matrix(cov);
    bool res = diagonalizeSelfAdjointMatrix(m, eigenvalues, eigenvectors);
    return res;
  }
}; // end class EigenDiagonalizeTraits
} // end namespace gamer