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