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 @*/
SNESApplyNPC(SNES snes,Vec x,Vec f,Vec y)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
SNESComputeFunctionDefaultNPC(SNES snes,Vec X,Vec F)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 @*/
SNESGetNPCFunction(SNES snes,Vec F,PetscReal * fnorm)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