Actual source code: snestest.c

petsc-3.7.7 2017-09-25
Report Typos and Errors
  2: #include <petsc/private/snesimpl.h>

  4: typedef struct {
  5:   PetscBool complete_print;
  6: } SNES_Test;


 11: PetscErrorCode SNESSolve_Test(SNES snes)
 12: {
 13:   Mat            A = snes->jacobian,B;
 14:   Vec            x = snes->vec_sol,f = snes->vec_func,f1 = snes->vec_sol_update;
 16:   PetscInt       i;
 17:   PetscReal      nrm,gnorm;
 18:   SNES_Test      *neP = (SNES_Test*)snes->data;
 19:   PetscErrorCode (*objective)(SNES,Vec,PetscReal*,void*);
 20:   void           *ctx;
 21:   PetscReal      fnorm,f1norm,dnorm;

 24:   if (A != snes->jacobian_pre) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONG,"Cannot test with alternative preconditioner");

 26:   PetscPrintf(PetscObjectComm((PetscObject)snes),"Testing hand-coded Jacobian, if the ratio is\n");
 27:   PetscPrintf(PetscObjectComm((PetscObject)snes),"O(1.e-8), the hand-coded Jacobian is probably correct.\n");
 28:   if (!neP->complete_print) {
 29:     PetscPrintf(PetscObjectComm((PetscObject)snes),"Run with -snes_test_display to show difference\n");
 30:     PetscPrintf(PetscObjectComm((PetscObject)snes),"of hand-coded and finite difference Jacobian.\n");
 31:   }

 33:   for (i=0; i<3; i++) {
 34:     void                     *functx;
 35:     static const char *const loc[] = {"user-defined state","constant state -1.0","constant state 1.0"};
 36:     PetscInt                 m,n,M,N;

 38:     if (i == 1) {
 39:       VecSet(x,-1.0);
 40:     } else if (i == 2) {
 41:       VecSet(x,1.0);
 42:     }

 44:     /* evaluate the function at this point because SNESComputeJacobianDefaultColor() assumes that the function has been evaluated and put into snes->vec_func */
 45:     SNESComputeFunction(snes,x,f);
 46:     if (snes->domainerror) {
 47:       PetscPrintf(PetscObjectComm((PetscObject)snes),"Domain error at %s\n",loc[i]);
 48:       snes->domainerror = PETSC_FALSE;
 49:       continue;
 50:     }

 52:     /* compute both versions of Jacobian */
 53:     SNESComputeJacobian(snes,x,A,A);

 55:     MatCreate(PetscObjectComm((PetscObject)A),&B);
 56:     MatGetSize(A,&M,&N);
 57:     MatGetLocalSize(A,&m,&n);
 58:     MatSetSizes(B,m,n,M,N);
 59:     MatSetType(B,((PetscObject)A)->type_name);
 60:     MatSetUp(B);
 61:     MatSetOption(B,MAT_NEW_NONZERO_ALLOCATION_ERR,PETSC_FALSE);

 63:     SNESGetFunction(snes,NULL,NULL,&functx);
 64:     SNESComputeJacobianDefault(snes,x,B,B,functx);
 65:     if (neP->complete_print) {
 66:       MPI_Comm    comm;
 67:       PetscViewer viewer;
 68:       PetscPrintf(PetscObjectComm((PetscObject)snes),"Finite difference Jacobian (%s)\n",loc[i]);
 69:       PetscObjectGetComm((PetscObject)B,&comm);
 70:       PetscViewerASCIIGetStdout(comm,&viewer);
 71:       MatView(B,viewer);
 72:     }
 73:     /* compare */
 74:     MatAYPX(B,-1.0,A,DIFFERENT_NONZERO_PATTERN);
 75:     MatNorm(B,NORM_FROBENIUS,&nrm);
 76:     MatNorm(A,NORM_FROBENIUS,&gnorm);
 77:     if (neP->complete_print) {
 78:       MPI_Comm    comm;
 79:       PetscViewer viewer;
 80:       PetscPrintf(PetscObjectComm((PetscObject)snes),"Hand-coded Jacobian (%s)\n",loc[i]);
 81:       PetscObjectGetComm((PetscObject)B,&comm);
 82:       PetscViewerASCIIGetStdout(comm,&viewer);
 83:       MatView(A,viewer);
 84:       PetscPrintf(PetscObjectComm((PetscObject)snes),"Hand-coded minus finite-difference Jacobian (%s)\n",loc[i]);
 85:       MatView(B,viewer);
 86:     }
 87:     if (!gnorm) gnorm = 1; /* just in case */
 88:     PetscPrintf(PetscObjectComm((PetscObject)snes),"Norm of matrix ratio %g, difference %g (%s)\n",(double)(nrm/gnorm),(double)nrm,loc[i]);

 90:     SNESGetObjective(snes,&objective,&ctx);
 91:     if (objective) {
 92:       SNESComputeFunction(snes,x,f);
 93:       VecNorm(f,NORM_2,&fnorm);
 94:       if (neP->complete_print) {
 95:         PetscViewer viewer;
 96:         PetscPrintf(PetscObjectComm((PetscObject)snes),"Hand-coded Function (%s)\n",loc[i]);
 97:         PetscViewerASCIIGetStdout(PetscObjectComm((PetscObject)snes),&viewer);
 98:         VecView(f,viewer);
 99:       }
100:       SNESObjectiveComputeFunctionDefaultFD(snes,x,f1,NULL);
101:       VecNorm(f1,NORM_2,&f1norm);
102:       if (neP->complete_print) {
103:         PetscViewer viewer;
104:         PetscPrintf(PetscObjectComm((PetscObject)snes),"Finite-difference Function (%s)\n",loc[i]);
105:         PetscViewerASCIIGetStdout(PetscObjectComm((PetscObject)snes),&viewer);
106:         VecView(f1,viewer);
107:       }
108:       /* compare the two */
109:       VecAXPY(f,-1.0,f1);
110:       VecNorm(f,NORM_2,&dnorm);
111:       if (!fnorm) fnorm = 1.;
112:       PetscPrintf(PetscObjectComm((PetscObject)snes),"Norm of function ratio %g, difference %g (%s)\n",dnorm/fnorm,dnorm,loc[i]);
113:       if (neP->complete_print) {
114:         PetscViewer viewer;
115:         PetscPrintf(PetscObjectComm((PetscObject)snes),"Difference (%s)\n",loc[i]);
116:         PetscViewerASCIIGetStdout(PetscObjectComm((PetscObject)snes),&viewer);
117:         VecView(f,viewer);
118:       }
119:     }
120:     MatDestroy(&B);
121:   }

123:   /*
124:    Abort after the first iteration due to the jacobian not being valid.
125:   */

127:   SETERRQ(PetscObjectComm((PetscObject)snes),PETSC_ERR_ARG_WRONGSTATE,"SNESTest aborts after Jacobian test: it is NORMAL behavior.");
128:   return(0);
129: }


132: /* ------------------------------------------------------------ */
135: PetscErrorCode SNESDestroy_Test(SNES snes)
136: {

140:   PetscFree(snes->data);
141:   return(0);
142: }

146: static PetscErrorCode SNESSetFromOptions_Test(PetscOptionItems *PetscOptionsObject,SNES snes)
147: {
148:   SNES_Test      *ls = (SNES_Test*)snes->data;

152:   PetscOptionsHead(PetscOptionsObject,"Hand-coded Jacobian tester options");
153:   PetscOptionsBool("-snes_test_display","Display difference between hand-coded and finite difference Jacobians","None",ls->complete_print,&ls->complete_print,NULL);
154:   PetscOptionsTail();
155:   return(0);
156: }

160: PetscErrorCode SNESSetUp_Test(SNES snes)
161: {

165:   SNESSetUpMatrices(snes);
166:   return(0);
167: }

169: /* ------------------------------------------------------------ */
170: /*MC
171:       SNESTEST - Test hand-coded Jacobian against finite difference Jacobian

173:    Options Database:
174: +  -snes_type test    - use a SNES solver that evaluates the difference between hand-code and finite-difference Jacobians
175: -  -snes_test_display - display the elements of the matrix, the difference between the Jacobian approximated by finite-differencing and hand-coded Jacobian

177:    Level: intermediate

179:    Notes: This solver is not a solver and does not converge to a solution.  SNESTEST checks the Jacobian at three
180:    points: the 0, 1, and -1 solution vectors.  At each point the following is reported.

182:    Output:
183: +  difference - ||J - Jd||, the norm of the difference of the hand-coded Jacobian J and the approximate Jacobian Jd obtained by finite-differencing
184:    the residual,
185: -  ratio      - ||J - Jd||/||J||, the ratio of the norms of the above difference and the hand-coded Jacobian.

187:    Frobenius norm is used in the above throughout. After doing these three tests, it always aborts with the error message
188:    "SNESTest aborts after Jacobian test".  No other behavior is to be expected.  It may be similarly used to check if a
189:    SNES function is the gradient of an objective function set with SNESSetObjective().

191: .seealso:  SNESCreate(), SNES, SNESSetType(), SNESUpdateCheckJacobian(), SNESNEWTONLS, SNESNEWTONTR

193: M*/
196: PETSC_EXTERN PetscErrorCode SNESCreate_Test(SNES snes)
197: {
198:   SNES_Test      *neP;

202:   snes->ops->solve          = SNESSolve_Test;
203:   snes->ops->destroy        = SNESDestroy_Test;
204:   snes->ops->setfromoptions = SNESSetFromOptions_Test;
205:   snes->ops->view           = 0;
206:   snes->ops->setup          = SNESSetUp_Test;
207:   snes->ops->reset          = 0;

209:   snes->usesksp = PETSC_FALSE;

211:   PetscNewLog(snes,&neP);
212:   snes->data          = (void*)neP;
213:   neP->complete_print = PETSC_FALSE;
214:   return(0);
215: }

219: /*@C
220:     SNESUpdateCheckJacobian - Checks each Jacobian computed by the nonlinear solver comparing the users function with a finite difference computation.

222:    Options Database:
223: +    -snes_check_jacobian - use this every time SNESSolve() is called
224: -    -snes_check_jacobian_view -  Display difference between Jacobian approximated by finite-differencing and the hand-coded Jacobian

226:    Output:
227: +  difference - ||J - Jd||, the norm of the difference of the hand-coded Jacobian J and the approximate Jacobian Jd obtained by finite-differencing
228:    the residual,
229: -  ratio      - ||J - Jd||/||J||, the ratio of the norms of the above difference and the hand-coded Jacobian.

231:    Notes:
232:    Frobenius norm is used in the above throughout.  This check is carried out every SNES iteration.

234:    Level: intermediate

236: .seealso:  SNESTEST, SNESCreate(), SNES, SNESSetType(), SNESNEWTONLS, SNESNEWTONTR, SNESSolve()

238: @*/
239: PetscErrorCode SNESUpdateCheckJacobian(SNES snes,PetscInt it)
240: {
241:   Mat            A = snes->jacobian,B;
242:   Vec            x = snes->vec_sol,f = snes->vec_func,f1 = snes->vec_sol_update;
244:   PetscReal      nrm,gnorm;
245:   PetscErrorCode (*objective)(SNES,Vec,PetscReal*,void*);
246:   void           *ctx;
247:   PetscReal      fnorm,f1norm,dnorm;
248:   PetscInt       m,n,M,N;
249:   PetscBool      complete_print = PETSC_FALSE;
250:   void           *functx;
251:   PetscViewer    viewer = PETSC_VIEWER_STDOUT_(PetscObjectComm((PetscObject)snes));

254:   PetscOptionsHasName(((PetscObject)snes)->options,((PetscObject)snes)->prefix,"-snes_check_jacobian_view",&complete_print);
255:   if (A != snes->jacobian_pre) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONG,"Cannot check Jacobian with alternative preconditioner");

257:   PetscViewerASCIIAddTab(viewer,((PetscObject)snes)->tablevel);
258:   PetscViewerASCIIPrintf(viewer,"      Testing hand-coded Jacobian, if the ratio is O(1.e-8), the hand-coded Jacobian is probably correct.\n");
259:   if (!complete_print) {
260:     PetscViewerASCIIPrintf(viewer,"      Run with -snes_check_jacobian_view [viewer][:filename][:format] to show difference of hand-coded and finite difference Jacobian.\n");
261:   }

263:   /* compute both versions of Jacobian */
264:   SNESComputeJacobian(snes,x,A,A);

266:   MatCreate(PetscObjectComm((PetscObject)A),&B);
267:   MatGetSize(A,&M,&N);
268:   MatGetLocalSize(A,&m,&n);
269:   MatSetSizes(B,m,n,M,N);
270:   MatSetType(B,((PetscObject)A)->type_name);
271:   MatSetUp(B);
272:   MatSetOption(B,MAT_NEW_NONZERO_ALLOCATION_ERR,PETSC_FALSE);
273:   SNESGetFunction(snes,NULL,NULL,&functx);
274:   SNESComputeJacobianDefault(snes,x,B,B,functx);

276:   if (complete_print) {
277:     PetscViewerASCIIPrintf(viewer,"    Finite difference Jacobian\n");
278:     MatViewFromOptions(B,(PetscObject)snes,"-snes_check_jacobian_view");
279:   }
280:   /* compare */
281:   MatAYPX(B,-1.0,A,DIFFERENT_NONZERO_PATTERN);
282:   MatNorm(B,NORM_FROBENIUS,&nrm);
283:   MatNorm(A,NORM_FROBENIUS,&gnorm);
284:   if (complete_print) {
285:     PetscViewerASCIIPrintf(viewer,"    Hand-coded Jacobian\n");
286:     MatViewFromOptions(A,(PetscObject)snes,"-snes_check_jacobian_view");
287:     PetscViewerASCIIPrintf(viewer,"    Hand-coded minus finite difference Jacobian\n");
288:     MatViewFromOptions(B,(PetscObject)snes,"-snes_check_jacobian_view");
289:   }
290:   if (!gnorm) gnorm = 1; /* just in case */
291:   PetscViewerASCIIPrintf(viewer,"    %g = ||J - Jfd||/||J|| %g  = ||J - Jfd||\n",(double)(nrm/gnorm),(double)nrm);

293:   SNESGetObjective(snes,&objective,&ctx);
294:   if (objective) {
295:     SNESComputeFunction(snes,x,f);
296:     VecNorm(f,NORM_2,&fnorm);
297:     if (complete_print) {
298:       PetscViewerASCIIPrintf(viewer,"    Hand-coded Objective Function \n");
299:       VecView(f,viewer);
300:     }
301:     SNESObjectiveComputeFunctionDefaultFD(snes,x,f1,NULL);
302:     VecNorm(f1,NORM_2,&f1norm);
303:     if (complete_print) {
304:       PetscViewerASCIIPrintf(viewer,"    Finite-Difference Objective Function\n");
305:       VecView(f1,viewer);
306:     }
307:     /* compare the two */
308:     VecAXPY(f,-1.0,f1);
309:     VecNorm(f,NORM_2,&dnorm);
310:     if (!fnorm) fnorm = 1.;
311:     PetscViewerASCIIPrintf(viewer,"    %g = Norm of objective function ratio %g = difference\n",dnorm/fnorm,dnorm);
312:     if (complete_print) {
313:       PetscViewerASCIIPrintf(viewer,"    Difference\n");
314:       VecView(f,viewer);
315:     }
316:   }
317:   PetscViewerASCIISubtractTab(viewer,((PetscObject)snes)->tablevel);

319:   MatDestroy(&B);
320:   return(0);
321: }