LCOV - code coverage report
Current view: top level - pep/tutorials - ex40.c (source / functions) Hit Total Coverage
Test: SLEPc Lines: 197 210 93.8 %
Date: 2025-01-22 00:40:06 Functions: 7 7 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*
       2             :    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
       3             :    SLEPc - Scalable Library for Eigenvalue Problem Computations
       4             :    Copyright (c) 2002-, Universitat Politecnica de Valencia, Spain
       5             : 
       6             :    This file is part of SLEPc.
       7             :    SLEPc is distributed under a 2-clause BSD license (see LICENSE).
       8             :    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
       9             : */
      10             : 
      11             : static char help[] = "Checking the definite property in quadratic symmetric eigenproblem.\n\n"
      12             :   "The command line options are:\n"
      13             :   "  -n <n> ... dimension of the matrices.\n"
      14             :   "  -transform... whether to transform to a hyperbolic problem or not.\n"
      15             :   "  -nonhyperbolic... to test with a modified (definite) problem that is not hyperbolic.\n\n";
      16             : 
      17             : #include <slepcpep.h>
      18             : 
      19             : /*
      20             :   This example is based on spring.c, for fixed values mu=1,tau=10,kappa=5
      21             : 
      22             :   The transformations are based on the method proposed in [Niendorf and Voss, LAA 2010].
      23             : */
      24             : 
      25             : PetscErrorCode QEPDefiniteTransformGetMatrices(PEP,PetscBool,PetscReal,PetscReal,Mat[3]);
      26             : PetscErrorCode QEPDefiniteTransformMap(PetscBool,PetscReal,PetscReal,PetscInt,PetscScalar*,PetscBool);
      27             : PetscErrorCode QEPDefiniteCheckError(Mat*,PEP,PetscBool,PetscReal,PetscReal);
      28             : PetscErrorCode TransformMatricesMoebius(Mat[3],MatStructure,PetscReal,PetscReal,PetscReal,PetscReal,Mat[3]);
      29             : 
      30           2 : int main(int argc,char **argv)
      31             : {
      32           2 :   Mat            M,C,K,*Op,A[3],At[3],B[3]; /* problem matrices */
      33           2 :   PEP            pep;        /* polynomial eigenproblem solver context */
      34           2 :   ST             st;         /* spectral transformation context */
      35           2 :   KSP            ksp;
      36           2 :   PC             pc;
      37           2 :   PEPProblemType type;
      38           2 :   PetscBool      terse,transform=PETSC_FALSE,nohyp=PETSC_FALSE;
      39           2 :   PetscInt       n=100,Istart,Iend,i,def=0,hyp;
      40           2 :   PetscReal      muu=1,tau=10,kappa=5,inta,intb;
      41           2 :   PetscReal      alpha,beta,xi,mu,at[2]={0.0,0.0},c=.857,s;
      42           2 :   PetscScalar    target,targett,ats[2];
      43             : 
      44           2 :   PetscFunctionBeginUser;
      45           2 :   PetscCall(SlepcInitialize(&argc,&argv,NULL,help));
      46             : 
      47           2 :   PetscCall(PetscOptionsGetInt(NULL,NULL,"-n",&n,NULL));
      48           2 :   PetscCall(PetscPrintf(PETSC_COMM_WORLD,"\nPEP example that checks definite property, n=%" PetscInt_FMT "\n\n",n));
      49             : 
      50             :   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
      51             :      Compute the matrices that define the eigensystem, (k^2*M+k*C+K)x=0
      52             :      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
      53             : 
      54             :   /* K is a tridiagonal */
      55           2 :   PetscCall(MatCreate(PETSC_COMM_WORLD,&K));
      56           2 :   PetscCall(MatSetSizes(K,PETSC_DECIDE,PETSC_DECIDE,n,n));
      57           2 :   PetscCall(MatSetFromOptions(K));
      58             : 
      59           2 :   PetscCall(MatGetOwnershipRange(K,&Istart,&Iend));
      60         202 :   for (i=Istart;i<Iend;i++) {
      61         200 :     if (i>0) PetscCall(MatSetValue(K,i,i-1,-kappa,INSERT_VALUES));
      62         200 :     PetscCall(MatSetValue(K,i,i,kappa*3.0,INSERT_VALUES));
      63         200 :     if (i<n-1) PetscCall(MatSetValue(K,i,i+1,-kappa,INSERT_VALUES));
      64             :   }
      65             : 
      66           2 :   PetscCall(MatAssemblyBegin(K,MAT_FINAL_ASSEMBLY));
      67           2 :   PetscCall(MatAssemblyEnd(K,MAT_FINAL_ASSEMBLY));
      68             : 
      69             :   /* C is a tridiagonal */
      70           2 :   PetscCall(MatCreate(PETSC_COMM_WORLD,&C));
      71           2 :   PetscCall(MatSetSizes(C,PETSC_DECIDE,PETSC_DECIDE,n,n));
      72           2 :   PetscCall(MatSetFromOptions(C));
      73             : 
      74           2 :   PetscCall(MatGetOwnershipRange(C,&Istart,&Iend));
      75         202 :   for (i=Istart;i<Iend;i++) {
      76         200 :     if (i>0) PetscCall(MatSetValue(C,i,i-1,-tau,INSERT_VALUES));
      77         200 :     PetscCall(MatSetValue(C,i,i,tau*3.0,INSERT_VALUES));
      78         200 :     if (i<n-1) PetscCall(MatSetValue(C,i,i+1,-tau,INSERT_VALUES));
      79             :   }
      80             : 
      81           2 :   PetscCall(MatAssemblyBegin(C,MAT_FINAL_ASSEMBLY));
      82           2 :   PetscCall(MatAssemblyEnd(C,MAT_FINAL_ASSEMBLY));
      83             : 
      84             :   /* M is a diagonal matrix */
      85           2 :   PetscCall(MatCreate(PETSC_COMM_WORLD,&M));
      86           2 :   PetscCall(MatSetSizes(M,PETSC_DECIDE,PETSC_DECIDE,n,n));
      87           2 :   PetscCall(MatSetFromOptions(M));
      88           2 :   PetscCall(MatGetOwnershipRange(M,&Istart,&Iend));
      89         202 :   for (i=Istart;i<Iend;i++) PetscCall(MatSetValue(M,i,i,muu,INSERT_VALUES));
      90           2 :   PetscCall(MatAssemblyBegin(M,MAT_FINAL_ASSEMBLY));
      91           2 :   PetscCall(MatAssemblyEnd(M,MAT_FINAL_ASSEMBLY));
      92             : 
      93           2 :   PetscCall(PetscOptionsGetBool(NULL,NULL,"-nonhyperbolic",&nohyp,NULL));
      94           2 :   A[0] = K; A[1] = C; A[2] = M;
      95           2 :   if (nohyp) {
      96           2 :     s = c*.6;
      97           2 :     PetscCall(TransformMatricesMoebius(A,UNKNOWN_NONZERO_PATTERN,c,s,-s,c,At));
      98           8 :     for (i=0;i<3;i++) PetscCall(MatDestroy(&A[i]));
      99             :     Op = At;
     100             :   } else Op = A;
     101             : 
     102             :   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
     103             :                 Create the eigensolver and solve the problem
     104             :      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     105             : 
     106             :   /*
     107             :      Create eigensolver context
     108             :   */
     109           2 :   PetscCall(PEPCreate(PETSC_COMM_WORLD,&pep));
     110           2 :   PetscCall(PEPSetProblemType(pep,PEP_HERMITIAN));
     111           2 :   PetscCall(PEPSetType(pep,PEPSTOAR));
     112             :   /*
     113             :      Set operators and set problem type
     114             :   */
     115           2 :   PetscCall(PEPSetOperators(pep,3,Op));
     116             : 
     117             :   /*
     118             :      Set shift-and-invert with Cholesky; select MUMPS if available
     119             :   */
     120           2 :   PetscCall(PEPGetST(pep,&st));
     121           2 :   PetscCall(STGetKSP(st,&ksp));
     122           2 :   PetscCall(KSPSetType(ksp,KSPPREONLY));
     123           2 :   PetscCall(KSPGetPC(ksp,&pc));
     124           2 :   PetscCall(PCSetType(pc,PCCHOLESKY));
     125             : 
     126             :   /*
     127             :      Use MUMPS if available.
     128             :      Note that in complex scalars we cannot use MUMPS for spectrum slicing,
     129             :      because MatGetInertia() is not available in that case.
     130             :   */
     131             : #if defined(PETSC_HAVE_MUMPS) && !defined(PETSC_USE_COMPLEX)
     132             :   PetscCall(PCFactorSetMatSolverType(pc,MATSOLVERMUMPS));
     133             :   /*
     134             :      Add several MUMPS options (see ex43.c for a better way of setting them in program):
     135             :      '-st_mat_mumps_icntl_13 1': turn off ScaLAPACK for matrix inertia
     136             :   */
     137             :   PetscCall(PetscOptionsInsertString(NULL,"-st_mat_mumps_icntl_13 1 -st_mat_mumps_icntl_24 1 -st_mat_mumps_cntl_3 1e-12"));
     138             : #endif
     139             : 
     140             :   /*
     141             :      Set solver parameters at runtime
     142             :   */
     143           2 :   PetscCall(PEPSetFromOptions(pep));
     144             : 
     145           2 :   PetscCall(PetscOptionsGetBool(NULL,NULL,"-transform",&transform,NULL));
     146           2 :   if (transform) {
     147             :     /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
     148             :                     Check if the problem is definite
     149             :        - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     150           1 :     PetscCall(PEPCheckDefiniteQEP(pep,&xi,&mu,&def,&hyp));
     151           1 :     switch (def) {
     152           1 :       case 1:
     153           1 :         if (hyp==1) PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Hyperbolic Problem xi=%g\n",(double)xi));
     154           1 :         else PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Definite Problem xi=%g mu=%g\n",(double)xi,(double)mu));
     155             :         break;
     156           0 :       case -1:
     157           0 :         PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Not Definite Problem\n"));
     158             :         break;
     159           0 :       default:
     160           0 :         PetscCall(PetscPrintf(PETSC_COMM_WORLD,"Cannot determine definiteness\n"));
     161             :         break;
     162             :     }
     163             : 
     164             :     /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
     165             :       Transform the QEP to have a definite inner product in the linearization
     166             :        - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     167           1 :     if (def==1) {
     168           1 :       PetscCall(QEPDefiniteTransformGetMatrices(pep,hyp==1?PETSC_TRUE:PETSC_FALSE,xi,mu,B));
     169           1 :       PetscCall(PEPSetOperators(pep,3,B));
     170           1 :       PetscCall(PEPGetTarget(pep,&target));
     171           1 :       targett = target;
     172           1 :       PetscCall(QEPDefiniteTransformMap(hyp==1?PETSC_TRUE:PETSC_FALSE,xi,mu,1,&targett,PETSC_FALSE));
     173           1 :       PetscCall(PEPSetTarget(pep,targett));
     174           1 :       PetscCall(PEPGetProblemType(pep,&type));
     175           1 :       PetscCall(PEPSetProblemType(pep,PEP_HYPERBOLIC));
     176           1 :       PetscCall(PEPSTOARGetLinearization(pep,&alpha,&beta));
     177           1 :       PetscCall(PEPSTOARSetLinearization(pep,1.0,0.0));
     178           1 :       PetscCall(PEPGetInterval(pep,&inta,&intb));
     179           1 :       if (inta!=intb) {
     180           0 :         ats[0] = inta; ats[1] = intb;
     181           0 :         PetscCall(QEPDefiniteTransformMap(hyp==1?PETSC_TRUE:PETSC_FALSE,xi,mu,2,ats,PETSC_FALSE));
     182           0 :         at[0] = PetscRealPart(ats[0]); at[1] = PetscRealPart(ats[1]);
     183           0 :         if (at[0]<at[1]) PetscCall(PEPSetInterval(pep,at[0],at[1]));
     184           0 :         else PetscCall(PEPSetInterval(pep,PETSC_MIN_REAL,at[1]));
     185             :       }
     186             :     }
     187             :   }
     188             : 
     189             :   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
     190             :                       Solve the eigensystem
     191             :      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     192           2 :   PetscCall(PEPSolve(pep));
     193             : 
     194             :   /* show detailed info unless -terse option is given by user */
     195           2 :   if (def!=1) {
     196           1 :     PetscCall(PetscOptionsHasName(NULL,NULL,"-terse",&terse));
     197           1 :     if (terse) PetscCall(PEPErrorView(pep,PEP_ERROR_BACKWARD,NULL));
     198             :     else {
     199           1 :       PetscCall(PetscViewerPushFormat(PETSC_VIEWER_STDOUT_WORLD,PETSC_VIEWER_ASCII_INFO_DETAIL));
     200           1 :       PetscCall(PEPConvergedReasonView(pep,PETSC_VIEWER_STDOUT_WORLD));
     201           1 :       PetscCall(PEPErrorView(pep,PEP_ERROR_RELATIVE,PETSC_VIEWER_STDOUT_WORLD));
     202           1 :       PetscCall(PetscViewerPopFormat(PETSC_VIEWER_STDOUT_WORLD));
     203             :     }
     204             :   } else {
     205             :     /* Map the solution */
     206           1 :     PetscCall(PEPConvergedReasonView(pep,PETSC_VIEWER_STDOUT_WORLD));
     207           1 :     PetscCall(QEPDefiniteCheckError(Op,pep,hyp==1?PETSC_TRUE:PETSC_FALSE,xi,mu));
     208           4 :     for (i=0;i<3;i++) PetscCall(MatDestroy(B+i));
     209             :   }
     210           2 :   if (at[0]>at[1]) {
     211           0 :     PetscCall(PEPSetInterval(pep,at[0],PETSC_MAX_REAL));
     212           0 :     PetscCall(PEPSolve(pep));
     213           0 :     PetscCall(PEPConvergedReasonView(pep,PETSC_VIEWER_STDOUT_WORLD));
     214             :     /* Map the solution */
     215           0 :     PetscCall(QEPDefiniteCheckError(Op,pep,hyp==1?PETSC_TRUE:PETSC_FALSE,xi,mu));
     216             :   }
     217           2 :   if (def==1) {
     218           1 :     PetscCall(PEPSetTarget(pep,target));
     219           1 :     PetscCall(PEPSetProblemType(pep,type));
     220           1 :     PetscCall(PEPSTOARSetLinearization(pep,alpha,beta));
     221           1 :     if (inta!=intb) PetscCall(PEPSetInterval(pep,inta,intb));
     222             :   }
     223             : 
     224             :   /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
     225             :                     Clean up
     226             :      - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
     227           2 :   PetscCall(PEPDestroy(&pep));
     228           8 :   for (i=0;i<3;i++) PetscCall(MatDestroy(Op+i));
     229           2 :   PetscCall(SlepcFinalize());
     230             :   return 0;
     231             : }
     232             : 
     233             : /* ------------------------------------------------------------------- */
     234             : /*
     235             :   QEPDefiniteTransformMap_Initial - map a scalar value with a certain Moebius transform
     236             : 
     237             :                    a theta + b
     238             :          lambda = --------------
     239             :                    c theta + d
     240             : 
     241             :   Input:
     242             :     xi,mu: real values such that Q(xi)<0 and Q(mu)>0
     243             :     hyperbolic: if true the problem is assumed hyperbolic (mu is not used)
     244             :   Input/Output:
     245             :     val (array of length n)
     246             :     if backtransform=true returns lambda from theta, else returns theta from lambda
     247             : */
     248          13 : static PetscErrorCode QEPDefiniteTransformMap_Initial(PetscBool hyperbolic,PetscReal xi,PetscReal mu,PetscInt n,PetscScalar *val,PetscBool backtransform)
     249             : {
     250          13 :   PetscInt  i;
     251          13 :   PetscReal a,b,c,d,s;
     252             : 
     253          13 :   PetscFunctionBegin;
     254          13 :   if (hyperbolic) { a = 1.0; b = xi; c =0.0; d = 1.0; }
     255          13 :   else { a = mu; b = mu*xi-1; c = 1.0; d = xi+mu; }
     256          13 :   if (!backtransform) { s = a; a = -d; d = -s; }
     257          26 :   for (i=0;i<n;i++) {
     258          13 :     if (PetscRealPart(val[i]) >= PETSC_MAX_REAL || PetscRealPart(val[i]) <= PETSC_MIN_REAL) val[i] = a/c;
     259          13 :     else if (val[i] == -d/c) val[i] = PETSC_MAX_REAL;
     260          13 :     else val[i] = (a*val[i]+b)/(c*val[i]+d);
     261             :   }
     262          13 :   PetscFunctionReturn(PETSC_SUCCESS);
     263             : }
     264             : 
     265             : /* ------------------------------------------------------------------- */
     266             : /*
     267             :   QEPDefiniteTransformMap - perform the mapping if the problem is hyperbolic, otherwise
     268             :   modify the value of xi in advance
     269             : */
     270           6 : PetscErrorCode QEPDefiniteTransformMap(PetscBool hyperbolic,PetscReal xi,PetscReal mu,PetscInt n,PetscScalar *val,PetscBool backtransform)
     271             : {
     272           6 :   PetscReal      xit;
     273           6 :   PetscScalar    alpha;
     274             : 
     275           6 :   PetscFunctionBegin;
     276           6 :   xit = xi;
     277           6 :   if (!hyperbolic) {
     278           6 :     alpha = xi;
     279           6 :     PetscCall(QEPDefiniteTransformMap_Initial(PETSC_FALSE,0.0,mu,1,&alpha,PETSC_FALSE));
     280           6 :     xit = PetscRealPart(alpha);
     281             :   }
     282           6 :   PetscCall(QEPDefiniteTransformMap_Initial(hyperbolic,xit,mu,n,val,backtransform));
     283           6 :   PetscFunctionReturn(PETSC_SUCCESS);
     284             : }
     285             : 
     286             : /* ------------------------------------------------------------------- */
     287             : /*
     288             :   TransformMatricesMoebius - transform the coefficient matrices of a QEP
     289             : 
     290             :   Input:
     291             :     A: coefficient matrices of the original QEP
     292             :     a,b,c,d: parameters of the Moebius transform
     293             :     str: structure flag for MatAXPY operations
     294             :   Output:
     295             :     B: transformed matrices
     296             : */
     297           3 : PetscErrorCode TransformMatricesMoebius(Mat A[3],MatStructure str,PetscReal a,PetscReal b,PetscReal c,PetscReal d,Mat B[3])
     298             : {
     299           3 :   PetscInt       i,k;
     300           3 :   PetscReal      cf[9];
     301             : 
     302           3 :   PetscFunctionBegin;
     303          12 :   for (i=0;i<3;i++) PetscCall(MatDuplicate(A[2],MAT_COPY_VALUES,&B[i]));
     304             :   /* Ct = b*b*A+b*d*B+d*d*C */
     305           3 :   cf[0] = d*d; cf[1] = b*d; cf[2] = b*b;
     306             :   /* Bt = 2*a*b*A+(b*c+a*d)*B+2*c*d*C*/
     307           3 :   cf[3] = 2*c*d; cf[4] = b*c+a*d; cf[5] = 2*a*b;
     308             :   /* At = a*a*A+a*c*B+c*c*C */
     309           3 :   cf[6] = c*c; cf[7] = a*c; cf[8] = a*a;
     310          12 :   for (k=0;k<3;k++) {
     311           9 :     PetscCall(MatScale(B[k],cf[k*3+2]));
     312          27 :     for (i=0;i<2;i++) PetscCall(MatAXPY(B[k],cf[3*k+i],A[i],str));
     313             :   }
     314           3 :   PetscFunctionReturn(PETSC_SUCCESS);
     315             : }
     316             : 
     317             : /* ------------------------------------------------------------------- */
     318             : /*
     319             :   QEPDefiniteTransformGetMatrices - given a PEP of degree 2, transform the three
     320             :   matrices with TransformMatricesMoebius
     321             : 
     322             :   Input:
     323             :     pep: polynomial eigenproblem to be transformed, with Q(.) being the quadratic polynomial
     324             :     xi,mu: real values such that Q(xi)<0 and Q(mu)>0
     325             :     hyperbolic: if true the problem is assumed hyperbolic (mu is not used)
     326             :   Output:
     327             :     T: coefficient matrices of the transformed polynomial
     328             : */
     329           1 : PetscErrorCode QEPDefiniteTransformGetMatrices(PEP pep,PetscBool hyperbolic,PetscReal xi,PetscReal mu,Mat T[3])
     330             : {
     331           1 :   MatStructure   str;
     332           1 :   ST             st;
     333           1 :   PetscInt       i;
     334           1 :   PetscReal      a,b,c,d;
     335           1 :   PetscScalar    xit;
     336           1 :   Mat            A[3];
     337             : 
     338           1 :   PetscFunctionBegin;
     339           4 :   for (i=2;i>=0;i--) PetscCall(PEPGetOperators(pep,i,&A[i]));
     340           1 :   if (hyperbolic) { a = 1.0; b = xi; c =0.0; d = 1.0; }
     341             :   else {
     342           1 :     xit = xi;
     343           1 :     PetscCall(QEPDefiniteTransformMap_Initial(PETSC_FALSE,0.0,mu,1,&xit,PETSC_FALSE));
     344           1 :     a = mu; b = mu*PetscRealPart(xit)-1.0; c = 1.0; d = PetscRealPart(xit)+mu;
     345             :   }
     346           1 :   PetscCall(PEPGetST(pep,&st));
     347           1 :   PetscCall(STGetMatStructure(st,&str));
     348           1 :   PetscCall(TransformMatricesMoebius(A,str,a,b,c,d,T));
     349           1 :   PetscFunctionReturn(PETSC_SUCCESS);
     350             : }
     351             : 
     352             : /* ------------------------------------------------------------------- */
     353             : /*
     354             :   Auxiliary function to compute the residual norm of an eigenpair of a QEP defined
     355             :   by coefficient matrices A
     356             : */
     357           5 : static PetscErrorCode PEPResidualNorm(Mat *A,PetscScalar kr,PetscScalar ki,Vec xr,Vec xi,Vec *z,PetscReal *norm)
     358             : {
     359           5 :   PetscInt       i,nmat=3;
     360           5 :   PetscScalar    vals[3];
     361           5 :   Vec            u,w;
     362             : #if !defined(PETSC_USE_COMPLEX)
     363             :   Vec            ui,wi;
     364             :   PetscReal      ni;
     365             :   PetscBool      imag;
     366             :   PetscScalar    ivals[3];
     367             : #endif
     368             : 
     369           5 :   PetscFunctionBegin;
     370           5 :   u = z[0]; w = z[1];
     371           5 :   PetscCall(VecSet(u,0.0));
     372             : #if !defined(PETSC_USE_COMPLEX)
     373             :   ui = z[2]; wi = z[3];
     374             : #endif
     375           5 :   vals[0] = 1.0;
     376           5 :   vals[1] = kr;
     377           5 :   vals[2] = kr*kr-ki*ki;
     378             : #if !defined(PETSC_USE_COMPLEX)
     379             :   ivals[0] = 0.0;
     380             :   ivals[1] = ki;
     381             :   ivals[2] = 2.0*kr*ki;
     382             :   if (ki == 0 || PetscAbsScalar(ki) < PetscAbsScalar(kr*PETSC_MACHINE_EPSILON))
     383             :     imag = PETSC_FALSE;
     384             :   else {
     385             :     imag = PETSC_TRUE;
     386             :     PetscCall(VecSet(ui,0.0));
     387             :   }
     388             : #endif
     389          20 :   for (i=0;i<nmat;i++) {
     390          15 :     if (vals[i]!=0.0) {
     391          15 :       PetscCall(MatMult(A[i],xr,w));
     392          15 :       PetscCall(VecAXPY(u,vals[i],w));
     393             :     }
     394             : #if !defined(PETSC_USE_COMPLEX)
     395             :     if (imag) {
     396             :       if (ivals[i]!=0 || vals[i]!=0) {
     397             :         PetscCall(MatMult(A[i],xi,wi));
     398             :         if (vals[i]==0) PetscCall(MatMult(A[i],xr,w));
     399             :       }
     400             :       if (ivals[i]!=0) {
     401             :         PetscCall(VecAXPY(u,-ivals[i],wi));
     402             :         PetscCall(VecAXPY(ui,ivals[i],w));
     403             :       }
     404             :       if (vals[i]!=0) PetscCall(VecAXPY(ui,vals[i],wi));
     405             :     }
     406             : #endif
     407             :   }
     408           5 :   PetscCall(VecNorm(u,NORM_2,norm));
     409             : #if !defined(PETSC_USE_COMPLEX)
     410             :   if (imag) {
     411             :     PetscCall(VecNorm(ui,NORM_2,&ni));
     412             :     *norm = SlepcAbsEigenvalue(*norm,ni);
     413             :   }
     414             : #endif
     415           5 :   PetscFunctionReturn(PETSC_SUCCESS);
     416             : }
     417             : 
     418             : /* ------------------------------------------------------------------- */
     419             : /*
     420             :   QEPDefiniteCheckError - check and print the residual norm of a transformed PEP
     421             : 
     422             :   Input:
     423             :     A: coefficient matrices of the original problem
     424             :     pep: solver containing the computed solution of the transformed problem
     425             :     xi,mu,hyperbolic: parameters used in transformation
     426             : */
     427           1 : PetscErrorCode QEPDefiniteCheckError(Mat *A,PEP pep,PetscBool hyperbolic,PetscReal xi,PetscReal mu)
     428             : {
     429           1 :   PetscScalar    er,ei;
     430           1 :   PetscReal      re,im,error;
     431           1 :   Vec            vr,vi,w[4];
     432           1 :   PetscInt       i,nconv;
     433           1 :   BV             bv;
     434           1 :   char           ex[30],sep[]=" ---------------------- --------------------\n";
     435             : 
     436           1 :   PetscFunctionBegin;
     437           1 :   PetscCall(PetscSNPrintf(ex,sizeof(ex),"||P(k)x||/||kx||"));
     438           1 :   PetscCall(PetscPrintf(PETSC_COMM_WORLD,"%s            k             %s\n%s",sep,ex,sep));
     439           1 :   PetscCall(PEPGetConverged(pep,&nconv));
     440           1 :   PetscCall(PEPGetBV(pep,&bv));
     441           1 :   PetscCall(BVCreateVec(bv,w));
     442           1 :   PetscCall(VecDuplicate(w[0],&vr));
     443           1 :   PetscCall(VecDuplicate(w[0],&vi));
     444           4 :   for (i=1;i<4;i++) PetscCall(VecDuplicate(w[0],w+i));
     445           6 :   for (i=0;i<nconv;i++) {
     446           5 :     PetscCall(PEPGetEigenpair(pep,i,&er,&ei,vr,vi));
     447           5 :     PetscCall(QEPDefiniteTransformMap(hyperbolic,xi,mu,1,&er,PETSC_TRUE));
     448           5 :     PetscCall(PEPResidualNorm(A,er,0.0,vr,vi,w,&error));
     449           5 :     error /= SlepcAbsEigenvalue(er,0.0);
     450             : #if defined(PETSC_USE_COMPLEX)
     451           5 :     re = PetscRealPart(er);
     452           5 :     im = PetscImaginaryPart(ei);
     453             : #else
     454             :     re = er;
     455             :     im = ei;
     456             : #endif
     457           5 :     if (im!=0.0) PetscCall(PetscPrintf(PETSC_COMM_WORLD,"  % 9f%+9fi      %12g\n",(double)re,(double)im,(double)error));
     458           5 :     else PetscCall(PetscPrintf(PETSC_COMM_WORLD,"    % 12f           %12g\n",(double)re,(double)error));
     459             :   }
     460           1 :   PetscCall(PetscPrintf(PETSC_COMM_WORLD,"%s",sep));
     461           5 :   for (i=0;i<4;i++) PetscCall(VecDestroy(w+i));
     462           1 :   PetscCall(VecDestroy(&vi));
     463           1 :   PetscCall(VecDestroy(&vr));
     464           1 :   PetscFunctionReturn(PETSC_SUCCESS);
     465             : }
     466             : 
     467             : /*TEST
     468             : 
     469             :    testset:
     470             :       requires: !single
     471             :       args: -pep_nev 3 -nonhyperbolic -pep_target 2
     472             :       output_file: output/ex40_1.out
     473             :       filter: grep -v "Definite" | sed -e "s/iterations [0-9]\([0-9]*\)/iterations xx/g" | sed -e "s/[0-9]\.[0-9]*e[+-]\([0-9]*\)/removed/g"
     474             :       test:
     475             :          suffix: 1
     476             :          requires: !complex
     477             :       test:
     478             :          suffix: 1_complex
     479             :          requires: complex !mumps
     480             :       test:
     481             :          suffix: 1_transform
     482             :          requires: !complex
     483             :          args: -transform
     484             :       test:
     485             :          suffix: 1_transform_complex
     486             :          requires: complex !mumps
     487             :          args: -transform
     488             : 
     489             : TEST*/

Generated by: LCOV version 1.14