xref: /petsc/src/snes/interface/snespc.c (revision efd4aadf157bf1ba2d80c2be092fcf4247860003)
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 
2332f3f7c2SPeter Brune .keywords: SNES, nonlinear, compute, function
2432f3f7c2SPeter Brune 
25be95d8f1SBarry Smith .seealso: SNESGetNPC(),SNESSetNPC(),SNESComputeFunction()
2632f3f7c2SPeter Brune @*/
27be95d8f1SBarry Smith PetscErrorCode  SNESApplyNPC(SNES snes,Vec x,Vec f,Vec y)
2832f3f7c2SPeter Brune {
2932f3f7c2SPeter Brune   PetscErrorCode ierr;
3032f3f7c2SPeter Brune 
3132f3f7c2SPeter Brune   PetscFunctionBegin;
3232f3f7c2SPeter Brune   PetscValidHeaderSpecific(snes,SNES_CLASSID,1);
3332f3f7c2SPeter Brune   PetscValidHeaderSpecific(x,VEC_CLASSID,2);
3432f3f7c2SPeter Brune   PetscValidHeaderSpecific(y,VEC_CLASSID,3);
3532f3f7c2SPeter Brune   PetscCheckSameComm(snes,1,x,2);
3632f3f7c2SPeter Brune   PetscCheckSameComm(snes,1,y,3);
3732f3f7c2SPeter Brune   ierr = VecValidValues(x,2,PETSC_TRUE);CHKERRQ(ierr);
38*efd4aadfSBarry Smith   if (snes->npc) {
3932f3f7c2SPeter Brune     if (f) {
40*efd4aadfSBarry Smith       ierr = SNESSetInitialFunction(snes->npc,f);CHKERRQ(ierr);
4132f3f7c2SPeter Brune     }
4232f3f7c2SPeter Brune     ierr = VecCopy(x,y);CHKERRQ(ierr);
43*efd4aadfSBarry Smith     ierr = PetscLogEventBegin(SNES_NPCSolve,snes->npc,x,y,0);CHKERRQ(ierr);
44*efd4aadfSBarry Smith     ierr = SNESSolve(snes->npc,snes->vec_rhs,y);CHKERRQ(ierr);
45*efd4aadfSBarry Smith     ierr = PetscLogEventEnd(SNES_NPCSolve,snes->npc,x,y,0);CHKERRQ(ierr);
4632f3f7c2SPeter Brune     ierr = VecAYPX(y,-1.0,x);CHKERRQ(ierr);
4732f3f7c2SPeter Brune     PetscFunctionReturn(0);
4832f3f7c2SPeter Brune   }
4932f3f7c2SPeter Brune   PetscFunctionReturn(0);
5032f3f7c2SPeter Brune }
5132f3f7c2SPeter Brune 
52be95d8f1SBarry Smith PetscErrorCode SNESComputeFunctionDefaultNPC(SNES snes,Vec X,Vec F)
53be95d8f1SBarry Smith {
5432f3f7c2SPeter Brune /* This is to be used as an argument to SNESMF -- NOT as a "function" */
55b7281c8aSPeter Brune   SNESConvergedReason reason;
5632f3f7c2SPeter Brune   PetscErrorCode      ierr;
57b7281c8aSPeter Brune 
5832f3f7c2SPeter Brune   PetscFunctionBegin;
59*efd4aadfSBarry Smith   if (snes->npc) {
60be95d8f1SBarry Smith     ierr = SNESApplyNPC(snes,X,NULL,F);CHKERRQ(ierr);
61*efd4aadfSBarry Smith     ierr = SNESGetConvergedReason(snes->npc,&reason);CHKERRQ(ierr);
62b7281c8aSPeter Brune     if (reason < 0  && reason != SNES_DIVERGED_MAX_IT) {
63b7281c8aSPeter Brune       ierr = SNESSetFunctionDomainError(snes);CHKERRQ(ierr);
64b7281c8aSPeter Brune     }
65b7281c8aSPeter Brune   } else {
66b7281c8aSPeter Brune     ierr = SNESComputeFunction(snes,X,F);CHKERRQ(ierr);
67b7281c8aSPeter Brune   }
6832f3f7c2SPeter Brune   PetscFunctionReturn(0);
6932f3f7c2SPeter Brune }
70ddd40ce5SPeter Brune 
71ddd40ce5SPeter Brune /*@
72be95d8f1SBarry Smith    SNESGetNPCFunction - Gets the function from a preconditioner after SNESSolve() has been called.
73ddd40ce5SPeter Brune 
74ddd40ce5SPeter Brune    Collective on SNES
75ddd40ce5SPeter Brune 
76ddd40ce5SPeter Brune    Input Parameters:
77ddd40ce5SPeter Brune .  snes - the SNES context
78ddd40ce5SPeter Brune 
79ddd40ce5SPeter Brune    Output Parameter:
80ddd40ce5SPeter Brune .  F - function vector
81ddd40ce5SPeter Brune .  fnorm - the norm of F
82ddd40ce5SPeter Brune 
83ddd40ce5SPeter Brune    Level: developer
84ddd40ce5SPeter Brune 
85ddd40ce5SPeter Brune .keywords: SNES, nonlinear, function
86ddd40ce5SPeter Brune 
87be95d8f1SBarry Smith .seealso: SNESGetNPC(),SNESSetNPC(),SNESComputeFunction(),SNESApplyNPC(),SNESSolve()
88ddd40ce5SPeter Brune @*/
89be95d8f1SBarry Smith PetscErrorCode SNESGetNPCFunction(SNES snes,Vec F,PetscReal *fnorm)
90ddd40ce5SPeter Brune {
91ddd40ce5SPeter Brune   PetscErrorCode   ierr;
92ddd40ce5SPeter Brune   PCSide           npcside;
93ddd40ce5SPeter Brune   SNESFunctionType functype;
94ddd40ce5SPeter Brune   SNESNormSchedule normschedule;
95ddd40ce5SPeter Brune   Vec              FPC,XPC;
96ddd40ce5SPeter Brune 
97ddd40ce5SPeter Brune   PetscFunctionBegin;
98*efd4aadfSBarry Smith   if (snes->npc) {
99*efd4aadfSBarry Smith     ierr = SNESGetNPCSide(snes->npc,&npcside);CHKERRQ(ierr);
100*efd4aadfSBarry Smith     ierr = SNESGetFunctionType(snes->npc,&functype);CHKERRQ(ierr);
101*efd4aadfSBarry Smith     ierr = SNESGetNormSchedule(snes->npc,&normschedule);CHKERRQ(ierr);
102ddd40ce5SPeter Brune 
103ddd40ce5SPeter Brune     /* check if the function is valid based upon how the inner solver is preconditioned */
104ddd40ce5SPeter Brune     if (normschedule != SNES_NORM_NONE && normschedule != SNES_NORM_INITIAL_ONLY && (npcside == PC_RIGHT || functype == SNES_FUNCTION_UNPRECONDITIONED)) {
105*efd4aadfSBarry Smith       ierr = SNESGetFunction(snes->npc,&FPC,NULL,NULL);CHKERRQ(ierr);
1061d3ed18cSPeter Brune       if (FPC) {
10771dbe336SPeter Brune         if (fnorm) {ierr = VecNorm(FPC,NORM_2,fnorm);CHKERRQ(ierr);}
108ddd40ce5SPeter Brune         ierr = VecCopy(FPC,F);CHKERRQ(ierr);
1096c4ed002SBarry Smith       } else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"Preconditioner has no function");
110ddd40ce5SPeter Brune     } else {
111*efd4aadfSBarry Smith       ierr = SNESGetSolution(snes->npc,&XPC);CHKERRQ(ierr);
112ddd40ce5SPeter Brune       if (XPC) {
113*efd4aadfSBarry Smith         ierr = SNESComputeFunction(snes->npc,XPC,F);CHKERRQ(ierr);
114ddd40ce5SPeter Brune         if (fnorm) {ierr = VecNorm(F,NORM_2,fnorm);CHKERRQ(ierr);}
11537703b41SBarry Smith       } else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"Preconditioner has no solution");
116ddd40ce5SPeter Brune     }
11737703b41SBarry Smith   } else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"No preconditioner set");
118ddd40ce5SPeter Brune   PetscFunctionReturn(0);
119ddd40ce5SPeter Brune }
120