xref: /petsc/src/snes/interface/snespc.c (revision 6a63999baf6acb6d98f8927762bec1b1df13a341)
132f3f7c2SPeter Brune 
232f3f7c2SPeter Brune #include <petsc-private/snesimpl.h>      /*I "petscsnes.h"  I*/
332f3f7c2SPeter Brune #include <petscdmshell.h>
432f3f7c2SPeter Brune 
532f3f7c2SPeter Brune 
632f3f7c2SPeter Brune #undef __FUNCT__
732f3f7c2SPeter Brune #define __FUNCT__ "SNESApplyPC"
832f3f7c2SPeter Brune /*@
9*6a63999bSPeter Brune    SNESApplyPC - Runs the nonlinear preconditioner and returns the step.
1032f3f7c2SPeter Brune 
1132f3f7c2SPeter Brune    Collective on SNES
1232f3f7c2SPeter Brune 
1332f3f7c2SPeter Brune    Input Parameters:
1432f3f7c2SPeter Brune +  snes - the SNES context
15*6a63999bSPeter Brune .  x - input vector
16*6a63999bSPeter Brune -  f - optional; the function evaluation on x
1732f3f7c2SPeter Brune 
1832f3f7c2SPeter Brune    Output Parameter:
1932f3f7c2SPeter Brune .  y - function vector, as set by SNESSetFunction()
2032f3f7c2SPeter Brune 
2132f3f7c2SPeter Brune    Notes:
22*6a63999bSPeter Brune    SNESComputeFunction() should be called on x before SNESApplyPC() is called, as it is
2332f3f7c2SPeter Brune    with SNESComuteJacobian().
2432f3f7c2SPeter Brune 
2532f3f7c2SPeter Brune    Level: developer
2632f3f7c2SPeter Brune 
2732f3f7c2SPeter Brune .keywords: SNES, nonlinear, compute, function
2832f3f7c2SPeter Brune 
2932f3f7c2SPeter Brune .seealso: SNESGetPC(),SNESSetPC(),SNESComputeFunction()
3032f3f7c2SPeter Brune @*/
31*6a63999bSPeter Brune PetscErrorCode  SNESApplyPC(SNES snes,Vec x,Vec f,Vec y)
3232f3f7c2SPeter Brune {
3332f3f7c2SPeter Brune   PetscErrorCode ierr;
3432f3f7c2SPeter Brune 
3532f3f7c2SPeter Brune   PetscFunctionBegin;
3632f3f7c2SPeter Brune   PetscValidHeaderSpecific(snes,SNES_CLASSID,1);
3732f3f7c2SPeter Brune   PetscValidHeaderSpecific(x,VEC_CLASSID,2);
3832f3f7c2SPeter Brune   PetscValidHeaderSpecific(y,VEC_CLASSID,3);
3932f3f7c2SPeter Brune   PetscCheckSameComm(snes,1,x,2);
4032f3f7c2SPeter Brune   PetscCheckSameComm(snes,1,y,3);
4132f3f7c2SPeter Brune   ierr = VecValidValues(x,2,PETSC_TRUE);CHKERRQ(ierr);
4232f3f7c2SPeter Brune   if (snes->pc) {
4332f3f7c2SPeter Brune     if (f) {
4432f3f7c2SPeter Brune       ierr = SNESSetInitialFunction(snes->pc,f);CHKERRQ(ierr);
4532f3f7c2SPeter Brune     }
4632f3f7c2SPeter Brune     ierr = VecCopy(x,y);CHKERRQ(ierr);
4732f3f7c2SPeter Brune     ierr = PetscLogEventBegin(SNES_NPCSolve,snes->pc,x,y,0);CHKERRQ(ierr);
4832f3f7c2SPeter Brune     ierr = SNESSolve(snes->pc,snes->vec_rhs,y);CHKERRQ(ierr);
4932f3f7c2SPeter Brune     ierr = PetscLogEventEnd(SNES_NPCSolve,snes->pc,x,y,0);CHKERRQ(ierr);
5032f3f7c2SPeter Brune     ierr = VecAYPX(y,-1.0,x);CHKERRQ(ierr);
5132f3f7c2SPeter Brune     ierr = VecValidValues(y,3,PETSC_FALSE);CHKERRQ(ierr);
5232f3f7c2SPeter Brune     PetscFunctionReturn(0);
5332f3f7c2SPeter Brune   }
5432f3f7c2SPeter Brune   ierr = VecValidValues(y,3,PETSC_FALSE);CHKERRQ(ierr);
5532f3f7c2SPeter Brune   PetscFunctionReturn(0);
5632f3f7c2SPeter Brune }
5732f3f7c2SPeter Brune 
5832f3f7c2SPeter Brune #undef __FUNCT__
5932f3f7c2SPeter Brune #define __FUNCT__ "SNESComputeFunctionDefaultPC"
60ed07d7d7SPeter Brune PetscErrorCode SNESComputeFunctionDefaultPC(SNES snes,Vec X,Vec F) {
6132f3f7c2SPeter Brune /* This is to be used as an argument to SNESMF -- NOT as a "function" */
62b7281c8aSPeter Brune   SNESConvergedReason reason;
6332f3f7c2SPeter Brune   PetscErrorCode      ierr;
64b7281c8aSPeter Brune 
6532f3f7c2SPeter Brune   PetscFunctionBegin;
66b7281c8aSPeter Brune   if (snes->pc) {
67*6a63999bSPeter Brune     ierr = SNESApplyPC(snes,X,NULL,F);CHKERRQ(ierr);
68b7281c8aSPeter Brune     ierr = SNESGetConvergedReason(snes->pc,&reason);CHKERRQ(ierr);
69b7281c8aSPeter Brune     if (reason < 0  && reason != SNES_DIVERGED_MAX_IT) {
70b7281c8aSPeter Brune       ierr = SNESSetFunctionDomainError(snes);CHKERRQ(ierr);
71b7281c8aSPeter Brune     }
72b7281c8aSPeter Brune   } else {
73b7281c8aSPeter Brune     ierr = SNESComputeFunction(snes,X,F);CHKERRQ(ierr);
74b7281c8aSPeter Brune   }
7532f3f7c2SPeter Brune PetscFunctionReturn(0);
7632f3f7c2SPeter Brune }
77ddd40ce5SPeter Brune 
78ddd40ce5SPeter Brune #undef __FUNCT__
79ddd40ce5SPeter Brune #define __FUNCT__ "SNESGetPCFunction"
80ddd40ce5SPeter Brune /*@
81ddd40ce5SPeter Brune    SNESGetPCFunction - Gets the function from a preconditioner after SNESSolve() has been called.
82ddd40ce5SPeter Brune 
83ddd40ce5SPeter Brune    Collective on SNES
84ddd40ce5SPeter Brune 
85ddd40ce5SPeter Brune    Input Parameters:
86ddd40ce5SPeter Brune .  snes - the SNES context
87ddd40ce5SPeter Brune 
88ddd40ce5SPeter Brune    Output Parameter:
89ddd40ce5SPeter Brune .  F - function vector
90ddd40ce5SPeter Brune .  fnorm - the norm of F
91ddd40ce5SPeter Brune 
92ddd40ce5SPeter Brune    Level: developer
93ddd40ce5SPeter Brune 
94ddd40ce5SPeter Brune .keywords: SNES, nonlinear, function
95ddd40ce5SPeter Brune 
96ddd40ce5SPeter Brune .seealso: SNESGetPC(),SNESSetPC(),SNESComputeFunction(),SNESApplyPC(),SNESSolve()
97ddd40ce5SPeter Brune @*/
98ddd40ce5SPeter Brune PetscErrorCode SNESGetPCFunction(SNES snes,Vec F,PetscReal *fnorm)
99ddd40ce5SPeter Brune {
100ddd40ce5SPeter Brune   PetscErrorCode   ierr;
101ddd40ce5SPeter Brune   PCSide           npcside;
102ddd40ce5SPeter Brune   SNESFunctionType functype;
103ddd40ce5SPeter Brune   SNESNormSchedule normschedule;
104ddd40ce5SPeter Brune   Vec              FPC,XPC;
105ddd40ce5SPeter Brune 
106ddd40ce5SPeter Brune   PetscFunctionBegin;
107ddd40ce5SPeter Brune   if (snes->pc) {
108ddd40ce5SPeter Brune     ierr = SNESGetPCSide(snes->pc,&npcside);CHKERRQ(ierr);
109ddd40ce5SPeter Brune     ierr = SNESGetFunctionType(snes->pc,&functype);CHKERRQ(ierr);
110ddd40ce5SPeter Brune     ierr = SNESGetNormSchedule(snes->pc,&normschedule);CHKERRQ(ierr);
111ddd40ce5SPeter Brune 
112ddd40ce5SPeter Brune     /* check if the function is valid based upon how the inner solver is preconditioned */
113ddd40ce5SPeter Brune     if (normschedule != SNES_NORM_NONE && normschedule != SNES_NORM_INITIAL_ONLY && (npcside == PC_RIGHT || functype == SNES_FUNCTION_UNPRECONDITIONED)) {
114ddd40ce5SPeter Brune       ierr = SNESGetFunction(snes->pc,&FPC,NULL,NULL);CHKERRQ(ierr);
1151d3ed18cSPeter Brune       if (FPC) {
116ddd40ce5SPeter Brune         if (fnorm) {ierr = SNESGetFunctionNorm(snes->pc,fnorm);CHKERRQ(ierr);}
117ddd40ce5SPeter Brune         ierr = VecCopy(FPC,F);CHKERRQ(ierr);
118ddd40ce5SPeter Brune       } else {
119ddd40ce5SPeter Brune         SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"Preconditioner has no function");
120ddd40ce5SPeter Brune       }
121ddd40ce5SPeter Brune     } else {
122ddd40ce5SPeter Brune       ierr = SNESGetSolution(snes->pc,&XPC);CHKERRQ(ierr);
123ddd40ce5SPeter Brune       if (XPC) {
124ddd40ce5SPeter Brune         ierr = SNESComputeFunction(snes->pc,XPC,F);CHKERRQ(ierr);
125ddd40ce5SPeter Brune         if (fnorm) {ierr = VecNorm(F,NORM_2,fnorm);CHKERRQ(ierr);}
126ddd40ce5SPeter Brune       } else {
127ddd40ce5SPeter Brune         SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"Preconditioner has no solution");
128ddd40ce5SPeter Brune       }
129ddd40ce5SPeter Brune     }
130ddd40ce5SPeter Brune   } else {
131ddd40ce5SPeter Brune     SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONGSTATE,"No preconditioner set");
132ddd40ce5SPeter Brune   }
133ddd40ce5SPeter Brune 
134ddd40ce5SPeter Brune   PetscFunctionReturn(0);
135ddd40ce5SPeter Brune }
136