1af0996ceSBarry Smith #include <petsc/private/snesimpl.h> /*I "petscsnes.h" I*/ 232f3f7c2SPeter Brune 332f3f7c2SPeter Brune /*@ 4420bcc1bSBarry Smith SNESApplyNPC - Calls `SNESSolve()` on the preconditioner for the `SNES` 532f3f7c2SPeter Brune 6c3339decSBarry Smith Collective 732f3f7c2SPeter Brune 832f3f7c2SPeter Brune Input Parameters: 9dc4c0fb0SBarry Smith + snes - the `SNES` context 106a63999bSPeter Brune . x - input vector 11dc4c0fb0SBarry Smith - f - optional; the function evaluation on `x` 1232f3f7c2SPeter Brune 1332f3f7c2SPeter Brune Output Parameter: 14dc4c0fb0SBarry Smith . y - function vector, as set by `SNESSetFunction()` 1532f3f7c2SPeter Brune 1632f3f7c2SPeter Brune Level: developer 1732f3f7c2SPeter Brune 18dc4c0fb0SBarry Smith Note: 19dc4c0fb0SBarry Smith `SNESComputeFunction()` should be called on `x` before `SNESApplyNPC()` is called, as it is 20dc4c0fb0SBarry Smith with `SNESComuteJacobian()`. 21dc4c0fb0SBarry Smith 22420bcc1bSBarry Smith .seealso: [](ch_snes), `SNES`, `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()` 2332f3f7c2SPeter Brune @*/ 24d71ae5a4SJacob Faibussowitsch PetscErrorCode SNESApplyNPC(SNES snes, Vec x, Vec f, Vec y) 25d71ae5a4SJacob Faibussowitsch { 2632f3f7c2SPeter Brune PetscFunctionBegin; 2732f3f7c2SPeter Brune PetscValidHeaderSpecific(snes, SNES_CLASSID, 1); 2832f3f7c2SPeter Brune PetscValidHeaderSpecific(x, VEC_CLASSID, 2); 29064a246eSJacob Faibussowitsch PetscValidHeaderSpecific(y, VEC_CLASSID, 4); 3032f3f7c2SPeter Brune PetscCheckSameComm(snes, 1, x, 2); 31064a246eSJacob Faibussowitsch PetscCheckSameComm(snes, 1, y, 4); 32e0f629ddSJacob Faibussowitsch PetscCall(VecValidValues_Internal(x, 2, PETSC_TRUE)); 33efd4aadfSBarry Smith if (snes->npc) { 341baa6e33SBarry Smith if (f) PetscCall(SNESSetInitialFunction(snes->npc, f)); 359566063dSJacob Faibussowitsch PetscCall(VecCopy(x, y)); 369566063dSJacob Faibussowitsch PetscCall(PetscLogEventBegin(SNES_NPCSolve, snes->npc, x, y, 0)); 379566063dSJacob Faibussowitsch PetscCall(SNESSolve(snes->npc, snes->vec_rhs, y)); 389566063dSJacob Faibussowitsch PetscCall(PetscLogEventEnd(SNES_NPCSolve, snes->npc, x, y, 0)); 399566063dSJacob Faibussowitsch PetscCall(VecAYPX(y, -1.0, x)); 4032f3f7c2SPeter Brune } 413ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS); 4232f3f7c2SPeter Brune } 4332f3f7c2SPeter Brune 44d71ae5a4SJacob Faibussowitsch PetscErrorCode SNESComputeFunctionDefaultNPC(SNES snes, Vec X, Vec F) 45d71ae5a4SJacob Faibussowitsch { 4632f3f7c2SPeter Brune /* This is to be used as an argument to SNESMF -- NOT as a "function" */ 47b7281c8aSPeter Brune SNESConvergedReason reason; 48b7281c8aSPeter Brune 4932f3f7c2SPeter Brune PetscFunctionBegin; 50efd4aadfSBarry Smith if (snes->npc) { 519566063dSJacob Faibussowitsch PetscCall(SNESApplyNPC(snes, X, NULL, F)); 529566063dSJacob Faibussowitsch PetscCall(SNESGetConvergedReason(snes->npc, &reason)); 5348a46eb9SPierre Jolivet if (reason < 0 && reason != SNES_DIVERGED_MAX_IT) PetscCall(SNESSetFunctionDomainError(snes)); 54b7281c8aSPeter Brune } else { 559566063dSJacob Faibussowitsch PetscCall(SNESComputeFunction(snes, X, F)); 56b7281c8aSPeter Brune } 573ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS); 5832f3f7c2SPeter Brune } 59ddd40ce5SPeter Brune 60ddd40ce5SPeter Brune /*@ 61*76c63389SBarry Smith SNESGetNPCFunction - Gets the current function value (for the callback function provided by `SNESSetFunction()`, 62*76c63389SBarry Smith and its norm from a nonlinear preconditioner after `SNESSolve()` has been called on that `SNES` 63ddd40ce5SPeter Brune 64c3339decSBarry Smith Collective 65ddd40ce5SPeter Brune 66f899ff85SJose E. Roman Input Parameter: 67f6dfbefdSBarry Smith . snes - the `SNES` context 68ddd40ce5SPeter Brune 69d8d19677SJose E. Roman Output Parameters: 70a2b725a8SWilliam Gropp + F - function vector 71dc4c0fb0SBarry Smith - fnorm - the norm of `F` 72ddd40ce5SPeter Brune 73ddd40ce5SPeter Brune Level: developer 74ddd40ce5SPeter Brune 75420bcc1bSBarry Smith .seealso: [](ch_snes), `SNES`, `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()`, `SNESApplyNPC()`, `SNESSolve()` 76ddd40ce5SPeter Brune @*/ 77d71ae5a4SJacob Faibussowitsch PetscErrorCode SNESGetNPCFunction(SNES snes, Vec F, PetscReal *fnorm) 78d71ae5a4SJacob Faibussowitsch { 79ddd40ce5SPeter Brune PCSide npcside; 80ddd40ce5SPeter Brune SNESFunctionType functype; 81ddd40ce5SPeter Brune SNESNormSchedule normschedule; 82ddd40ce5SPeter Brune Vec FPC, XPC; 83ddd40ce5SPeter Brune 84ddd40ce5SPeter Brune PetscFunctionBegin; 850fdf79fbSJacob Faibussowitsch PetscValidHeaderSpecific(snes, SNES_CLASSID, 1); 864f572ea9SToby Isaac if (fnorm) PetscAssertPointer(fnorm, 3); 870fdf79fbSJacob Faibussowitsch PetscCheck(snes->npc, PETSC_COMM_SELF, PETSC_ERR_ARG_WRONGSTATE, "No nonlinear preconditioner set"); 889566063dSJacob Faibussowitsch PetscCall(SNESGetNPCSide(snes->npc, &npcside)); 899566063dSJacob Faibussowitsch PetscCall(SNESGetFunctionType(snes->npc, &functype)); 909566063dSJacob Faibussowitsch PetscCall(SNESGetNormSchedule(snes->npc, &normschedule)); 91ddd40ce5SPeter Brune 92ddd40ce5SPeter Brune /* check if the function is valid based upon how the inner solver is preconditioned */ 93ddd40ce5SPeter Brune if (normschedule != SNES_NORM_NONE && normschedule != SNES_NORM_INITIAL_ONLY && (npcside == PC_RIGHT || functype == SNES_FUNCTION_UNPRECONDITIONED)) { 949566063dSJacob Faibussowitsch PetscCall(SNESGetFunction(snes->npc, &FPC, NULL, NULL)); 950fdf79fbSJacob Faibussowitsch PetscCheck(FPC, PETSC_COMM_SELF, PETSC_ERR_ARG_WRONGSTATE, "Nonlinear preconditioner has no function"); 969566063dSJacob Faibussowitsch if (fnorm) PetscCall(VecNorm(FPC, NORM_2, fnorm)); 979566063dSJacob Faibussowitsch PetscCall(VecCopy(FPC, F)); 98ddd40ce5SPeter Brune } else { 999566063dSJacob Faibussowitsch PetscCall(SNESGetSolution(snes->npc, &XPC)); 1000fdf79fbSJacob Faibussowitsch PetscCheck(XPC, PETSC_COMM_SELF, PETSC_ERR_ARG_WRONGSTATE, "Nonlinear preconditioner has no solution"); 1019566063dSJacob Faibussowitsch PetscCall(SNESComputeFunction(snes->npc, XPC, F)); 1029566063dSJacob Faibussowitsch if (fnorm) PetscCall(VecNorm(F, NORM_2, fnorm)); 103ddd40ce5SPeter Brune } 1043ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS); 105ddd40ce5SPeter Brune } 106