xref: /petsc/src/snes/interface/snespc.c (revision dc4c0fb0d12fea86f250fc6da7e50c6579cf47ac)
132f3f7c2SPeter Brune 
2af0996ceSBarry Smith #include <petsc/private/snesimpl.h> /*I "petscsnes.h"  I*/
332f3f7c2SPeter Brune 
432f3f7c2SPeter Brune /*@
5*dc4c0fb0SBarry Smith    SNESApplyNPC - Calls `SNESSolve()` on preconditioner for the `SNES`
632f3f7c2SPeter Brune 
7c3339decSBarry Smith    Collective
832f3f7c2SPeter Brune 
932f3f7c2SPeter Brune    Input Parameters:
10*dc4c0fb0SBarry Smith +  snes - the `SNES` context
116a63999bSPeter Brune .  x - input vector
12*dc4c0fb0SBarry Smith -  f - optional; the function evaluation on `x`
1332f3f7c2SPeter Brune 
1432f3f7c2SPeter Brune    Output Parameter:
15*dc4c0fb0SBarry Smith .  y - function vector, as set by `SNESSetFunction()`
1632f3f7c2SPeter Brune 
1732f3f7c2SPeter Brune    Level: developer
1832f3f7c2SPeter Brune 
19*dc4c0fb0SBarry Smith    Note:
20*dc4c0fb0SBarry Smith    `SNESComputeFunction()` should be called on `x` before `SNESApplyNPC()` is called, as it is
21*dc4c0fb0SBarry Smith    with `SNESComuteJacobian()`.
22*dc4c0fb0SBarry Smith 
23c2e3fba1SPatrick Sanan .seealso: `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()`
2432f3f7c2SPeter Brune @*/
25d71ae5a4SJacob Faibussowitsch PetscErrorCode SNESApplyNPC(SNES snes, Vec x, Vec f, Vec y)
26d71ae5a4SJacob Faibussowitsch {
2732f3f7c2SPeter Brune   PetscFunctionBegin;
2832f3f7c2SPeter Brune   PetscValidHeaderSpecific(snes, SNES_CLASSID, 1);
2932f3f7c2SPeter Brune   PetscValidHeaderSpecific(x, VEC_CLASSID, 2);
30064a246eSJacob Faibussowitsch   PetscValidHeaderSpecific(y, VEC_CLASSID, 4);
3132f3f7c2SPeter Brune   PetscCheckSameComm(snes, 1, x, 2);
32064a246eSJacob Faibussowitsch   PetscCheckSameComm(snes, 1, y, 4);
33e0f629ddSJacob Faibussowitsch   PetscCall(VecValidValues_Internal(x, 2, PETSC_TRUE));
34efd4aadfSBarry Smith   if (snes->npc) {
351baa6e33SBarry Smith     if (f) PetscCall(SNESSetInitialFunction(snes->npc, f));
369566063dSJacob Faibussowitsch     PetscCall(VecCopy(x, y));
379566063dSJacob Faibussowitsch     PetscCall(PetscLogEventBegin(SNES_NPCSolve, snes->npc, x, y, 0));
389566063dSJacob Faibussowitsch     PetscCall(SNESSolve(snes->npc, snes->vec_rhs, y));
399566063dSJacob Faibussowitsch     PetscCall(PetscLogEventEnd(SNES_NPCSolve, snes->npc, x, y, 0));
409566063dSJacob Faibussowitsch     PetscCall(VecAYPX(y, -1.0, x));
4132f3f7c2SPeter Brune   }
423ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
4332f3f7c2SPeter Brune }
4432f3f7c2SPeter Brune 
45d71ae5a4SJacob Faibussowitsch PetscErrorCode SNESComputeFunctionDefaultNPC(SNES snes, Vec X, Vec F)
46d71ae5a4SJacob Faibussowitsch {
4732f3f7c2SPeter Brune   /* This is to be used as an argument to SNESMF -- NOT as a "function" */
48b7281c8aSPeter Brune   SNESConvergedReason reason;
49b7281c8aSPeter Brune 
5032f3f7c2SPeter Brune   PetscFunctionBegin;
51efd4aadfSBarry Smith   if (snes->npc) {
529566063dSJacob Faibussowitsch     PetscCall(SNESApplyNPC(snes, X, NULL, F));
539566063dSJacob Faibussowitsch     PetscCall(SNESGetConvergedReason(snes->npc, &reason));
5448a46eb9SPierre Jolivet     if (reason < 0 && reason != SNES_DIVERGED_MAX_IT) PetscCall(SNESSetFunctionDomainError(snes));
55b7281c8aSPeter Brune   } else {
569566063dSJacob Faibussowitsch     PetscCall(SNESComputeFunction(snes, X, F));
57b7281c8aSPeter Brune   }
583ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
5932f3f7c2SPeter Brune }
60ddd40ce5SPeter Brune 
61ddd40ce5SPeter Brune /*@
62f6dfbefdSBarry Smith    SNESGetNPCFunction - Gets the current function value 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
71*dc4c0fb0SBarry Smith -  fnorm - the norm of `F`
72ddd40ce5SPeter Brune 
73ddd40ce5SPeter Brune    Level: developer
74ddd40ce5SPeter Brune 
75c2e3fba1SPatrick Sanan .seealso: `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);
860fdf79fbSJacob Faibussowitsch   if (fnorm) PetscValidRealPointer(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