Actual source code: nepsolve.c
slepc-main 2025-01-19
1: /*
2: - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
3: SLEPc - Scalable Library for Eigenvalue Problem Computations
4: Copyright (c) 2002-, Universitat Politecnica de Valencia, Spain
6: This file is part of SLEPc.
7: SLEPc is distributed under a 2-clause BSD license (see LICENSE).
8: - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
9: */
10: /*
11: NEP routines related to the solution process
13: References:
15: [1] C. Campos and J.E. Roman, "NEP: a module for the parallel solution
16: of nonlinear eigenvalue problems in SLEPc", ACM Trans. Math. Soft.
17: 47(3), 23:1--23:29, 2021.
18: */
20: #include <slepc/private/nepimpl.h>
21: #include <slepc/private/bvimpl.h>
22: #include <petscdraw.h>
24: static PetscBool cited = PETSC_FALSE;
25: static const char citation[] =
26: "@Article{slepc-nep,\n"
27: " author = \"C. Campos and J. E. Roman\",\n"
28: " title = \"{NEP}: a module for the parallel solution of nonlinear eigenvalue problems in {SLEPc}\",\n"
29: " journal = \"{ACM} Trans. Math. Software\",\n"
30: " volume = \"47\",\n"
31: " number = \"3\",\n"
32: " pages = \"23:1--23:29\",\n"
33: " year = \"2021\",\n"
34: " doi = \"10.1145/3447544\"\n"
35: "}\n";
37: PetscErrorCode NEPComputeVectors(NEP nep)
38: {
39: PetscFunctionBegin;
40: NEPCheckSolved(nep,1);
41: if (nep->state==NEP_STATE_SOLVED) PetscTryTypeMethod(nep,computevectors);
42: nep->state = NEP_STATE_EIGENVECTORS;
43: PetscFunctionReturn(PETSC_SUCCESS);
44: }
46: /*@
47: NEPSolve - Solves the nonlinear eigensystem.
49: Collective
51: Input Parameter:
52: . nep - eigensolver context obtained from NEPCreate()
54: Options Database Keys:
55: + -nep_view - print information about the solver used
56: . -nep_view_matk - view the split form matrix Ak (replace k by an integer from 0 to nt-1)
57: . -nep_view_fnk - view the split form function fk (replace k by an integer from 0 to nt-1)
58: . -nep_view_vectors - view the computed eigenvectors
59: . -nep_view_values - view the computed eigenvalues
60: . -nep_converged_reason - print reason for convergence, and number of iterations
61: . -nep_error_absolute - print absolute errors of each eigenpair
62: . -nep_error_relative - print relative errors of each eigenpair
63: - -nep_error_backward - print backward errors of each eigenpair
65: Notes:
66: All the command-line options listed above admit an optional argument specifying
67: the viewer type and options. For instance, use '-nep_view_vectors binary:myvecs.bin'
68: to save the eigenvectors to a binary file, '-nep_view_values draw' to draw the computed
69: eigenvalues graphically, or '-nep_error_relative :myerr.m:ascii_matlab' to save
70: the errors in a file that can be executed in Matlab.
72: Level: beginner
74: .seealso: NEPCreate(), NEPSetUp(), NEPDestroy(), NEPSetTolerances()
75: @*/
76: PetscErrorCode NEPSolve(NEP nep)
77: {
78: PetscInt i;
79: char str[16];
81: PetscFunctionBegin;
83: if (nep->state>=NEP_STATE_SOLVED) PetscFunctionReturn(PETSC_SUCCESS);
84: PetscCall(PetscCitationsRegister(citation,&cited));
85: PetscCall(PetscLogEventBegin(NEP_Solve,nep,0,0,0));
87: /* call setup */
88: PetscCall(NEPSetUp(nep));
89: nep->nconv = 0;
90: nep->its = 0;
91: for (i=0;i<nep->ncv;i++) {
92: nep->eigr[i] = 0.0;
93: nep->eigi[i] = 0.0;
94: nep->errest[i] = 0.0;
95: nep->perm[i] = i;
96: }
97: PetscCall(NEPViewFromOptions(nep,NULL,"-nep_view_pre"));
98: PetscCall(RGViewFromOptions(nep->rg,NULL,"-rg_view"));
100: /* call solver */
101: PetscUseTypeMethod(nep,solve);
102: PetscCheck(nep->reason,PetscObjectComm((PetscObject)nep),PETSC_ERR_PLIB,"Internal error, solver returned without setting converged reason");
103: nep->state = NEP_STATE_SOLVED;
105: /* Only the first nconv columns contain useful information */
106: PetscCall(BVSetActiveColumns(nep->V,0,nep->nconv));
107: if (nep->twosided) PetscCall(BVSetActiveColumns(nep->W,0,nep->nconv));
109: if (nep->refine==NEP_REFINE_SIMPLE && nep->rits>0 && nep->nconv>0) {
110: PetscCall(NEPComputeVectors(nep));
111: PetscCall(NEPNewtonRefinementSimple(nep,&nep->rits,nep->rtol,nep->nconv));
112: nep->state = NEP_STATE_EIGENVECTORS;
113: }
115: /* sort eigenvalues according to nep->which parameter */
116: PetscCall(SlepcSortEigenvalues(nep->sc,nep->nconv,nep->eigr,nep->eigi,nep->perm));
117: PetscCall(PetscLogEventEnd(NEP_Solve,nep,0,0,0));
119: /* various viewers */
120: PetscCall(NEPViewFromOptions(nep,NULL,"-nep_view"));
121: PetscCall(NEPConvergedReasonViewFromOptions(nep));
122: PetscCall(NEPErrorViewFromOptions(nep));
123: PetscCall(NEPValuesViewFromOptions(nep));
124: PetscCall(NEPVectorsViewFromOptions(nep));
125: if (nep->fui==NEP_USER_INTERFACE_SPLIT) {
126: for (i=0;i<nep->nt;i++) {
127: PetscCall(PetscSNPrintf(str,sizeof(str),"-nep_view_mat%" PetscInt_FMT,i));
128: PetscCall(MatViewFromOptions(nep->A[i],(PetscObject)nep,str));
129: PetscCall(PetscSNPrintf(str,sizeof(str),"-nep_view_fn%" PetscInt_FMT,i));
130: PetscCall(FNViewFromOptions(nep->f[i],(PetscObject)nep,str));
131: }
132: }
134: /* Remove the initial subspace */
135: nep->nini = 0;
137: /* Reset resolvent information */
138: PetscCall(MatDestroy(&nep->resolvent));
139: PetscFunctionReturn(PETSC_SUCCESS);
140: }
142: /*@
143: NEPProjectOperator - Computes the projection of the nonlinear operator.
145: Collective
147: Input Parameters:
148: + nep - the nonlinear eigensolver context
149: . j0 - initial index
150: - j1 - final index
152: Notes:
153: This is available for split operator only.
155: The nonlinear operator T(lambda) is projected onto span(V), where V is
156: an orthonormal basis built internally by the solver. The projected
157: operator is equal to sum_i V'*A_i*V*f_i(lambda), so this function
158: computes all matrices Ei = V'*A_i*V, and stores them in the extra
159: matrices inside DS. Only rows/columns in the range [j0,j1-1] are computed,
160: the previous ones are assumed to be available already.
162: Level: developer
164: .seealso: NEPSetSplitOperator()
165: @*/
166: PetscErrorCode NEPProjectOperator(NEP nep,PetscInt j0,PetscInt j1)
167: {
168: PetscInt k;
169: Mat G;
171: PetscFunctionBegin;
175: NEPCheckProblem(nep,1);
176: NEPCheckSplit(nep,1);
177: PetscCall(BVSetActiveColumns(nep->V,j0,j1));
178: for (k=0;k<nep->nt;k++) {
179: PetscCall(DSGetMat(nep->ds,DSMatExtra[k],&G));
180: PetscCall(BVMatProject(nep->V,nep->A[k],nep->V,G));
181: PetscCall(DSRestoreMat(nep->ds,DSMatExtra[k],&G));
182: }
183: PetscFunctionReturn(PETSC_SUCCESS);
184: }
186: /*@
187: NEPApplyFunction - Applies the nonlinear function T(lambda) to a given vector.
189: Collective
191: Input Parameters:
192: + nep - the nonlinear eigensolver context
193: . lambda - scalar argument
194: . x - vector to be multiplied against
195: - v - workspace vector (used only in the case of split form)
197: Output Parameters:
198: + y - result vector
199: . A - (optional) Function matrix, for callback interface only
200: - B - (unused) preconditioning matrix
202: Note:
203: If the nonlinear operator is represented in split form, the result
204: y = T(lambda)*x is computed without building T(lambda) explicitly. In
205: that case, parameters A and B are not used. Otherwise, the matrix
206: T(lambda) is built and the effect is the same as a call to
207: NEPComputeFunction() followed by a MatMult().
209: Level: developer
211: .seealso: NEPSetSplitOperator(), NEPComputeFunction(), NEPApplyAdjoint()
212: @*/
213: PetscErrorCode NEPApplyFunction(NEP nep,PetscScalar lambda,Vec x,Vec v,Vec y,Mat A,Mat B)
214: {
215: PetscInt i;
216: PetscScalar alpha;
218: PetscFunctionBegin;
227: if (nep->fui==NEP_USER_INTERFACE_SPLIT) {
228: PetscCall(VecSet(y,0.0));
229: for (i=0;i<nep->nt;i++) {
230: PetscCall(FNEvaluateFunction(nep->f[i],lambda,&alpha));
231: PetscCall(MatMult(nep->A[i],x,v));
232: PetscCall(VecAXPY(y,alpha,v));
233: }
234: } else {
235: if (!A) A = nep->function;
236: PetscCall(NEPComputeFunction(nep,lambda,A,A));
237: PetscCall(MatMult(A,x,y));
238: }
239: PetscFunctionReturn(PETSC_SUCCESS);
240: }
242: /*@
243: NEPApplyAdjoint - Applies the adjoint nonlinear function T(lambda)^* to a given vector.
245: Collective
247: Input Parameters:
248: + nep - the nonlinear eigensolver context
249: . lambda - scalar argument
250: . x - vector to be multiplied against
251: - v - workspace vector (used only in the case of split form)
253: Output Parameters:
254: + y - result vector
255: . A - (optional) Function matrix, for callback interface only
256: - B - (unused) preconditioning matrix
258: Level: developer
260: .seealso: NEPSetSplitOperator(), NEPComputeFunction(), NEPApplyFunction()
261: @*/
262: PetscErrorCode NEPApplyAdjoint(NEP nep,PetscScalar lambda,Vec x,Vec v,Vec y,Mat A,Mat B)
263: {
264: PetscInt i;
265: PetscScalar alpha;
266: Vec w;
268: PetscFunctionBegin;
277: PetscCall(VecDuplicate(x,&w));
278: PetscCall(VecCopy(x,w));
279: PetscCall(VecConjugate(w));
280: if (nep->fui==NEP_USER_INTERFACE_SPLIT) {
281: PetscCall(VecSet(y,0.0));
282: for (i=0;i<nep->nt;i++) {
283: PetscCall(FNEvaluateFunction(nep->f[i],lambda,&alpha));
284: PetscCall(MatMultTranspose(nep->A[i],w,v));
285: PetscCall(VecAXPY(y,alpha,v));
286: }
287: } else {
288: if (!A) A = nep->function;
289: PetscCall(NEPComputeFunction(nep,lambda,A,A));
290: PetscCall(MatMultTranspose(A,w,y));
291: }
292: PetscCall(VecDestroy(&w));
293: PetscCall(VecConjugate(y));
294: PetscFunctionReturn(PETSC_SUCCESS);
295: }
297: /*@
298: NEPApplyJacobian - Applies the nonlinear Jacobian T'(lambda) to a given vector.
300: Collective
302: Input Parameters:
303: + nep - the nonlinear eigensolver context
304: . lambda - scalar argument
305: . x - vector to be multiplied against
306: - v - workspace vector (used only in the case of split form)
308: Output Parameters:
309: + y - result vector
310: - A - (optional) Jacobian matrix, for callback interface only
312: Note:
313: If the nonlinear operator is represented in split form, the result
314: y = T'(lambda)*x is computed without building T'(lambda) explicitly. In
315: that case, parameter A is not used. Otherwise, the matrix
316: T'(lambda) is built and the effect is the same as a call to
317: NEPComputeJacobian() followed by a MatMult().
319: Level: developer
321: .seealso: NEPSetSplitOperator(), NEPComputeJacobian()
322: @*/
323: PetscErrorCode NEPApplyJacobian(NEP nep,PetscScalar lambda,Vec x,Vec v,Vec y,Mat A)
324: {
325: PetscInt i;
326: PetscScalar alpha;
328: PetscFunctionBegin;
336: if (nep->fui==NEP_USER_INTERFACE_SPLIT) {
337: PetscCall(VecSet(y,0.0));
338: for (i=0;i<nep->nt;i++) {
339: PetscCall(FNEvaluateDerivative(nep->f[i],lambda,&alpha));
340: PetscCall(MatMult(nep->A[i],x,v));
341: PetscCall(VecAXPY(y,alpha,v));
342: }
343: } else {
344: if (!A) A = nep->jacobian;
345: PetscCall(NEPComputeJacobian(nep,lambda,A));
346: PetscCall(MatMult(A,x,y));
347: }
348: PetscFunctionReturn(PETSC_SUCCESS);
349: }
351: /*@
352: NEPGetIterationNumber - Gets the current iteration number. If the
353: call to NEPSolve() is complete, then it returns the number of iterations
354: carried out by the solution method.
356: Not Collective
358: Input Parameter:
359: . nep - the nonlinear eigensolver context
361: Output Parameter:
362: . its - number of iterations
364: Note:
365: During the i-th iteration this call returns i-1. If NEPSolve() is
366: complete, then parameter "its" contains either the iteration number at
367: which convergence was successfully reached, or failure was detected.
368: Call NEPGetConvergedReason() to determine if the solver converged or
369: failed and why.
371: Level: intermediate
373: .seealso: NEPGetConvergedReason(), NEPSetTolerances()
374: @*/
375: PetscErrorCode NEPGetIterationNumber(NEP nep,PetscInt *its)
376: {
377: PetscFunctionBegin;
379: PetscAssertPointer(its,2);
380: *its = nep->its;
381: PetscFunctionReturn(PETSC_SUCCESS);
382: }
384: /*@
385: NEPGetConverged - Gets the number of converged eigenpairs.
387: Not Collective
389: Input Parameter:
390: . nep - the nonlinear eigensolver context
392: Output Parameter:
393: . nconv - number of converged eigenpairs
395: Note:
396: This function should be called after NEPSolve() has finished.
398: Level: beginner
400: .seealso: NEPSetDimensions(), NEPSolve(), NEPGetEigenpair()
401: @*/
402: PetscErrorCode NEPGetConverged(NEP nep,PetscInt *nconv)
403: {
404: PetscFunctionBegin;
406: PetscAssertPointer(nconv,2);
407: NEPCheckSolved(nep,1);
408: *nconv = nep->nconv;
409: PetscFunctionReturn(PETSC_SUCCESS);
410: }
412: /*@
413: NEPGetConvergedReason - Gets the reason why the NEPSolve() iteration was
414: stopped.
416: Not Collective
418: Input Parameter:
419: . nep - the nonlinear eigensolver context
421: Output Parameter:
422: . reason - negative value indicates diverged, positive value converged
424: Options Database Key:
425: . -nep_converged_reason - print the reason to a viewer
427: Notes:
428: Possible values for reason are
429: + NEP_CONVERGED_TOL - converged up to tolerance
430: . NEP_CONVERGED_USER - converged due to a user-defined condition
431: . NEP_DIVERGED_ITS - required more than max_it iterations to reach convergence
432: . NEP_DIVERGED_BREAKDOWN - generic breakdown in method
433: . NEP_DIVERGED_LINEAR_SOLVE - inner linear solve failed
434: - NEP_DIVERGED_SUBSPACE_EXHAUSTED - run out of space for the basis in an
435: unrestarted solver
437: Can only be called after the call to NEPSolve() is complete.
439: Level: intermediate
441: .seealso: NEPSetTolerances(), NEPSolve(), NEPConvergedReason
442: @*/
443: PetscErrorCode NEPGetConvergedReason(NEP nep,NEPConvergedReason *reason)
444: {
445: PetscFunctionBegin;
447: PetscAssertPointer(reason,2);
448: NEPCheckSolved(nep,1);
449: *reason = nep->reason;
450: PetscFunctionReturn(PETSC_SUCCESS);
451: }
453: /*@
454: NEPGetEigenpair - Gets the i-th solution of the eigenproblem as computed by
455: NEPSolve(). The solution consists in both the eigenvalue and the eigenvector.
457: Collective
459: Input Parameters:
460: + nep - nonlinear eigensolver context
461: - i - index of the solution
463: Output Parameters:
464: + eigr - real part of eigenvalue
465: . eigi - imaginary part of eigenvalue
466: . Vr - real part of eigenvector
467: - Vi - imaginary part of eigenvector
469: Notes:
470: It is allowed to pass NULL for Vr and Vi, if the eigenvector is not
471: required. Otherwise, the caller must provide valid Vec objects, i.e.,
472: they must be created by the calling program with e.g. MatCreateVecs().
474: If the eigenvalue is real, then eigi and Vi are set to zero. If PETSc is
475: configured with complex scalars the eigenvalue is stored
476: directly in eigr (eigi is set to zero) and the eigenvector in Vr (Vi is
477: set to zero). In any case, the user can pass NULL in Vr or Vi if one of
478: them is not required.
480: The index i should be a value between 0 and nconv-1 (see NEPGetConverged()).
481: Eigenpairs are indexed according to the ordering criterion established
482: with NEPSetWhichEigenpairs().
484: Level: beginner
486: .seealso: NEPSolve(), NEPGetConverged(), NEPSetWhichEigenpairs(), NEPGetLeftEigenvector()
487: @*/
488: PetscErrorCode NEPGetEigenpair(NEP nep,PetscInt i,PetscScalar *eigr,PetscScalar *eigi,Vec Vr,Vec Vi)
489: {
490: PetscInt k;
492: PetscFunctionBegin;
497: NEPCheckSolved(nep,1);
498: PetscCheck(i>=0,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"The index cannot be negative");
499: PetscCheck(i<nep->nconv,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"The index can be nconv-1 at most, see NEPGetConverged()");
501: PetscCall(NEPComputeVectors(nep));
502: k = nep->perm[i];
504: /* eigenvalue */
505: #if defined(PETSC_USE_COMPLEX)
506: if (eigr) *eigr = nep->eigr[k];
507: if (eigi) *eigi = 0;
508: #else
509: if (eigr) *eigr = nep->eigr[k];
510: if (eigi) *eigi = nep->eigi[k];
511: #endif
513: /* eigenvector */
514: PetscCall(BV_GetEigenvector(nep->V,k,nep->eigi[k],Vr,Vi));
515: PetscFunctionReturn(PETSC_SUCCESS);
516: }
518: /*@
519: NEPGetLeftEigenvector - Gets the i-th left eigenvector as computed by NEPSolve().
521: Collective
523: Input Parameters:
524: + nep - eigensolver context
525: - i - index of the solution
527: Output Parameters:
528: + Wr - real part of left eigenvector
529: - Wi - imaginary part of left eigenvector
531: Notes:
532: The caller must provide valid Vec objects, i.e., they must be created
533: by the calling program with e.g. MatCreateVecs().
535: If the corresponding eigenvalue is real, then Wi is set to zero. If PETSc is
536: configured with complex scalars the eigenvector is stored directly in Wr
537: (Wi is set to zero). In any case, the user can pass NULL in Wr or Wi if one of
538: them is not required.
540: The index i should be a value between 0 and nconv-1 (see NEPGetConverged()).
541: Eigensolutions are indexed according to the ordering criterion established
542: with NEPSetWhichEigenpairs().
544: Left eigenvectors are available only if the twosided flag was set, see
545: NEPSetTwoSided().
547: Level: intermediate
549: .seealso: NEPGetEigenpair(), NEPGetConverged(), NEPSetWhichEigenpairs(), NEPSetTwoSided()
550: @*/
551: PetscErrorCode NEPGetLeftEigenvector(NEP nep,PetscInt i,Vec Wr,Vec Wi)
552: {
553: PetscInt k;
555: PetscFunctionBegin;
560: NEPCheckSolved(nep,1);
561: PetscCheck(nep->twosided,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_WRONGSTATE,"Must request left vectors with NEPSetTwoSided");
562: PetscCheck(i>=0,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"The index cannot be negative");
563: PetscCheck(i<nep->nconv,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"The index can be nconv-1 at most, see NEPGetConverged()");
564: PetscCall(NEPComputeVectors(nep));
565: k = nep->perm[i];
566: PetscCall(BV_GetEigenvector(nep->W,k,nep->eigi[k],Wr,Wi));
567: PetscFunctionReturn(PETSC_SUCCESS);
568: }
570: /*@
571: NEPGetErrorEstimate - Returns the error estimate associated to the i-th
572: computed eigenpair.
574: Not Collective
576: Input Parameters:
577: + nep - nonlinear eigensolver context
578: - i - index of eigenpair
580: Output Parameter:
581: . errest - the error estimate
583: Notes:
584: This is the error estimate used internally by the eigensolver. The actual
585: error bound can be computed with NEPComputeError().
587: Level: advanced
589: .seealso: NEPComputeError()
590: @*/
591: PetscErrorCode NEPGetErrorEstimate(NEP nep,PetscInt i,PetscReal *errest)
592: {
593: PetscFunctionBegin;
595: PetscAssertPointer(errest,3);
596: NEPCheckSolved(nep,1);
597: PetscCheck(i>=0,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"The index cannot be negative");
598: PetscCheck(i<nep->nconv,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"The index can be nconv-1 at most, see NEPGetConverged()");
599: *errest = nep->errest[nep->perm[i]];
600: PetscFunctionReturn(PETSC_SUCCESS);
601: }
603: /*
604: NEPComputeResidualNorm_Private - Computes the norm of the residual vector
605: associated with an eigenpair.
607: Input Parameters:
608: adj - whether the adjoint T^* must be used instead of T
609: lambda - eigenvalue
610: x - eigenvector
611: w - array of work vectors (two vectors in split form, one vector otherwise)
612: */
613: PetscErrorCode NEPComputeResidualNorm_Private(NEP nep,PetscBool adj,PetscScalar lambda,Vec x,Vec *w,PetscReal *norm)
614: {
615: Vec y,z=NULL;
617: PetscFunctionBegin;
618: y = w[0];
619: if (nep->fui==NEP_USER_INTERFACE_SPLIT) z = w[1];
620: if (adj) PetscCall(NEPApplyAdjoint(nep,lambda,x,z,y,NULL,NULL));
621: else PetscCall(NEPApplyFunction(nep,lambda,x,z,y,NULL,NULL));
622: PetscCall(VecNorm(y,NORM_2,norm));
623: PetscFunctionReturn(PETSC_SUCCESS);
624: }
626: /*@
627: NEPComputeError - Computes the error (based on the residual norm) associated
628: with the i-th computed eigenpair.
630: Collective
632: Input Parameters:
633: + nep - the nonlinear eigensolver context
634: . i - the solution index
635: - type - the type of error to compute
637: Output Parameter:
638: . error - the error
640: Notes:
641: The error can be computed in various ways, all of them based on the residual
642: norm computed as ||T(lambda)x||_2 where lambda is the eigenvalue and x is the
643: eigenvector.
645: Level: beginner
647: .seealso: NEPErrorType, NEPSolve(), NEPGetErrorEstimate()
648: @*/
649: PetscErrorCode NEPComputeError(NEP nep,PetscInt i,NEPErrorType type,PetscReal *error)
650: {
651: Vec xr,xi=NULL;
652: PetscInt j,nwork,issplit=0;
653: PetscScalar kr,ki,s;
654: PetscReal er,z=0.0,errorl,nrm;
655: PetscBool flg;
657: PetscFunctionBegin;
661: PetscAssertPointer(error,4);
662: NEPCheckSolved(nep,1);
664: /* allocate work vectors */
665: #if defined(PETSC_USE_COMPLEX)
666: nwork = 2;
667: #else
668: nwork = 3;
669: #endif
670: if (nep->fui==NEP_USER_INTERFACE_SPLIT) {
671: issplit = 1;
672: nwork++; /* need an extra work vector for NEPComputeResidualNorm_Private */
673: }
674: PetscCall(NEPSetWorkVecs(nep,nwork));
675: xr = nep->work[issplit+1];
676: #if !defined(PETSC_USE_COMPLEX)
677: xi = nep->work[issplit+2];
678: #endif
680: /* compute residual norms */
681: PetscCall(NEPGetEigenpair(nep,i,&kr,&ki,xr,xi));
682: #if !defined(PETSC_USE_COMPLEX)
683: PetscCheck(ki==0.0,PetscObjectComm((PetscObject)nep),PETSC_ERR_SUP,"Not implemented for complex eigenvalues with real scalars");
684: #endif
685: PetscCall(NEPComputeResidualNorm_Private(nep,PETSC_FALSE,kr,xr,nep->work,error));
686: PetscCall(VecNorm(xr,NORM_2,&er));
688: /* if two-sided, compute left residual norm and take the maximum */
689: if (nep->twosided) {
690: PetscCall(NEPGetLeftEigenvector(nep,i,xr,xi));
691: PetscCall(NEPComputeResidualNorm_Private(nep,PETSC_TRUE,kr,xr,nep->work,&errorl));
692: *error = PetscMax(*error,errorl);
693: }
695: /* compute error */
696: switch (type) {
697: case NEP_ERROR_ABSOLUTE:
698: break;
699: case NEP_ERROR_RELATIVE:
700: *error /= PetscAbsScalar(kr)*er;
701: break;
702: case NEP_ERROR_BACKWARD:
703: if (nep->fui!=NEP_USER_INTERFACE_SPLIT) {
704: PetscCall(NEPComputeFunction(nep,kr,nep->function,nep->function));
705: PetscCall(MatHasOperation(nep->function,MATOP_NORM,&flg));
706: PetscCheck(flg,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_WRONG,"The computation of backward errors requires a matrix norm operation");
707: PetscCall(MatNorm(nep->function,NORM_INFINITY,&nrm));
708: *error /= nrm*er;
709: break;
710: }
711: /* initialization of matrix norms */
712: if (!nep->nrma[0]) {
713: for (j=0;j<nep->nt;j++) {
714: PetscCall(MatHasOperation(nep->A[j],MATOP_NORM,&flg));
715: PetscCheck(flg,PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_WRONG,"The computation of backward errors requires a matrix norm operation");
716: PetscCall(MatNorm(nep->A[j],NORM_INFINITY,&nep->nrma[j]));
717: }
718: }
719: for (j=0;j<nep->nt;j++) {
720: PetscCall(FNEvaluateFunction(nep->f[j],kr,&s));
721: z = z + nep->nrma[j]*PetscAbsScalar(s);
722: }
723: *error /= z*er;
724: break;
725: default:
726: SETERRQ(PetscObjectComm((PetscObject)nep),PETSC_ERR_ARG_OUTOFRANGE,"Invalid error type");
727: }
728: PetscFunctionReturn(PETSC_SUCCESS);
729: }
731: /*@
732: NEPComputeFunction - Computes the function matrix T(lambda) that has been
733: set with NEPSetFunction().
735: Collective
737: Input Parameters:
738: + nep - the NEP context
739: - lambda - the scalar argument
741: Output Parameters:
742: + A - Function matrix
743: - B - optional preconditioning matrix
745: Notes:
746: NEPComputeFunction() is typically used within nonlinear eigensolvers
747: implementations, so most users would not generally call this routine
748: themselves.
750: Level: developer
752: .seealso: NEPSetFunction(), NEPGetFunction()
753: @*/
754: PetscErrorCode NEPComputeFunction(NEP nep,PetscScalar lambda,Mat A,Mat B)
755: {
756: PetscInt i;
757: PetscScalar alpha;
759: PetscFunctionBegin;
761: NEPCheckProblem(nep,1);
762: switch (nep->fui) {
763: case NEP_USER_INTERFACE_CALLBACK:
764: PetscCheck(nep->computefunction,PetscObjectComm((PetscObject)nep),PETSC_ERR_USER,"Must call NEPSetFunction() first");
765: PetscCall(PetscLogEventBegin(NEP_FunctionEval,nep,A,B,0));
766: PetscCallBack("NEP user Function function",(*nep->computefunction)(nep,lambda,A,B,nep->functionctx));
767: PetscCall(PetscLogEventEnd(NEP_FunctionEval,nep,A,B,0));
768: break;
769: case NEP_USER_INTERFACE_SPLIT:
770: PetscCall(MatZeroEntries(A));
771: if (A != B) PetscCall(MatZeroEntries(B));
772: for (i=0;i<nep->nt;i++) {
773: PetscCall(FNEvaluateFunction(nep->f[i],lambda,&alpha));
774: PetscCall(MatAXPY(A,alpha,nep->A[i],nep->mstr));
775: if (A != B) PetscCall(MatAXPY(B,alpha,nep->P[i],nep->mstrp));
776: }
777: break;
778: }
779: PetscFunctionReturn(PETSC_SUCCESS);
780: }
782: /*@
783: NEPComputeJacobian - Computes the Jacobian matrix T'(lambda) that has been
784: set with NEPSetJacobian().
786: Collective
788: Input Parameters:
789: + nep - the NEP context
790: - lambda - the scalar argument
792: Output Parameters:
793: . A - Jacobian matrix
795: Notes:
796: Most users should not need to explicitly call this routine, as it
797: is used internally within the nonlinear eigensolvers.
799: Level: developer
801: .seealso: NEPSetJacobian(), NEPGetJacobian()
802: @*/
803: PetscErrorCode NEPComputeJacobian(NEP nep,PetscScalar lambda,Mat A)
804: {
805: PetscInt i;
806: PetscScalar alpha;
808: PetscFunctionBegin;
810: NEPCheckProblem(nep,1);
811: switch (nep->fui) {
812: case NEP_USER_INTERFACE_CALLBACK:
813: PetscCheck(nep->computejacobian,PetscObjectComm((PetscObject)nep),PETSC_ERR_USER,"Must call NEPSetJacobian() first");
814: PetscCall(PetscLogEventBegin(NEP_JacobianEval,nep,A,0,0));
815: PetscCallBack("NEP user Jacobian function",(*nep->computejacobian)(nep,lambda,A,nep->jacobianctx));
816: PetscCall(PetscLogEventEnd(NEP_JacobianEval,nep,A,0,0));
817: break;
818: case NEP_USER_INTERFACE_SPLIT:
819: PetscCall(MatZeroEntries(A));
820: for (i=0;i<nep->nt;i++) {
821: PetscCall(FNEvaluateDerivative(nep->f[i],lambda,&alpha));
822: PetscCall(MatAXPY(A,alpha,nep->A[i],nep->mstr));
823: }
824: break;
825: }
826: PetscFunctionReturn(PETSC_SUCCESS);
827: }