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