Forked from
ogs / ogs
25910 commits behind the upstream repository.
-
Norihiro Watanabe authoredNorihiro Watanabe authored
Code owners
Assign users and groups as approvers for specific file changes. Learn more.
LisLinearSystem.cpp 6.14 KiB
/**
* Copyright (c) 2012, OpenGeoSys Community (http://www.opengeosys.org)
* Distributed under a Modified BSD License.
* See accompanying file LICENSE.txt or
* http://www.opengeosys.org/project/license
*
*
* \file LisLinearSystem.cpp
*
* Created on 2012-06-25 by Norihiro Watanabe
*/
#include "LisLinearSystem.h"
#include <string>
#include <sstream>
#ifdef _OPENMP
#include <omp.h>
#endif
#include "logog/include/logog.hpp"
namespace MathLib
{
bool LisLinearSystem::checkError(int err)
{
bool ok = (err == LIS_SUCCESS);
if (!ok) {
ERR("***ERROR: Lis error code = %d", err);
}
return ok;
}
LisLinearSystem::LisLinearSystem(std::size_t dimension)
: _dim(dimension), _max_diag_coeff(.0)
{
int ierr = 0;
ierr = lis_matrix_create(0, &_AA); checkError(ierr);
ierr = lis_matrix_set_size(_AA, 0, dimension); checkError(ierr);
ierr = lis_vector_duplicate(_AA, &_bb); checkError(ierr);
ierr = lis_vector_duplicate(_AA, &_xx); checkError(ierr);
}
LisLinearSystem::~LisLinearSystem()
{
int ierr = 0;
ierr = lis_matrix_destroy(_AA); checkError(ierr);
ierr = lis_vector_destroy(_bb); checkError(ierr);
ierr = lis_vector_destroy(_xx); checkError(ierr);
}
void LisLinearSystem::setOption(const boost::property_tree::ptree &option)
{
using boost::property_tree::ptree;
boost::optional<ptree> ptSolver = option.get_child("LinearSolver");
if (!ptSolver)
return;
boost::optional<std::string> solver_type = ptSolver->get_optional<std::string>("solver_type");
if (solver_type) {
_option.solver_type = _option.getSolverType(*solver_type);
}
boost::optional<std::string> precon_type = ptSolver->get_optional<std::string>("precon_type");
if (precon_type) {
_option.precon_type = _option.getPreconType(*precon_type);
}
boost::optional<std::string> matrix_type = ptSolver->get_optional<std::string>("matrix_type");
if (matrix_type) {
_option.matrix_type = _option.getMatrixType(*matrix_type);
}
boost::optional<double> error_tolerance = ptSolver->get_optional<double>("error_tolerance");
if (error_tolerance) {
_option.error_tolerance = *error_tolerance;
}
boost::optional<int> max_iteration_step = ptSolver->get_optional<int>("max_iteration_step");
if (max_iteration_step) {
_option.max_iterations = *max_iteration_step;
}
}
void LisLinearSystem::setZero()
{
int ierr = 0;
// A matrix has to be removed and created every time because Lis doesn't provide a
// function to set matrix entries to zero
ierr = lis_matrix_destroy(_AA); checkError(ierr);
ierr = lis_matrix_create(0, &_AA); checkError(ierr);
ierr = lis_matrix_set_size(_AA, 0, getDimension()); checkError(ierr);
// set zero RHS, x
ierr = lis_vector_set_all(0.0, _bb); checkError(ierr);
ierr = lis_vector_set_all(0.0, _xx); checkError(ierr);
_max_diag_coeff = .0;
}
void LisLinearSystem::applyKnownSolution()
{
//Use penalty parameter
const double penalty_scaling = 1e+10;
const double penalty = _max_diag_coeff * penalty_scaling;
INFO("-> max. absolute value of diagonal entries = %e", _max_diag_coeff);
INFO("-> penalty scaling = %e", penalty_scaling);
const std::size_t n_bc = _vec_knownX_id.size();
for (std::size_t i_bc=0; i_bc<n_bc; i_bc++) {
const std::size_t rowId = _vec_knownX_id[i_bc];
const double x = _vec_knownX_x[i_bc];
//A(k, k) = penalty
setMatEntry(rowId, rowId, penalty);
//b(k) = x*penalty
setRHSVec(rowId, x*penalty);
}
}
void LisLinearSystem::solve()
{
INFO("------------------------------------------------------------------");
INFO("*** LIS solver computation");
#ifdef _OPENMP
INFO("-> max number of threads = %d", omp_get_num_procs());
INFO("-> number of threads = %d", omp_get_max_threads());
#endif
applyKnownSolution();
// assemble a matrix
int ierr = 0;
ierr = lis_matrix_set_type(_AA, static_cast<int>(_option.matrix_type)); checkError(ierr);
ierr = lis_matrix_assemble(_AA); checkError(ierr);
// configure option
std::string solver_options;
if (_option.solver_precon_arg.empty()) {
std::stringstream ss;
ss << "-i " << static_cast<int>(_option.solver_type);
ss << " -p " << static_cast<int>(_option.precon_type);
if (!_option.extra_arg.empty())
ss << " " << _option.extra_arg;
solver_options = ss.str();
} else {
solver_options = _option.solver_precon_arg;
}
std::string tol_option;
{
std::stringstream ss;
ss << "-tol " << _option.error_tolerance;
ss << " -maxiter " << _option.max_iterations;
ss << " -initx_zeros 0"; //0: use given x as initial guess, 1: x0=0
#ifdef _OPENMP
const int nthreads = omp_get_max_threads();
ss << " -omp_num_threads " << nthreads;
#endif
tol_option = ss.str();
}
// Create solver
LIS_SOLVER solver;
ierr = lis_solver_create(&solver); checkError(ierr);
ierr = lis_solver_set_option(const_cast<char*>(solver_options.c_str()), solver); checkError(ierr);
ierr = lis_solver_set_option(const_cast<char*>(tol_option.c_str()), solver); checkError(ierr);
ierr = lis_solver_set_option(const_cast<char*>("-print mem"), solver); checkError(ierr);
// solve
INFO("-> solve");
ierr = lis_solve(_AA, _bb, _xx, solver); checkError(ierr);
checkError(ierr);
//lis_output(_AA, _bb, _xx, LIS_FMT_MM, "/home/localadmin/tasks/20120814_ogs6test/matrix1.txt");
int iter = 0;
double resid = 0.0;
ierr = lis_solver_get_iters(solver, &iter); checkError(ierr);
ierr = lis_solver_get_residualnorm(solver, &resid); checkError(ierr);
INFO("\t iteration: %d/%ld\n", iter, _option.max_iterations);
INFO("\t residuals: %e\n", resid);
// Clear solver
ierr = lis_solver_destroy(solver); checkError(ierr);
INFO("------------------------------------------------------------------");
}
void LisLinearSystem::printout(std::ostream &os) const
{
os << "#A=" << std::endl;
os << "#x=" << std::endl;
lis_vector_print(_xx);
os << "#b=" << std::endl;
lis_vector_print(_bb);
}
} //MathLib