xref: /petsc/src/snes/interface/snespc.c (revision e0f629dd96229b85b8d4fbf4a5ed4f197d232daa)
132f3f7c2SPeter Brune 
2af0996ceSBarry Smith #include <petsc/private/snesimpl.h> /*I "petscsnes.h"  I*/
332f3f7c2SPeter Brune 
432f3f7c2SPeter Brune /*@
5be95d8f1SBarry Smith    SNESApplyNPC - Calls SNESSolve() on preconditioner for the SNES
632f3f7c2SPeter Brune 
7f6dfbefdSBarry Smith    Collective on snes
832f3f7c2SPeter Brune 
932f3f7c2SPeter Brune    Input Parameters:
1032f3f7c2SPeter Brune +  snes - the SNES context
116a63999bSPeter Brune .  x - input vector
126a63999bSPeter Brune -  f - optional; the function evaluation on x
1332f3f7c2SPeter Brune 
1432f3f7c2SPeter Brune    Output Parameter:
1532f3f7c2SPeter Brune .  y - function vector, as set by SNESSetFunction()
1632f3f7c2SPeter Brune 
17f6dfbefdSBarry Smith    Note:
18be95d8f1SBarry Smith    SNESComputeFunction() should be called on x before SNESApplyNPC() is called, as it is
1932f3f7c2SPeter Brune    with SNESComuteJacobian().
2032f3f7c2SPeter Brune 
2132f3f7c2SPeter Brune    Level: developer
2232f3f7c2SPeter Brune 
23c2e3fba1SPatrick Sanan .seealso: `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()`
2432f3f7c2SPeter Brune @*/
259371c9d4SSatish Balay PetscErrorCode SNESApplyNPC(SNES snes, Vec x, Vec f, Vec y) {
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);
32*e0f629ddSJacob 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     PetscFunctionReturn(0);
4132f3f7c2SPeter Brune   }
4232f3f7c2SPeter Brune   PetscFunctionReturn(0);
4332f3f7c2SPeter Brune }
4432f3f7c2SPeter Brune 
459371c9d4SSatish Balay PetscErrorCode SNESComputeFunctionDefaultNPC(SNES snes, Vec X, Vec F) {
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   }
5732f3f7c2SPeter Brune   PetscFunctionReturn(0);
5832f3f7c2SPeter Brune }
59ddd40ce5SPeter Brune 
60ddd40ce5SPeter Brune /*@
61f6dfbefdSBarry Smith    SNESGetNPCFunction - Gets the current function value and its norm from a nonlinear preconditioner after `SNESSolve()` has been called on that `SNES`
62ddd40ce5SPeter Brune 
63f6dfbefdSBarry Smith    Collective on snes
64ddd40ce5SPeter Brune 
65f899ff85SJose E. Roman    Input Parameter:
66f6dfbefdSBarry Smith .  snes - the `SNES` context
67ddd40ce5SPeter Brune 
68d8d19677SJose E. Roman    Output Parameters:
69a2b725a8SWilliam Gropp +  F - function vector
70a2b725a8SWilliam Gropp -  fnorm - the norm of F
71ddd40ce5SPeter Brune 
72ddd40ce5SPeter Brune    Level: developer
73ddd40ce5SPeter Brune 
74c2e3fba1SPatrick Sanan .seealso: `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()`, `SNESApplyNPC()`, `SNESSolve()`
75ddd40ce5SPeter Brune @*/
769371c9d4SSatish Balay PetscErrorCode SNESGetNPCFunction(SNES snes, Vec F, PetscReal *fnorm) {
77ddd40ce5SPeter Brune   PCSide           npcside;
78ddd40ce5SPeter Brune   SNESFunctionType functype;
79ddd40ce5SPeter Brune   SNESNormSchedule normschedule;
80ddd40ce5SPeter Brune   Vec              FPC, XPC;
81ddd40ce5SPeter Brune 
82ddd40ce5SPeter Brune   PetscFunctionBegin;
83efd4aadfSBarry Smith   if (snes->npc) {
849566063dSJacob Faibussowitsch     PetscCall(SNESGetNPCSide(snes->npc, &npcside));
859566063dSJacob Faibussowitsch     PetscCall(SNESGetFunctionType(snes->npc, &functype));
869566063dSJacob Faibussowitsch     PetscCall(SNESGetNormSchedule(snes->npc, &normschedule));
87ddd40ce5SPeter Brune 
88ddd40ce5SPeter Brune     /* check if the function is valid based upon how the inner solver is preconditioned */
89ddd40ce5SPeter Brune     if (normschedule != SNES_NORM_NONE && normschedule != SNES_NORM_INITIAL_ONLY && (npcside == PC_RIGHT || functype == SNES_FUNCTION_UNPRECONDITIONED)) {
909566063dSJacob Faibussowitsch       PetscCall(SNESGetFunction(snes->npc, &FPC, NULL, NULL));
911d3ed18cSPeter Brune       if (FPC) {
929566063dSJacob Faibussowitsch         if (fnorm) PetscCall(VecNorm(FPC, NORM_2, fnorm));
939566063dSJacob Faibussowitsch         PetscCall(VecCopy(FPC, F));
94f6dfbefdSBarry Smith       } else SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONGSTATE, "Nonlinear preconditioner has no function");
95ddd40ce5SPeter Brune     } else {
969566063dSJacob Faibussowitsch       PetscCall(SNESGetSolution(snes->npc, &XPC));
97ddd40ce5SPeter Brune       if (XPC) {
989566063dSJacob Faibussowitsch         PetscCall(SNESComputeFunction(snes->npc, XPC, F));
999566063dSJacob Faibussowitsch         if (fnorm) PetscCall(VecNorm(F, NORM_2, fnorm));
100f6dfbefdSBarry Smith       } else SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONGSTATE, "Nonlinear preconditioner has no solution");
101ddd40ce5SPeter Brune     }
102f6dfbefdSBarry Smith   } else SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONGSTATE, "No nonlinear preconditioner set");
103ddd40ce5SPeter Brune   PetscFunctionReturn(0);
104ddd40ce5SPeter Brune }
105