xref: /petsc/src/snes/interface/snespc.c (revision c2e3fba1fe1cda7e6350bbca19c4ed35ce95940a)
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 
732f3f7c2SPeter Brune    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 
1732f3f7c2SPeter Brune    Notes:
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 
23*c2e3fba1SPatrick Sanan .seealso: `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()`
2432f3f7c2SPeter Brune @*/
25be95d8f1SBarry Smith PetscErrorCode  SNESApplyNPC(SNES snes,Vec x,Vec f,Vec y)
2632f3f7c2SPeter Brune {
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);
339566063dSJacob Faibussowitsch   PetscCall(VecValidValues(x,2,PETSC_TRUE));
34efd4aadfSBarry Smith   if (snes->npc) {
3532f3f7c2SPeter Brune     if (f) {
369566063dSJacob Faibussowitsch       PetscCall(SNESSetInitialFunction(snes->npc,f));
3732f3f7c2SPeter Brune     }
389566063dSJacob Faibussowitsch     PetscCall(VecCopy(x,y));
399566063dSJacob Faibussowitsch     PetscCall(PetscLogEventBegin(SNES_NPCSolve,snes->npc,x,y,0));
409566063dSJacob Faibussowitsch     PetscCall(SNESSolve(snes->npc,snes->vec_rhs,y));
419566063dSJacob Faibussowitsch     PetscCall(PetscLogEventEnd(SNES_NPCSolve,snes->npc,x,y,0));
429566063dSJacob Faibussowitsch     PetscCall(VecAYPX(y,-1.0,x));
4332f3f7c2SPeter Brune     PetscFunctionReturn(0);
4432f3f7c2SPeter Brune   }
4532f3f7c2SPeter Brune   PetscFunctionReturn(0);
4632f3f7c2SPeter Brune }
4732f3f7c2SPeter Brune 
48be95d8f1SBarry Smith PetscErrorCode SNESComputeFunctionDefaultNPC(SNES snes,Vec X,Vec F)
49be95d8f1SBarry Smith {
5032f3f7c2SPeter Brune /* This is to be used as an argument to SNESMF -- NOT as a "function" */
51b7281c8aSPeter Brune   SNESConvergedReason reason;
52b7281c8aSPeter Brune 
5332f3f7c2SPeter Brune   PetscFunctionBegin;
54efd4aadfSBarry Smith   if (snes->npc) {
559566063dSJacob Faibussowitsch     PetscCall(SNESApplyNPC(snes,X,NULL,F));
569566063dSJacob Faibussowitsch     PetscCall(SNESGetConvergedReason(snes->npc,&reason));
57b7281c8aSPeter Brune     if (reason < 0  && reason != SNES_DIVERGED_MAX_IT) {
589566063dSJacob Faibussowitsch       PetscCall(SNESSetFunctionDomainError(snes));
59b7281c8aSPeter Brune     }
60b7281c8aSPeter Brune   } else {
619566063dSJacob Faibussowitsch     PetscCall(SNESComputeFunction(snes,X,F));
62b7281c8aSPeter Brune   }
6332f3f7c2SPeter Brune   PetscFunctionReturn(0);
6432f3f7c2SPeter Brune }
65ddd40ce5SPeter Brune 
66ddd40ce5SPeter Brune /*@
67be95d8f1SBarry Smith    SNESGetNPCFunction - Gets the function from a preconditioner after SNESSolve() has been called.
68ddd40ce5SPeter Brune 
69ddd40ce5SPeter Brune    Collective on SNES
70ddd40ce5SPeter Brune 
71f899ff85SJose E. Roman    Input Parameter:
72ddd40ce5SPeter Brune .  snes - the SNES context
73ddd40ce5SPeter Brune 
74d8d19677SJose E. Roman    Output Parameters:
75a2b725a8SWilliam Gropp +  F - function vector
76a2b725a8SWilliam Gropp -  fnorm - the norm of F
77ddd40ce5SPeter Brune 
78ddd40ce5SPeter Brune    Level: developer
79ddd40ce5SPeter Brune 
80*c2e3fba1SPatrick Sanan .seealso: `SNESGetNPC()`, `SNESSetNPC()`, `SNESComputeFunction()`, `SNESApplyNPC()`, `SNESSolve()`
81ddd40ce5SPeter Brune @*/
82be95d8f1SBarry Smith PetscErrorCode SNESGetNPCFunction(SNES snes,Vec F,PetscReal *fnorm)
83ddd40ce5SPeter Brune {
84ddd40ce5SPeter Brune   PCSide           npcside;
85ddd40ce5SPeter Brune   SNESFunctionType functype;
86ddd40ce5SPeter Brune   SNESNormSchedule normschedule;
87ddd40ce5SPeter Brune   Vec              FPC,XPC;
88ddd40ce5SPeter Brune 
89ddd40ce5SPeter Brune   PetscFunctionBegin;
90efd4aadfSBarry Smith   if (snes->npc) {
919566063dSJacob Faibussowitsch     PetscCall(SNESGetNPCSide(snes->npc,&npcside));
929566063dSJacob Faibussowitsch     PetscCall(SNESGetFunctionType(snes->npc,&functype));
939566063dSJacob Faibussowitsch     PetscCall(SNESGetNormSchedule(snes->npc,&normschedule));
94ddd40ce5SPeter Brune 
95ddd40ce5SPeter Brune     /* check if the function is valid based upon how the inner solver is preconditioned */
96ddd40ce5SPeter Brune     if (normschedule != SNES_NORM_NONE && normschedule != SNES_NORM_INITIAL_ONLY && (npcside == PC_RIGHT || functype == SNES_FUNCTION_UNPRECONDITIONED)) {
979566063dSJacob Faibussowitsch       PetscCall(SNESGetFunction(snes->npc,&FPC,NULL,NULL));
981d3ed18cSPeter Brune       if (FPC) {
999566063dSJacob Faibussowitsch         if (fnorm) PetscCall(VecNorm(FPC,NORM_2,fnorm));
1009566063dSJacob Faibussowitsch         PetscCall(VecCopy(FPC,F));
1016c4ed002SBarry Smith       } else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"Preconditioner has no function");
102ddd40ce5SPeter Brune     } else {
1039566063dSJacob Faibussowitsch       PetscCall(SNESGetSolution(snes->npc,&XPC));
104ddd40ce5SPeter Brune       if (XPC) {
1059566063dSJacob Faibussowitsch         PetscCall(SNESComputeFunction(snes->npc,XPC,F));
1069566063dSJacob Faibussowitsch         if (fnorm) PetscCall(VecNorm(F,NORM_2,fnorm));
10737703b41SBarry Smith       } else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"Preconditioner has no solution");
108ddd40ce5SPeter Brune     }
10937703b41SBarry Smith   } else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"No preconditioner set");
110ddd40ce5SPeter Brune   PetscFunctionReturn(0);
111ddd40ce5SPeter Brune }
112