From d235d9b21a1c54acde0c1515a01ccc82c53ade79 Mon Sep 17 00:00:00 2001 From: Jason Landini Date: Fri, 18 Oct 2024 15:59:53 -0400 Subject: [PATCH 1/2] refactor implicit update --- applications/allenCahn_conserved/customPDE.h | 274 +--------- include/matrixFreePDE.h | 4 + src/matrixfree/solveIncrement.cc | 505 +++++++++---------- 3 files changed, 260 insertions(+), 523 deletions(-) diff --git a/applications/allenCahn_conserved/customPDE.h b/applications/allenCahn_conserved/customPDE.h index c6b7505d6..8b3e9ad8a 100644 --- a/applications/allenCahn_conserved/customPDE.h +++ b/applications/allenCahn_conserved/customPDE.h @@ -211,277 +211,11 @@ customPDE::solveIncrement(bool skip_time_dependent) this->pcout << buffer; } - LinearAlgebra::distributed::Vector solution_diff = - *this->solutionSet[fieldIndex]; - - // apply Dirichlet BC's - // This clears the residual where we want to apply Dirichlet - // BCs, otherwise the solver sees a positive residual - this->constraintsDirichletSet[fieldIndex]->set_zero( - *this->residualSet[fieldIndex]); - - // solver controls - double tol_value; - if (MatrixFreePDE::userInputs.linear_solver_parameters - .getToleranceType(fieldIndex) == ABSOLUTE_RESIDUAL) - { - tol_value = - MatrixFreePDE::userInputs.linear_solver_parameters - .getToleranceValue(fieldIndex); - } - else - { - tol_value = - MatrixFreePDE::userInputs.linear_solver_parameters - .getToleranceValue(fieldIndex) * - this->residualSet[fieldIndex]->l2_norm(); - } - - SolverControl solver_control( - MatrixFreePDE::userInputs.linear_solver_parameters - .getMaxIterations(fieldIndex), - tol_value); - - // Currently the only allowed solver is SolverCG, the - // SolverType input variable is a dummy - SolverCG solver(solver_control); - - // solve - try - { - if (this->fields[fieldIndex].type == SCALAR) - { - this->dU_scalar = 0.0; - solver.solve(*this, - this->dU_scalar, - *this->residualSet[fieldIndex], - IdentityMatrix( - this->solutionSet[fieldIndex]->size())); - } - else - { - this->dU_vector = 0.0; - solver.solve(*this, - this->dU_vector, - *this->residualSet[fieldIndex], - IdentityMatrix( - this->solutionSet[fieldIndex]->size())); - } - } - catch (...) - { - this->pcout << "\nWarning: linear solver did not converge as per " - "set tolerances. consider increasing the maximum " - "number of iterations or decreasing the solver " - "tolerance.\n"; - } - - if (userInputs.var_nonlinear[fieldIndex]) - { - // Now that we have the calculated change in the solution, - // we need to select a damping coefficient - double damping_coefficient; - - if (MatrixFreePDE::userInputs - .nonlinear_solver_parameters.getBacktrackDampingFlag( - fieldIndex)) - { - vectorType solutionSet_old = *this->solutionSet[fieldIndex]; - double residual_old = this->residualSet[fieldIndex]->l2_norm(); - - damping_coefficient = 1.0; - bool damping_coefficient_found = false; - while (!damping_coefficient_found) - { - if (this->fields[fieldIndex].type == SCALAR) - { - this->solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - this->dU_scalar); - } - else - { - this->solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - this->dU_vector); - } - - this->computeNonexplicitRHS(); - - for (const auto &it : *this->valuesDirichletSet[fieldIndex]) - { - if (this->residualSet[fieldIndex]->in_local_range( - it.first)) - { - (*this->residualSet[fieldIndex])(it.first) = 0.0; - } - } - - double residual_new = - this->residualSet[fieldIndex]->l2_norm(); - - if (this->currentIncrement % userInputs.skip_print_steps == - 0) - { - this->pcout << " Old residual: " << residual_old - << " Damping Coeff: " << damping_coefficient - << " New Residual: " << residual_new - << std::endl; - } - - // An improved approach would use the - // Armijo–Goldstein condition to ensure a - // sufficent decrease in the residual. This way is - // just scales the residual. - if ((residual_new < - (residual_old * - MatrixFreePDE::userInputs - .nonlinear_solver_parameters - .getBacktrackResidualDecreaseCoeff(fieldIndex))) || - damping_coefficient < 1.0e-4) - { - damping_coefficient_found = true; - } - else - { - damping_coefficient *= - MatrixFreePDE::userInputs - .nonlinear_solver_parameters - .getBacktrackStepModifier(fieldIndex); - *this->solutionSet[fieldIndex] = solutionSet_old; - } - } - } - else - { - damping_coefficient = - MatrixFreePDE::userInputs - .nonlinear_solver_parameters.getDefaultDampingCoefficient( - fieldIndex); - - if (this->fields[fieldIndex].type == SCALAR) - { - this->solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - this->dU_scalar); - } - else - { - this->solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - this->dU_vector); - } - } - - if (this->currentIncrement % userInputs.skip_print_steps == 0) - { - double dU_norm; - if (this->fields[fieldIndex].type == SCALAR) - { - dU_norm = this->dU_scalar.l2_norm(); - } - else - { - dU_norm = this->dU_vector.l2_norm(); - } - snprintf(buffer, - sizeof(buffer), - "field '%2s' [linear solve]: initial " - "residual:%12.6e, current residual:%12.6e, " - "nsteps:%u, tolerance criterion:%12.6e, " - "solution: %12.6e, dU: %12.6e\n", - this->fields[fieldIndex].name.c_str(), - this->residualSet[fieldIndex]->l2_norm(), - solver_control.last_value(), - solver_control.last_step(), - solver_control.tolerance(), - this->solutionSet[fieldIndex]->l2_norm(), - dU_norm); - this->pcout << buffer; - } - - // Check to see if this individual variable has converged - if (MatrixFreePDE::userInputs - .nonlinear_solver_parameters.getToleranceType(fieldIndex) == - ABSOLUTE_SOLUTION_CHANGE) - { - double diff; - - if (this->fields[fieldIndex].type == SCALAR) - { - diff = this->dU_scalar.l2_norm(); - } - else - { - diff = this->dU_vector.l2_norm(); - } - if (this->currentIncrement % userInputs.skip_print_steps == 0) - { - this->pcout << "Relative difference between " - "nonlinear iterations: " - << diff << " " << nonlinear_it_index << " " - << this->currentIncrement << std::endl; - } + nonlinear_it_converged = + this->updateImplicitSolution(fieldIndex, nonlinear_it_index); - if (diff > MatrixFreePDE::userInputs - .nonlinear_solver_parameters.getToleranceValue( - fieldIndex) && - nonlinear_it_index < - MatrixFreePDE::userInputs - .nonlinear_solver_parameters.getMaxIterations()) - { - nonlinear_it_converged = false; - } - } - else - { - std::cerr << "PRISMS-PF Error: Nonlinear solver tolerance " - "types other than ABSOLUTE_CHANGE have yet to " - "be implemented." - << std::endl; - } - } - else - { - if (nonlinear_it_index == 0) - { - if (this->fields[fieldIndex].type == SCALAR) - { - *this->solutionSet[fieldIndex] += this->dU_scalar; - } - else - { - *this->solutionSet[fieldIndex] += this->dU_vector; - } - - if (this->currentIncrement % userInputs.skip_print_steps == 0) - { - double dU_norm; - if (this->fields[fieldIndex].type == SCALAR) - { - dU_norm = this->dU_scalar.l2_norm(); - } - else - { - dU_norm = this->dU_vector.l2_norm(); - } - snprintf(buffer, - sizeof(buffer), - "field '%2s' [linear solve]: initial " - "residual:%12.6e, current residual:%12.6e, " - "nsteps:%u, tolerance criterion:%12.6e, " - "solution: %12.6e, dU: %12.6e\n", - this->fields[fieldIndex].name.c_str(), - this->residualSet[fieldIndex]->l2_norm(), - solver_control.last_value(), - solver_control.last_step(), - solver_control.tolerance(), - this->solutionSet[fieldIndex]->l2_norm(), - dU_norm); - this->pcout << buffer; - } - } - } + // Apply Boundary conditions + this->applyBCs(fieldIndex); } else if (this->fields[fieldIndex].pdetype == AUXILIARY) { diff --git a/include/matrixFreePDE.h b/include/matrixFreePDE.h index 58267185d..b157da3ab 100644 --- a/include/matrixFreePDE.h +++ b/include/matrixFreePDE.h @@ -262,6 +262,10 @@ class MatrixFreePDE : public Subscriptor void updateExplicitSolution(unsigned int fieldIndex); + /*Method to compute an implicit timestep*/ + bool + updateImplicitSolution(unsigned int fieldIndex, unsigned int nonlinear_it_index); + /*Method to apply boundary conditions*/ void applyBCs(unsigned int fieldIndex); diff --git a/src/matrixfree/solveIncrement.cc b/src/matrixfree/solveIncrement.cc index 43a9eeefc..b0f01987d 100644 --- a/src/matrixfree/solveIncrement.cc +++ b/src/matrixfree/solveIncrement.cc @@ -101,259 +101,8 @@ MatrixFreePDE::solveIncrement(bool skip_time_dependent) pcout << buffer; } - // apply Dirichlet BC'se - // This clears the residual where we want to apply Dirichlet - // BCs, otherwise the solver sees a positive residual - constraintsDirichletSet[fieldIndex]->set_zero(*residualSet[fieldIndex]); - - // solver controls - - double tol_value; - if (userInputs.linear_solver_parameters.getToleranceType(fieldIndex) == - ABSOLUTE_RESIDUAL) - { - tol_value = - userInputs.linear_solver_parameters.getToleranceValue(fieldIndex); - } - else - { - tol_value = userInputs.linear_solver_parameters.getToleranceValue( - fieldIndex) * - residualSet[fieldIndex]->l2_norm(); - } - - SolverControl solver_control( - userInputs.linear_solver_parameters.getMaxIterations(fieldIndex), - tol_value); - - // Currently the only allowed solver is SolverCG, the - // SolverType input variable is a dummy - SolverCG solver(solver_control); - - // solve - try - { - if (fields[fieldIndex].type == SCALAR) - { - dU_scalar = 0.0; - solver.solve(*this, - dU_scalar, - *residualSet[fieldIndex], - IdentityMatrix(solutionSet[fieldIndex]->size())); - } - else - { - dU_vector = 0.0; - solver.solve(*this, - dU_vector, - *residualSet[fieldIndex], - IdentityMatrix(solutionSet[fieldIndex]->size())); - } - } - catch (...) - { - pcout << "\nWarning: linear solver did not converge as " - "per set tolerances. consider increasing the " - "maximum number of iterations or decreasing the " - "solver tolerance.\n"; - } - - if (userInputs.var_nonlinear[fieldIndex]) - { - // Now that we have the calculated change in the solution, - // we need to select a damping coefficient - double damping_coefficient; - - if (userInputs.nonlinear_solver_parameters.getBacktrackDampingFlag( - fieldIndex)) - { - vectorType solutionSet_old = *solutionSet[fieldIndex]; - double residual_old = residualSet[fieldIndex]->l2_norm(); - - damping_coefficient = 1.0; - bool damping_coefficient_found = false; - while (!damping_coefficient_found) - { - if (fields[fieldIndex].type == SCALAR) - { - solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - dU_scalar); - } - else - { - solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - dU_vector); - } - - computeNonexplicitRHS(); - - for (const auto &it : *valuesDirichletSet[fieldIndex]) - { - if (residualSet[fieldIndex]->in_local_range(it.first)) - { - (*residualSet[fieldIndex])(it.first) = 0.0; - } - } - - double residual_new = residualSet[fieldIndex]->l2_norm(); - - if (currentIncrement % userInputs.skip_print_steps == 0) - { - pcout << " Old residual: " << residual_old - << " Damping Coeff: " << damping_coefficient - << " New Residual: " << residual_new << std::endl; - } - - // An improved approach would use the - // Armijo–Goldstein condition to ensure a - // sufficent decrease in the residual. This way is - // just scales the residual. - if ((residual_new < - (residual_old * - userInputs.nonlinear_solver_parameters - .getBacktrackResidualDecreaseCoeff(fieldIndex))) || - damping_coefficient < 1.0e-4) - { - damping_coefficient_found = true; - } - else - { - damping_coefficient *= - userInputs.nonlinear_solver_parameters - .getBacktrackStepModifier(fieldIndex); - *solutionSet[fieldIndex] = solutionSet_old; - } - } - } - else - { - damping_coefficient = - userInputs.nonlinear_solver_parameters - .getDefaultDampingCoefficient(fieldIndex); - - if (fields[fieldIndex].type == SCALAR) - { - solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - dU_scalar); - } - else - { - solutionSet[fieldIndex]->sadd(1.0, - damping_coefficient, - dU_vector); - } - } - - if (currentIncrement % userInputs.skip_print_steps == 0) - { - double dU_norm; - if (fields[fieldIndex].type == SCALAR) - { - dU_norm = dU_scalar.l2_norm(); - } - else - { - dU_norm = dU_vector.l2_norm(); - } - snprintf(buffer, - sizeof(buffer), - "field '%2s' [linear solve]: initial " - "residual:%12.6e, current residual:%12.6e, " - "nsteps:%u, tolerance criterion:%12.6e, " - "solution: %12.6e, dU: %12.6e\n", - fields[fieldIndex].name.c_str(), - residualSet[fieldIndex]->l2_norm(), - solver_control.last_value(), - solver_control.last_step(), - solver_control.tolerance(), - solutionSet[fieldIndex]->l2_norm(), - dU_norm); - pcout << buffer; - } - - // Check to see if this individual variable has converged - if (userInputs.nonlinear_solver_parameters.getToleranceType( - fieldIndex) == ABSOLUTE_SOLUTION_CHANGE) - { - double diff; - - if (fields[fieldIndex].type == SCALAR) - { - diff = dU_scalar.l2_norm(); - } - else - { - diff = dU_vector.l2_norm(); - } - if (currentIncrement % userInputs.skip_print_steps == 0) - { - pcout << "Relative difference between nonlinear " - "iterations: " - << diff << " " << nonlinear_it_index << " " - << currentIncrement << std::endl; - } - - if (diff > - userInputs.nonlinear_solver_parameters.getToleranceValue( - fieldIndex) && - nonlinear_it_index < - userInputs.nonlinear_solver_parameters.getMaxIterations()) - { - nonlinear_it_converged = false; - } - } - else - { - std::cerr << "PRISMS-PF Error: Nonlinear solver tolerance " - "types other than ABSOLUTE_CHANGE have yet to " - "be implemented." - << std::endl; - } - } - else - { - if (nonlinear_it_index == 0) - { - if (fields[fieldIndex].type == SCALAR) - { - *solutionSet[fieldIndex] += dU_scalar; - } - else - { - *solutionSet[fieldIndex] += dU_vector; - } - - if (currentIncrement % userInputs.skip_print_steps == 0) - { - double dU_norm; - if (fields[fieldIndex].type == SCALAR) - { - dU_norm = dU_scalar.l2_norm(); - } - else - { - dU_norm = dU_vector.l2_norm(); - } - snprintf(buffer, - sizeof(buffer), - "field '%2s' [linear solve]: initial " - "residual:%12.6e, current residual:%12.6e, " - "nsteps:%u, tolerance criterion:%12.6e, " - "solution: %12.6e, dU: %12.6e\n", - fields[fieldIndex].name.c_str(), - residualSet[fieldIndex]->l2_norm(), - solver_control.last_value(), - solver_control.last_step(), - solver_control.tolerance(), - solutionSet[fieldIndex]->l2_norm(), - dU_norm); - pcout << buffer; - } - } - } + nonlinear_it_converged = + updateImplicitSolution(fieldIndex, nonlinear_it_index); // Apply Boundary conditions applyBCs(fieldIndex); @@ -546,4 +295,254 @@ MatrixFreePDE::updateExplicitSolution(unsigned int fieldIndex) } } +template +bool +MatrixFreePDE::updateImplicitSolution(unsigned int fieldIndex, + unsigned int nonlinear_it_index) +{ + char buffer[200]; + bool nonlinear_it_converged = false; + + // Apply Dirichlet BC's. This clears the residual where we want to apply Dirichlet BCs, + // otherwise the solver sees a positive residual + constraintsDirichletSet[fieldIndex]->set_zero(*residualSet[fieldIndex]); + + // Grab solver controls + double tol_value; + if (userInputs.linear_solver_parameters.getToleranceType(fieldIndex) == + ABSOLUTE_RESIDUAL) + { + tol_value = userInputs.linear_solver_parameters.getToleranceValue(fieldIndex); + } + else + { + tol_value = userInputs.linear_solver_parameters.getToleranceValue(fieldIndex) * + residualSet[fieldIndex]->l2_norm(); + } + + SolverControl solver_control(userInputs.linear_solver_parameters.getMaxIterations( + fieldIndex), + tol_value); + + // Currently the only allowed solver is SolverCG, the + // SolverType input variable is a dummy + SolverCG solver(solver_control); + + // Solve + try + { + if (fields[fieldIndex].type == SCALAR) + { + dU_scalar = 0.0; + solver.solve(*this, + dU_scalar, + *residualSet[fieldIndex], + IdentityMatrix(solutionSet[fieldIndex]->size())); + } + else + { + dU_vector = 0.0; + solver.solve(*this, + dU_vector, + *residualSet[fieldIndex], + IdentityMatrix(solutionSet[fieldIndex]->size())); + } + } + catch (...) + { + pcout << "\nWarning: linear solver did not converge as " + "per set tolerances. consider increasing the " + "maximum number of iterations or decreasing the " + "solver tolerance.\n"; + } + + if (userInputs.var_nonlinear[fieldIndex]) + { + // Now that we have the calculated change in the solution, + // we need to select a damping coefficient + double damping_coefficient; + + if (userInputs.nonlinear_solver_parameters.getBacktrackDampingFlag(fieldIndex)) + { + vectorType solutionSet_old = *solutionSet[fieldIndex]; + double residual_old = residualSet[fieldIndex]->l2_norm(); + + damping_coefficient = 1.0; + bool damping_coefficient_found = false; + while (!damping_coefficient_found) + { + if (fields[fieldIndex].type == SCALAR) + { + solutionSet[fieldIndex]->sadd(1.0, damping_coefficient, dU_scalar); + } + else + { + solutionSet[fieldIndex]->sadd(1.0, damping_coefficient, dU_vector); + } + + computeNonexplicitRHS(); + + for (const auto &it : *valuesDirichletSet[fieldIndex]) + { + if (residualSet[fieldIndex]->in_local_range(it.first)) + { + (*residualSet[fieldIndex])(it.first) = 0.0; + } + } + + double residual_new = residualSet[fieldIndex]->l2_norm(); + + if (currentIncrement % userInputs.skip_print_steps == 0) + { + pcout << " Old residual: " << residual_old + << " Damping Coeff: " << damping_coefficient + << " New Residual: " << residual_new << std::endl; + } + + // An improved approach would use the + // Armijo–Goldstein condition to ensure a + // sufficent decrease in the residual. This way is + // just scales the residual. + if ((residual_new < + (residual_old * userInputs.nonlinear_solver_parameters + .getBacktrackResidualDecreaseCoeff(fieldIndex))) || + damping_coefficient < 1.0e-4) + { + damping_coefficient_found = true; + } + else + { + damping_coefficient *= + userInputs.nonlinear_solver_parameters.getBacktrackStepModifier( + fieldIndex); + *solutionSet[fieldIndex] = solutionSet_old; + } + } + } + else + { + damping_coefficient = + userInputs.nonlinear_solver_parameters.getDefaultDampingCoefficient( + fieldIndex); + + if (fields[fieldIndex].type == SCALAR) + { + solutionSet[fieldIndex]->sadd(1.0, damping_coefficient, dU_scalar); + } + else + { + solutionSet[fieldIndex]->sadd(1.0, damping_coefficient, dU_vector); + } + } + + if (currentIncrement % userInputs.skip_print_steps == 0) + { + double dU_norm; + if (fields[fieldIndex].type == SCALAR) + { + dU_norm = dU_scalar.l2_norm(); + } + else + { + dU_norm = dU_vector.l2_norm(); + } + snprintf(buffer, + sizeof(buffer), + "field '%2s' [linear solve]: initial " + "residual:%12.6e, current residual:%12.6e, " + "nsteps:%u, tolerance criterion:%12.6e, " + "solution: %12.6e, dU: %12.6e\n", + fields[fieldIndex].name.c_str(), + residualSet[fieldIndex]->l2_norm(), + solver_control.last_value(), + solver_control.last_step(), + solver_control.tolerance(), + solutionSet[fieldIndex]->l2_norm(), + dU_norm); + pcout << buffer; + } + + // Check to see if this individual variable has converged + if (userInputs.nonlinear_solver_parameters.getToleranceType(fieldIndex) == + ABSOLUTE_SOLUTION_CHANGE) + { + double diff; + + if (fields[fieldIndex].type == SCALAR) + { + diff = dU_scalar.l2_norm(); + } + else + { + diff = dU_vector.l2_norm(); + } + if (currentIncrement % userInputs.skip_print_steps == 0) + { + pcout << "Relative difference between nonlinear " + "iterations: " + << diff << " " << nonlinear_it_index << " " << currentIncrement + << std::endl; + } + + if (diff > + userInputs.nonlinear_solver_parameters.getToleranceValue(fieldIndex) && + nonlinear_it_index < + userInputs.nonlinear_solver_parameters.getMaxIterations()) + { + nonlinear_it_converged = false; + } + } + else + { + std::cerr << "PRISMS-PF Error: Nonlinear solver tolerance " + "types other than ABSOLUTE_CHANGE have yet to " + "be implemented." + << std::endl; + } + } + else + { + if (nonlinear_it_index == 0) + { + if (fields[fieldIndex].type == SCALAR) + { + *solutionSet[fieldIndex] += dU_scalar; + } + else + { + *solutionSet[fieldIndex] += dU_vector; + } + + if (currentIncrement % userInputs.skip_print_steps == 0) + { + double dU_norm; + if (fields[fieldIndex].type == SCALAR) + { + dU_norm = dU_scalar.l2_norm(); + } + else + { + dU_norm = dU_vector.l2_norm(); + } + snprintf(buffer, + sizeof(buffer), + "field '%2s' [linear solve]: initial " + "residual:%12.6e, current residual:%12.6e, " + "nsteps:%u, tolerance criterion:%12.6e, " + "solution: %12.6e, dU: %12.6e\n", + fields[fieldIndex].name.c_str(), + residualSet[fieldIndex]->l2_norm(), + solver_control.last_value(), + solver_control.last_step(), + solver_control.tolerance(), + solutionSet[fieldIndex]->l2_norm(), + dU_norm); + pcout << buffer; + } + } + } + + return nonlinear_it_converged; +} + #include "../../include/matrixFreePDE_template_instantiations.h" From 0fc9366d1bbf71c663150d2c36a88ac2535e47c8 Mon Sep 17 00:00:00 2001 From: landinjm Date: Sun, 20 Oct 2024 20:50:24 -0400 Subject: [PATCH 2/2] changed bool default initialization to true/converged --- src/matrixfree/solveIncrement.cc | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/matrixfree/solveIncrement.cc b/src/matrixfree/solveIncrement.cc index b0f01987d..3199cf0ff 100644 --- a/src/matrixfree/solveIncrement.cc +++ b/src/matrixfree/solveIncrement.cc @@ -301,7 +301,9 @@ MatrixFreePDE::updateImplicitSolution(unsigned int fieldIndex, unsigned int nonlinear_it_index) { char buffer[200]; - bool nonlinear_it_converged = false; + + // Assume convergence criterion is met, unless otherwise proven later on. + bool nonlinear_it_converged = true; // Apply Dirichlet BC's. This clears the residual where we want to apply Dirichlet BCs, // otherwise the solver sees a positive residual