LCOV - code coverage report
Current view: top level - sys/classes/bv/interface - bvglobal.c (source / functions) Hit Total Coverage
Test: SLEPc Lines: 558 566 98.6 %
Date: 2025-01-22 00:40:06 Functions: 24 24 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             :    BV operations involving global communication
      12             : */
      13             : 
      14             : #include <slepc/private/bvimpl.h>      /*I "slepcbv.h" I*/
      15             : 
      16             : /*
      17             :   BVDot for the particular case of non-standard inner product with
      18             :   matrix B, which is assumed to be symmetric (or complex Hermitian)
      19             : */
      20          68 : static inline PetscErrorCode BVDot_Private(BV X,BV Y,Mat M)
      21             : {
      22          68 :   PetscObjectId  idx,idy;
      23          68 :   PetscInt       i,j,jend,m;
      24          68 :   PetscScalar    *marray;
      25          68 :   PetscBool      symm=PETSC_FALSE;
      26          68 :   Mat            B;
      27          68 :   Vec            z;
      28             : 
      29          68 :   PetscFunctionBegin;
      30          68 :   BVCheckOp(Y,1,dotvec);
      31          68 :   PetscCall(MatGetSize(M,&m,NULL));
      32          68 :   PetscCall(MatDenseGetArray(M,&marray));
      33          68 :   PetscCall(PetscObjectGetId((PetscObject)X,&idx));
      34          68 :   PetscCall(PetscObjectGetId((PetscObject)Y,&idy));
      35          68 :   B = Y->matrix;
      36          68 :   Y->matrix = NULL;
      37          68 :   if (idx==idy) symm=PETSC_TRUE;  /* M=X'BX is symmetric */
      38          68 :   jend = X->k;
      39         412 :   for (j=X->l;j<jend;j++) {
      40         344 :     if (symm) Y->k = j+1;
      41         344 :     PetscCall(BVGetColumn(X->cached,j,&z));
      42         344 :     PetscUseTypeMethod(Y,dotvec,z,marray+j*m+Y->l);
      43         344 :     PetscCall(BVRestoreColumn(X->cached,j,&z));
      44         344 :     if (symm) {
      45        1032 :       for (i=X->l;i<j;i++)
      46         736 :         marray[j+i*m] = PetscConj(marray[i+j*m]);
      47             :     }
      48             :   }
      49          68 :   PetscCall(MatDenseRestoreArray(M,&marray));
      50          68 :   Y->matrix = B;
      51          68 :   PetscFunctionReturn(PETSC_SUCCESS);
      52             : }
      53             : 
      54             : /*@
      55             :    BVDot - Computes the 'block-dot' product of two basis vectors objects.
      56             : 
      57             :    Collective
      58             : 
      59             :    Input Parameters:
      60             : +  X - first basis vectors
      61             : -  Y - second basis vectors
      62             : 
      63             :    Output Parameter:
      64             : .  M - the resulting matrix
      65             : 
      66             :    Notes:
      67             :    This is the generalization of VecDot() for a collection of vectors, M = Y^H*X.
      68             :    The result is a matrix M whose entry m_ij is equal to y_i^H x_j (where y_i^H
      69             :    denotes the conjugate transpose of y_i).
      70             : 
      71             :    If a non-standard inner product has been specified with BVSetMatrix(),
      72             :    then the result is M = Y^H*B*X. In this case, both X and Y must have
      73             :    the same associated matrix.
      74             : 
      75             :    On entry, M must be a sequential dense Mat with dimensions m,n at least, where
      76             :    m is the number of active columns of Y and n is the number of active columns of X.
      77             :    Only rows (resp. columns) of M starting from ly (resp. lx) are computed,
      78             :    where ly (resp. lx) is the number of leading columns of Y (resp. X).
      79             : 
      80             :    X and Y need not be different objects.
      81             : 
      82             :    Level: intermediate
      83             : 
      84             : .seealso: BVDotVec(), BVDotColumn(), BVSetActiveColumns(), BVSetMatrix()
      85             : @*/
      86       30408 : PetscErrorCode BVDot(BV X,BV Y,Mat M)
      87             : {
      88       30408 :   PetscInt       m,n;
      89             : 
      90       30408 :   PetscFunctionBegin;
      91       30408 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
      92       30408 :   PetscValidHeaderSpecific(Y,BV_CLASSID,2);
      93       30408 :   PetscValidHeaderSpecific(M,MAT_CLASSID,3);
      94       30408 :   PetscValidType(X,1);
      95       30408 :   BVCheckSizes(X,1);
      96       30408 :   PetscValidType(Y,2);
      97       30408 :   BVCheckSizes(Y,2);
      98       30408 :   PetscValidType(M,3);
      99       30408 :   PetscCheckSameTypeAndComm(X,1,Y,2);
     100       30408 :   SlepcMatCheckSeq(M);
     101             : 
     102       30408 :   PetscCall(MatGetSize(M,&m,&n));
     103       30408 :   PetscCheck(m>=Y->k,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_SIZ,"Mat argument has %" PetscInt_FMT " rows, should have at least %" PetscInt_FMT,m,Y->k);
     104       30408 :   PetscCheck(n>=X->k,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_SIZ,"Mat argument has %" PetscInt_FMT " columns, should have at least %" PetscInt_FMT,n,X->k);
     105       30408 :   PetscCheck(X->n==Y->n,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_INCOMP,"Mismatching local dimension X %" PetscInt_FMT ", Y %" PetscInt_FMT,X->n,Y->n);
     106       30408 :   PetscCheck(X->matrix==Y->matrix,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"X and Y must have the same inner product matrix");
     107       30408 :   if (X->l==X->k || Y->l==Y->k) PetscFunctionReturn(PETSC_SUCCESS);
     108             : 
     109       30310 :   PetscCall(PetscLogEventBegin(BV_Dot,X,Y,0,0));
     110       30310 :   if (X->matrix) { /* non-standard inner product */
     111             :     /* compute BX first */
     112         303 :     PetscCall(BV_IPMatMultBV(X));
     113         303 :     if (X->vmm==BV_MATMULT_VECS) {
     114             :       /* perform computation column by column */
     115          68 :       PetscCall(BVDot_Private(X,Y,M));
     116         235 :     } else PetscUseTypeMethod(X->cached,dot,Y,M);
     117       30007 :   } else PetscUseTypeMethod(X,dot,Y,M);
     118       30310 :   PetscCall(PetscLogEventEnd(BV_Dot,X,Y,0,0));
     119       30310 :   PetscFunctionReturn(PETSC_SUCCESS);
     120             : }
     121             : 
     122             : /*@
     123             :    BVDotVec - Computes multiple dot products of a vector against all the
     124             :    column vectors of a BV.
     125             : 
     126             :    Collective
     127             : 
     128             :    Input Parameters:
     129             : +  X - basis vectors
     130             : -  y - a vector
     131             : 
     132             :    Output Parameter:
     133             : .  m - an array where the result must be placed
     134             : 
     135             :    Notes:
     136             :    This is analogue to VecMDot(), but using BV to represent a collection
     137             :    of vectors. The result is m = X^H*y, so m_i is equal to x_i^H y. Note
     138             :    that here X is transposed as opposed to BVDot().
     139             : 
     140             :    If a non-standard inner product has been specified with BVSetMatrix(),
     141             :    then the result is m = X^H*B*y.
     142             : 
     143             :    The length of array m must be equal to the number of active columns of X
     144             :    minus the number of leading columns, i.e. the first entry of m is the
     145             :    product of the first non-leading column with y.
     146             : 
     147             :    Level: intermediate
     148             : 
     149             : .seealso: BVDot(), BVDotColumn(), BVSetActiveColumns(), BVSetMatrix()
     150             : @*/
     151      166784 : PetscErrorCode BVDotVec(BV X,Vec y,PetscScalar m[])
     152             : {
     153      166784 :   PetscInt       n;
     154             : 
     155      166784 :   PetscFunctionBegin;
     156      166784 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
     157      166784 :   PetscValidHeaderSpecific(y,VEC_CLASSID,2);
     158      166784 :   PetscValidType(X,1);
     159      166784 :   BVCheckSizes(X,1);
     160      166784 :   BVCheckOp(X,1,dotvec);
     161      166784 :   PetscValidType(y,2);
     162      166784 :   PetscCheckSameTypeAndComm(X,1,y,2);
     163             : 
     164      166784 :   PetscCall(VecGetLocalSize(y,&n));
     165      166784 :   PetscCheck(X->n==n,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_INCOMP,"Mismatching local dimension X %" PetscInt_FMT ", y %" PetscInt_FMT,X->n,n);
     166             : 
     167      166784 :   PetscCall(PetscLogEventBegin(BV_DotVec,X,y,0,0));
     168      166784 :   PetscUseTypeMethod(X,dotvec,y,m);
     169      166784 :   PetscCall(PetscLogEventEnd(BV_DotVec,X,y,0,0));
     170      166784 :   PetscFunctionReturn(PETSC_SUCCESS);
     171             : }
     172             : 
     173             : /*@
     174             :    BVDotVecBegin - Starts a split phase dot product computation.
     175             : 
     176             :    Input Parameters:
     177             : +  X - basis vectors
     178             : .  y - a vector
     179             : -  m - an array where the result will go (can be NULL)
     180             : 
     181             :    Note:
     182             :    Each call to BVDotVecBegin() should be paired with a call to BVDotVecEnd().
     183             : 
     184             :    Level: advanced
     185             : 
     186             : .seealso: BVDotVecEnd(), BVDotVec()
     187             : @*/
     188        8601 : PetscErrorCode BVDotVecBegin(BV X,Vec y,PetscScalar *m)
     189             : {
     190        8601 :   PetscInt            i,n,nv;
     191        8601 :   PetscSplitReduction *sr;
     192        8601 :   MPI_Comm            comm;
     193             : 
     194        8601 :   PetscFunctionBegin;
     195        8601 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
     196        8601 :   PetscValidHeaderSpecific(y,VEC_CLASSID,2);
     197        8601 :   PetscValidType(X,1);
     198        8601 :   BVCheckSizes(X,1);
     199        8601 :   PetscValidType(y,2);
     200        8601 :   PetscCheckSameTypeAndComm(X,1,y,2);
     201             : 
     202        8601 :   PetscCall(VecGetLocalSize(y,&n));
     203        8601 :   PetscCheck(X->n==n,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_INCOMP,"Mismatching local dimension X %" PetscInt_FMT ", y %" PetscInt_FMT,X->n,n);
     204             : 
     205        8601 :   if (X->ops->dotvec_begin) PetscUseTypeMethod(X,dotvec_begin,y,m);
     206             :   else {
     207        8599 :     BVCheckOp(X,1,dotvec_local);
     208        8599 :     nv = X->k-X->l;
     209        8599 :     PetscCall(PetscObjectGetComm((PetscObject)X,&comm));
     210        8599 :     PetscCall(PetscSplitReductionGet(comm,&sr));
     211        8599 :     PetscCheck(sr->state==STATE_BEGIN,PetscObjectComm((PetscObject)X),PETSC_ERR_ORDER,"Called before all VecxxxEnd() called");
     212      114664 :     for (i=0;i<nv;i++) {
     213      106065 :       if (sr->numopsbegin+i >= sr->maxops) PetscCall(PetscSplitReductionExtend(sr));
     214      106065 :       sr->reducetype[sr->numopsbegin+i] = PETSC_SR_REDUCE_SUM;
     215      106065 :       sr->invecs[sr->numopsbegin+i]     = (void*)X;
     216             :     }
     217        8599 :     PetscCall(PetscLogEventBegin(BV_DotVec,X,y,0,0));
     218        8599 :     PetscUseTypeMethod(X,dotvec_local,y,sr->lvalues+sr->numopsbegin);
     219        8599 :     sr->numopsbegin += nv;
     220        8599 :     PetscCall(PetscLogEventEnd(BV_DotVec,X,y,0,0));
     221             :   }
     222        8601 :   PetscFunctionReturn(PETSC_SUCCESS);
     223             : }
     224             : 
     225             : /*@
     226             :    BVDotVecEnd - Ends a split phase dot product computation.
     227             : 
     228             :    Input Parameters:
     229             : +  X - basis vectors
     230             : .  y - a vector
     231             : -  m - an array where the result will go
     232             : 
     233             :    Note:
     234             :    Each call to BVDotVecBegin() should be paired with a call to BVDotVecEnd().
     235             : 
     236             :    Level: advanced
     237             : 
     238             : .seealso: BVDotVecBegin(), BVDotVec()
     239             : @*/
     240        8601 : PetscErrorCode BVDotVecEnd(BV X,Vec y,PetscScalar *m)
     241             : {
     242        8601 :   PetscInt            i,nv;
     243        8601 :   PetscSplitReduction *sr;
     244        8601 :   MPI_Comm            comm;
     245             : 
     246        8601 :   PetscFunctionBegin;
     247        8601 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
     248        8601 :   PetscValidType(X,1);
     249        8601 :   BVCheckSizes(X,1);
     250             : 
     251        8601 :   if (X->ops->dotvec_end) PetscUseTypeMethod(X,dotvec_end,y,m);
     252             :   else {
     253        8599 :     nv = X->k-X->l;
     254        8599 :     PetscCall(PetscObjectGetComm((PetscObject)X,&comm));
     255        8599 :     PetscCall(PetscSplitReductionGet(comm,&sr));
     256        8599 :     PetscCall(PetscSplitReductionEnd(sr));
     257             : 
     258        8599 :     PetscCheck(sr->numopsend<sr->numopsbegin,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"Called VecxxxEnd() more times than VecxxxBegin()");
     259        8599 :     PetscCheck((void*)X==sr->invecs[sr->numopsend],PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"Called BVxxxEnd() in a different order or with a different BV than BVxxxBegin()");
     260        8599 :     PetscCheck(sr->reducetype[sr->numopsend]==PETSC_SR_REDUCE_SUM,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"Wrong type of reduction");
     261      114664 :     for (i=0;i<nv;i++) m[i] = sr->gvalues[sr->numopsend++];
     262             : 
     263             :     /* Finished getting all the results so reset to no outstanding requests */
     264        8599 :     if (sr->numopsend == sr->numopsbegin) {
     265        4352 :       sr->state       = STATE_BEGIN;
     266        4352 :       sr->numopsend   = 0;
     267        4352 :       sr->numopsbegin = 0;
     268             :     }
     269             :   }
     270        8601 :   PetscFunctionReturn(PETSC_SUCCESS);
     271             : }
     272             : 
     273             : /*@
     274             :    BVDotColumn - Computes multiple dot products of a column against all the
     275             :    previous columns of a BV.
     276             : 
     277             :    Collective
     278             : 
     279             :    Input Parameters:
     280             : +  X - basis vectors
     281             : -  j - the column index
     282             : 
     283             :    Output Parameter:
     284             : .  q - an array where the result must be placed
     285             : 
     286             :    Notes:
     287             :    This operation is equivalent to BVDotVec() but it uses column j of X
     288             :    rather than taking a Vec as an argument. The number of active columns of
     289             :    X is set to j before the computation, and restored afterwards.
     290             :    If X has leading columns specified, then these columns do not participate
     291             :    in the computation. Therefore, the length of array q must be equal to j
     292             :    minus the number of leading columns.
     293             : 
     294             :    Developer Notes:
     295             :    If q is NULL, then the result is written in position nc+l of the internal
     296             :    buffer vector, see BVGetBufferVec().
     297             : 
     298             :    Level: advanced
     299             : 
     300             : .seealso: BVDot(), BVDotVec(), BVSetActiveColumns(), BVSetMatrix()
     301             : @*/
     302        3183 : PetscErrorCode BVDotColumn(BV X,PetscInt j,PetscScalar *q)
     303             : {
     304        3183 :   PetscInt       ksave;
     305        3183 :   Vec            y;
     306             : 
     307        3183 :   PetscFunctionBegin;
     308        3183 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
     309        9549 :   PetscValidLogicalCollectiveInt(X,j,2);
     310        3183 :   PetscValidType(X,1);
     311        3183 :   BVCheckSizes(X,1);
     312        3183 :   BVCheckOp(X,1,dotvec);
     313             : 
     314        3183 :   PetscCheck(j>=0,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_OUTOFRANGE,"Index j must be non-negative");
     315        3183 :   PetscCheck(j<X->m,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_OUTOFRANGE,"Index j=%" PetscInt_FMT " but BV only has %" PetscInt_FMT " columns",j,X->m);
     316             : 
     317        3183 :   PetscCall(PetscLogEventBegin(BV_DotVec,X,0,0,0));
     318        3183 :   ksave = X->k;
     319        3183 :   X->k = j;
     320        3183 :   if (!q && !X->buffer) PetscCall(BVGetBufferVec(X,&X->buffer));
     321        3183 :   PetscCall(BVGetColumn(X,j,&y));
     322        3183 :   PetscUseTypeMethod(X,dotvec,y,q);
     323        3183 :   PetscCall(BVRestoreColumn(X,j,&y));
     324        3183 :   X->k = ksave;
     325        3183 :   PetscCall(PetscLogEventEnd(BV_DotVec,X,0,0,0));
     326        3183 :   PetscFunctionReturn(PETSC_SUCCESS);
     327             : }
     328             : 
     329             : /*@
     330             :    BVDotColumnBegin - Starts a split phase dot product computation.
     331             : 
     332             :    Input Parameters:
     333             : +  X - basis vectors
     334             : .  j - the column index
     335             : -  m - an array where the result will go (can be NULL)
     336             : 
     337             :    Note:
     338             :    Each call to BVDotColumnBegin() should be paired with a call to BVDotColumnEnd().
     339             : 
     340             :    Level: advanced
     341             : 
     342             : .seealso: BVDotColumnEnd(), BVDotColumn()
     343             : @*/
     344        2251 : PetscErrorCode BVDotColumnBegin(BV X,PetscInt j,PetscScalar *m)
     345             : {
     346        2251 :   PetscInt            i,nv,ksave;
     347        2251 :   PetscSplitReduction *sr;
     348        2251 :   MPI_Comm            comm;
     349        2251 :   Vec                 y;
     350             : 
     351        2251 :   PetscFunctionBegin;
     352        2251 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
     353        6753 :   PetscValidLogicalCollectiveInt(X,j,2);
     354        2251 :   PetscValidType(X,1);
     355        2251 :   BVCheckSizes(X,1);
     356             : 
     357        2251 :   PetscCheck(j>=0,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_OUTOFRANGE,"Index j must be non-negative");
     358        2251 :   PetscCheck(j<X->m,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_OUTOFRANGE,"Index j=%" PetscInt_FMT " but BV only has %" PetscInt_FMT " columns",j,X->m);
     359        2251 :   ksave = X->k;
     360        2251 :   X->k = j;
     361        2251 :   PetscCall(BVGetColumn(X,j,&y));
     362             : 
     363        2251 :   if (X->ops->dotvec_begin) PetscUseTypeMethod(X,dotvec_begin,y,m);
     364             :   else {
     365        2249 :     BVCheckOp(X,1,dotvec_local);
     366        2249 :     nv = X->k-X->l;
     367        2249 :     PetscCall(PetscObjectGetComm((PetscObject)X,&comm));
     368        2249 :     PetscCall(PetscSplitReductionGet(comm,&sr));
     369        2249 :     PetscCheck(sr->state==STATE_BEGIN,PetscObjectComm((PetscObject)X),PETSC_ERR_ORDER,"Called before all VecxxxEnd() called");
     370       22902 :     for (i=0;i<nv;i++) {
     371       20653 :       if (sr->numopsbegin+i >= sr->maxops) PetscCall(PetscSplitReductionExtend(sr));
     372       20653 :       sr->reducetype[sr->numopsbegin+i] = PETSC_SR_REDUCE_SUM;
     373       20653 :       sr->invecs[sr->numopsbegin+i]     = (void*)X;
     374             :     }
     375        2249 :     PetscCall(PetscLogEventBegin(BV_DotVec,X,0,0,0));
     376        2249 :     PetscUseTypeMethod(X,dotvec_local,y,sr->lvalues+sr->numopsbegin);
     377        2249 :     sr->numopsbegin += nv;
     378        2249 :     PetscCall(PetscLogEventEnd(BV_DotVec,X,0,0,0));
     379             :   }
     380        2251 :   PetscCall(BVRestoreColumn(X,j,&y));
     381        2251 :   X->k = ksave;
     382        2251 :   PetscFunctionReturn(PETSC_SUCCESS);
     383             : }
     384             : 
     385             : /*@
     386             :    BVDotColumnEnd - Ends a split phase dot product computation.
     387             : 
     388             :    Input Parameters:
     389             : +  X - basis vectors
     390             : .  j - the column index
     391             : -  m - an array where the result will go
     392             : 
     393             :    Notes:
     394             :    Each call to BVDotColumnBegin() should be paired with a call to BVDotColumnEnd().
     395             : 
     396             :    Level: advanced
     397             : 
     398             : .seealso: BVDotColumnBegin(), BVDotColumn()
     399             : @*/
     400        2251 : PetscErrorCode BVDotColumnEnd(BV X,PetscInt j,PetscScalar *m)
     401             : {
     402        2251 :   PetscInt            i,nv,ksave;
     403        2251 :   PetscSplitReduction *sr;
     404        2251 :   MPI_Comm            comm;
     405        2251 :   Vec                 y;
     406             : 
     407        2251 :   PetscFunctionBegin;
     408        2251 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
     409        6753 :   PetscValidLogicalCollectiveInt(X,j,2);
     410        2251 :   PetscValidType(X,1);
     411        2251 :   BVCheckSizes(X,1);
     412             : 
     413        2251 :   PetscCheck(j>=0,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_OUTOFRANGE,"Index j must be non-negative");
     414        2251 :   PetscCheck(j<X->m,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_OUTOFRANGE,"Index j=%" PetscInt_FMT " but BV only has %" PetscInt_FMT " columns",j,X->m);
     415        2251 :   ksave = X->k;
     416        2251 :   X->k = j;
     417             : 
     418        2251 :   if (X->ops->dotvec_end) {
     419           2 :     PetscCall(BVGetColumn(X,j,&y));
     420           2 :     PetscUseTypeMethod(X,dotvec_end,y,m);
     421           2 :     PetscCall(BVRestoreColumn(X,j,&y));
     422             :   } else {
     423        2249 :     nv = X->k-X->l;
     424        2249 :     PetscCall(PetscObjectGetComm((PetscObject)X,&comm));
     425        2249 :     PetscCall(PetscSplitReductionGet(comm,&sr));
     426        2249 :     PetscCall(PetscSplitReductionEnd(sr));
     427             : 
     428        2249 :     PetscCheck(sr->numopsend<sr->numopsbegin,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"Called VecxxxEnd() more times than VecxxxBegin()");
     429        2249 :     PetscCheck((void*)X==sr->invecs[sr->numopsend],PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"Called BVxxxEnd() in a different order or with a different BV than BVxxxBegin()");
     430        2249 :     PetscCheck(sr->reducetype[sr->numopsend]==PETSC_SR_REDUCE_SUM,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_WRONGSTATE,"Wrong type of reduction");
     431       22902 :     for (i=0;i<nv;i++) m[i] = sr->gvalues[sr->numopsend++];
     432             : 
     433             :     /* Finished getting all the results so reset to no outstanding requests */
     434        2249 :     if (sr->numopsend == sr->numopsbegin) {
     435         619 :       sr->state       = STATE_BEGIN;
     436         619 :       sr->numopsend   = 0;
     437         619 :       sr->numopsbegin = 0;
     438             :     }
     439             :   }
     440        2251 :   X->k = ksave;
     441        2251 :   PetscFunctionReturn(PETSC_SUCCESS);
     442             : }
     443             : 
     444       12487 : static inline PetscErrorCode BVNorm_Private(BV bv,Vec z,NormType type,PetscReal *val)
     445             : {
     446       12487 :   PetscScalar    p;
     447             : 
     448       12487 :   PetscFunctionBegin;
     449       12487 :   PetscCall(BV_IPMatMult(bv,z));
     450       12487 :   PetscCall(VecDot(bv->Bx,z,&p));
     451       12487 :   PetscCall(BV_SafeSqrt(bv,p,val));
     452       12487 :   PetscFunctionReturn(PETSC_SUCCESS);
     453             : }
     454             : 
     455         431 : static inline PetscErrorCode BVNorm_Begin_Private(BV bv,Vec z,NormType type,PetscReal *val)
     456             : {
     457         431 :   PetscScalar    p;
     458             : 
     459         431 :   PetscFunctionBegin;
     460         431 :   PetscCall(BV_IPMatMult(bv,z));
     461         431 :   PetscCall(VecDotBegin(bv->Bx,z,&p));
     462         431 :   PetscFunctionReturn(PETSC_SUCCESS);
     463             : }
     464             : 
     465         431 : static inline PetscErrorCode BVNorm_End_Private(BV bv,Vec z,NormType type,PetscReal *val)
     466             : {
     467         431 :   PetscScalar    p;
     468             : 
     469         431 :   PetscFunctionBegin;
     470         431 :   PetscCall(VecDotEnd(bv->Bx,z,&p));
     471         431 :   PetscCall(BV_SafeSqrt(bv,p,val));
     472         431 :   PetscFunctionReturn(PETSC_SUCCESS);
     473             : }
     474             : 
     475             : /*@
     476             :    BVNorm - Computes the matrix norm of the BV.
     477             : 
     478             :    Collective
     479             : 
     480             :    Input Parameters:
     481             : +  bv   - basis vectors
     482             : -  type - the norm type
     483             : 
     484             :    Output Parameter:
     485             : .  val  - the norm
     486             : 
     487             :    Notes:
     488             :    All active columns (except the leading ones) are considered as a matrix.
     489             :    The allowed norms are NORM_1, NORM_FROBENIUS, and NORM_INFINITY.
     490             : 
     491             :    This operation fails if a non-standard inner product has been
     492             :    specified with BVSetMatrix().
     493             : 
     494             :    Level: intermediate
     495             : 
     496             : .seealso: BVNormVec(), BVNormColumn(), BVNormalize(), BVSetActiveColumns(), BVSetMatrix()
     497             : @*/
     498         283 : PetscErrorCode BVNorm(BV bv,NormType type,PetscReal *val)
     499             : {
     500         283 :   PetscFunctionBegin;
     501         283 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     502         849 :   PetscValidLogicalCollectiveEnum(bv,type,2);
     503         283 :   PetscAssertPointer(val,3);
     504         283 :   PetscValidType(bv,1);
     505         283 :   BVCheckSizes(bv,1);
     506             : 
     507         283 :   PetscCheck(type!=NORM_2 && type!=NORM_1_AND_2,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     508         283 :   PetscCheck(!bv->matrix,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Matrix norm not available for non-standard inner product");
     509             : 
     510         283 :   PetscCall(PetscLogEventBegin(BV_Norm,bv,0,0,0));
     511         283 :   PetscUseTypeMethod(bv,norm,-1,type,val);
     512         283 :   PetscCall(PetscLogEventEnd(BV_Norm,bv,0,0,0));
     513         283 :   PetscFunctionReturn(PETSC_SUCCESS);
     514             : }
     515             : 
     516             : /*@
     517             :    BVNormVec - Computes the norm of a given vector.
     518             : 
     519             :    Collective
     520             : 
     521             :    Input Parameters:
     522             : +  bv   - basis vectors
     523             : .  v    - the vector
     524             : -  type - the norm type
     525             : 
     526             :    Output Parameter:
     527             : .  val  - the norm
     528             : 
     529             :    Notes:
     530             :    This is the analogue of BVNormColumn() but for a vector that is not in the BV.
     531             :    If a non-standard inner product has been specified with BVSetMatrix(),
     532             :    then the returned value is sqrt(v'*B*v), where B is the inner product
     533             :    matrix (argument 'type' is ignored). Otherwise, VecNorm() is called.
     534             : 
     535             :    Level: developer
     536             : 
     537             : .seealso: BVNorm(), BVNormColumn(), BVSetMatrix()
     538             : @*/
     539       83569 : PetscErrorCode BVNormVec(BV bv,Vec v,NormType type,PetscReal *val)
     540             : {
     541       83569 :   PetscInt       n;
     542             : 
     543       83569 :   PetscFunctionBegin;
     544       83569 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     545       83569 :   PetscValidHeaderSpecific(v,VEC_CLASSID,2);
     546      250707 :   PetscValidLogicalCollectiveEnum(bv,type,3);
     547       83569 :   PetscAssertPointer(val,4);
     548       83569 :   PetscValidType(bv,1);
     549       83569 :   BVCheckSizes(bv,1);
     550       83569 :   PetscValidType(v,2);
     551       83569 :   PetscCheckSameComm(bv,1,v,2);
     552             : 
     553       83569 :   PetscCheck(type!=NORM_1_AND_2 || bv->matrix,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     554             : 
     555       83569 :   PetscCall(PetscLogEventBegin(BV_NormVec,bv,0,0,0));
     556       83569 :   if (bv->matrix) { /* non-standard inner product */
     557        6758 :     PetscCall(VecGetLocalSize(v,&n));
     558        6758 :     PetscCheck(bv->n==n,PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_INCOMP,"Mismatching local dimension bv %" PetscInt_FMT ", v %" PetscInt_FMT,bv->n,n);
     559        6758 :     PetscCall(BVNorm_Private(bv,v,type,val));
     560       76811 :   } else PetscCall(VecNorm(v,type,val));
     561       83569 :   PetscCall(PetscLogEventEnd(BV_NormVec,bv,0,0,0));
     562       83569 :   PetscFunctionReturn(PETSC_SUCCESS);
     563             : }
     564             : 
     565             : /*@
     566             :    BVNormVecBegin - Starts a split phase norm computation.
     567             : 
     568             :    Input Parameters:
     569             : +  bv   - basis vectors
     570             : .  v    - the vector
     571             : .  type - the norm type
     572             : -  val  - the norm
     573             : 
     574             :    Note:
     575             :    Each call to BVNormVecBegin() should be paired with a call to BVNormVecEnd().
     576             : 
     577             :    Level: advanced
     578             : 
     579             : .seealso: BVNormVecEnd(), BVNormVec()
     580             : @*/
     581         658 : PetscErrorCode BVNormVecBegin(BV bv,Vec v,NormType type,PetscReal *val)
     582             : {
     583         658 :   PetscInt       n;
     584             : 
     585         658 :   PetscFunctionBegin;
     586         658 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     587         658 :   PetscValidHeaderSpecific(v,VEC_CLASSID,2);
     588        1974 :   PetscValidLogicalCollectiveEnum(bv,type,3);
     589         658 :   PetscAssertPointer(val,4);
     590         658 :   PetscValidType(bv,1);
     591         658 :   BVCheckSizes(bv,1);
     592         658 :   PetscValidType(v,2);
     593         658 :   PetscCheckSameTypeAndComm(bv,1,v,2);
     594             : 
     595         658 :   PetscCheck(type!=NORM_1_AND_2 || bv->matrix,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     596             : 
     597         658 :   PetscCall(PetscLogEventBegin(BV_NormVec,bv,0,0,0));
     598         658 :   if (bv->matrix) { /* non-standard inner product */
     599          21 :     PetscCall(VecGetLocalSize(v,&n));
     600          21 :     PetscCheck(bv->n==n,PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_INCOMP,"Mismatching local dimension bv %" PetscInt_FMT ", v %" PetscInt_FMT,bv->n,n);
     601          21 :     PetscCall(BVNorm_Begin_Private(bv,v,type,val));
     602         637 :   } else PetscCall(VecNormBegin(v,type,val));
     603         658 :   PetscCall(PetscLogEventEnd(BV_NormVec,bv,0,0,0));
     604         658 :   PetscFunctionReturn(PETSC_SUCCESS);
     605             : }
     606             : 
     607             : /*@
     608             :    BVNormVecEnd - Ends a split phase norm computation.
     609             : 
     610             :    Input Parameters:
     611             : +  bv   - basis vectors
     612             : .  v    - the vector
     613             : .  type - the norm type
     614             : -  val  - the norm
     615             : 
     616             :    Note:
     617             :    Each call to BVNormVecBegin() should be paired with a call to BVNormVecEnd().
     618             : 
     619             :    Level: advanced
     620             : 
     621             : .seealso: BVNormVecBegin(), BVNormVec()
     622             : @*/
     623         658 : PetscErrorCode BVNormVecEnd(BV bv,Vec v,NormType type,PetscReal *val)
     624             : {
     625         658 :   PetscFunctionBegin;
     626         658 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     627        1974 :   PetscValidLogicalCollectiveEnum(bv,type,3);
     628         658 :   PetscAssertPointer(val,4);
     629         658 :   PetscValidType(bv,1);
     630         658 :   BVCheckSizes(bv,1);
     631             : 
     632         658 :   PetscCheck(type!=NORM_1_AND_2,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     633             : 
     634         658 :   if (bv->matrix) PetscCall(BVNorm_End_Private(bv,v,type,val));  /* non-standard inner product */
     635         637 :   else PetscCall(VecNormEnd(v,type,val));
     636         658 :   PetscFunctionReturn(PETSC_SUCCESS);
     637             : }
     638             : 
     639             : /*@
     640             :    BVNormColumn - Computes the vector norm of a selected column.
     641             : 
     642             :    Collective
     643             : 
     644             :    Input Parameters:
     645             : +  bv   - basis vectors
     646             : .  j    - column number to be used
     647             : -  type - the norm type
     648             : 
     649             :    Output Parameter:
     650             : .  val  - the norm
     651             : 
     652             :    Notes:
     653             :    The norm of V[j] is computed (NORM_1, NORM_2, or NORM_INFINITY).
     654             :    If a non-standard inner product has been specified with BVSetMatrix(),
     655             :    then the returned value is sqrt(V[j]'*B*V[j]),
     656             :    where B is the inner product matrix (argument 'type' is ignored).
     657             : 
     658             :    Level: intermediate
     659             : 
     660             : .seealso: BVNorm(), BVNormVec(), BVNormalize(), BVSetActiveColumns(), BVSetMatrix()
     661             : @*/
     662       14064 : PetscErrorCode BVNormColumn(BV bv,PetscInt j,NormType type,PetscReal *val)
     663             : {
     664       14064 :   Vec            z;
     665             : 
     666       14064 :   PetscFunctionBegin;
     667       14064 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     668       42192 :   PetscValidLogicalCollectiveInt(bv,j,2);
     669       42192 :   PetscValidLogicalCollectiveEnum(bv,type,3);
     670       14064 :   PetscAssertPointer(val,4);
     671       14064 :   PetscValidType(bv,1);
     672       14064 :   BVCheckSizes(bv,1);
     673             : 
     674       14064 :   PetscCheck(j>=0 && j<bv->m,PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_OUTOFRANGE,"Argument j has wrong value %" PetscInt_FMT ", the number of columns is %" PetscInt_FMT,j,bv->m);
     675       14064 :   PetscCheck(type!=NORM_1_AND_2 || bv->matrix,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     676             : 
     677       14064 :   PetscCall(PetscLogEventBegin(BV_NormVec,bv,0,0,0));
     678       14064 :   if (bv->matrix) { /* non-standard inner product */
     679        5729 :     PetscCall(BVGetColumn(bv,j,&z));
     680        5729 :     PetscCall(BVNorm_Private(bv,z,type,val));
     681        5729 :     PetscCall(BVRestoreColumn(bv,j,&z));
     682        8335 :   } else PetscUseTypeMethod(bv,norm,j,type,val);
     683       14064 :   PetscCall(PetscLogEventEnd(BV_NormVec,bv,0,0,0));
     684       14064 :   PetscFunctionReturn(PETSC_SUCCESS);
     685             : }
     686             : 
     687             : /*@
     688             :    BVNormColumnBegin - Starts a split phase norm computation.
     689             : 
     690             :    Input Parameters:
     691             : +  bv   - basis vectors
     692             : .  j    - column number to be used
     693             : .  type - the norm type
     694             : -  val  - the norm
     695             : 
     696             :    Note:
     697             :    Each call to BVNormColumnBegin() should be paired with a call to BVNormColumnEnd().
     698             : 
     699             :    Level: advanced
     700             : 
     701             : .seealso: BVNormColumnEnd(), BVNormColumn()
     702             : @*/
     703        2959 : PetscErrorCode BVNormColumnBegin(BV bv,PetscInt j,NormType type,PetscReal *val)
     704             : {
     705        2959 :   PetscSplitReduction *sr;
     706        2959 :   PetscReal           lresult;
     707        2959 :   MPI_Comm            comm;
     708        2959 :   Vec                 z;
     709             : 
     710        2959 :   PetscFunctionBegin;
     711        2959 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     712        8877 :   PetscValidLogicalCollectiveInt(bv,j,2);
     713        8877 :   PetscValidLogicalCollectiveEnum(bv,type,3);
     714        2959 :   PetscAssertPointer(val,4);
     715        2959 :   PetscValidType(bv,1);
     716        2959 :   BVCheckSizes(bv,1);
     717             : 
     718        2959 :   PetscCheck(j>=0 && j<bv->m,PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_OUTOFRANGE,"Argument j has wrong value %" PetscInt_FMT ", the number of columns is %" PetscInt_FMT,j,bv->m);
     719        2959 :   PetscCheck(type!=NORM_1_AND_2 || bv->matrix,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     720             : 
     721        2959 :   PetscCall(PetscLogEventBegin(BV_NormVec,bv,0,0,0));
     722        2959 :   PetscCall(BVGetColumn(bv,j,&z));
     723        2959 :   if (bv->matrix) PetscCall(BVNorm_Begin_Private(bv,z,type,val)); /* non-standard inner product */
     724        2549 :   else if (bv->ops->norm_begin) PetscUseTypeMethod(bv,norm_begin,j,type,val);
     725             :   else {
     726        2547 :     BVCheckOp(bv,1,norm_local);
     727        2547 :     PetscCall(PetscObjectGetComm((PetscObject)z,&comm));
     728        2547 :     PetscCall(PetscSplitReductionGet(comm,&sr));
     729        2547 :     PetscCheck(sr->state==STATE_BEGIN,PetscObjectComm((PetscObject)bv),PETSC_ERR_ORDER,"Called before all VecxxxEnd() called");
     730        2547 :     if (sr->numopsbegin >= sr->maxops) PetscCall(PetscSplitReductionExtend(sr));
     731        2547 :     sr->invecs[sr->numopsbegin] = (void*)bv;
     732        2547 :     PetscUseTypeMethod(bv,norm_local,j,type,&lresult);
     733        2547 :     if (type == NORM_2) lresult = lresult*lresult;
     734        2547 :     if (type == NORM_MAX) sr->reducetype[sr->numopsbegin] = PETSC_SR_REDUCE_MAX;
     735        2547 :     else sr->reducetype[sr->numopsbegin] = PETSC_SR_REDUCE_SUM;
     736        2547 :     sr->lvalues[sr->numopsbegin++] = lresult;
     737             :   }
     738        2959 :   PetscCall(BVRestoreColumn(bv,j,&z));
     739        2959 :   PetscCall(PetscLogEventEnd(BV_NormVec,bv,0,0,0));
     740        2959 :   PetscFunctionReturn(PETSC_SUCCESS);
     741             : }
     742             : 
     743             : /*@
     744             :    BVNormColumnEnd - Ends a split phase norm computation.
     745             : 
     746             :    Input Parameters:
     747             : +  bv   - basis vectors
     748             : .  j    - column number to be used
     749             : .  type - the norm type
     750             : -  val  - the norm
     751             : 
     752             :    Note:
     753             :    Each call to BVNormColumnBegin() should be paired with a call to BVNormColumnEnd().
     754             : 
     755             :    Level: advanced
     756             : 
     757             : .seealso: BVNormColumnBegin(), BVNormColumn()
     758             : @*/
     759        2959 : PetscErrorCode BVNormColumnEnd(BV bv,PetscInt j,NormType type,PetscReal *val)
     760             : {
     761        2959 :   PetscSplitReduction *sr;
     762        2959 :   MPI_Comm            comm;
     763        2959 :   Vec                 z;
     764             : 
     765        2959 :   PetscFunctionBegin;
     766        2959 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     767        8877 :   PetscValidLogicalCollectiveInt(bv,j,2);
     768        8877 :   PetscValidLogicalCollectiveEnum(bv,type,3);
     769        2959 :   PetscAssertPointer(val,4);
     770        2959 :   PetscValidType(bv,1);
     771        2959 :   BVCheckSizes(bv,1);
     772             : 
     773        2959 :   PetscCheck(type!=NORM_1_AND_2,PetscObjectComm((PetscObject)bv),PETSC_ERR_SUP,"Requested norm not available");
     774             : 
     775        2959 :   PetscCall(BVGetColumn(bv,j,&z));
     776        2959 :   if (bv->matrix) PetscCall(BVNorm_End_Private(bv,z,type,val)); /* non-standard inner product */
     777        2549 :   else if (bv->ops->norm_end) PetscUseTypeMethod(bv,norm_end,j,type,val);
     778             :   else {
     779        2547 :     PetscCall(PetscObjectGetComm((PetscObject)z,&comm));
     780        2547 :     PetscCall(PetscSplitReductionGet(comm,&sr));
     781        2547 :     PetscCall(PetscSplitReductionEnd(sr));
     782             : 
     783        2547 :     PetscCheck(sr->numopsend<sr->numopsbegin,PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_WRONGSTATE,"Called VecxxxEnd() more times then VecxxxBegin()");
     784        2547 :     PetscCheck((void*)bv==sr->invecs[sr->numopsend],PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_WRONGSTATE,"Called VecxxxEnd() in a different order or with a different vector than VecxxxBegin()");
     785        2547 :     PetscCheck(sr->reducetype[sr->numopsend]==PETSC_SR_REDUCE_MAX || type!=NORM_MAX,PetscObjectComm((PetscObject)bv),PETSC_ERR_ARG_WRONGSTATE,"Called BVNormEnd(,NORM_MAX,) on a reduction started with VecDotBegin() or NORM_1 or NORM_2");
     786        2547 :     *val = PetscRealPart(sr->gvalues[sr->numopsend++]);
     787        2547 :     if (type == NORM_2) *val = PetscSqrtReal(*val);
     788        2547 :     if (sr->numopsend == sr->numopsbegin) {
     789         144 :       sr->state       = STATE_BEGIN;
     790         144 :       sr->numopsend   = 0;
     791         144 :       sr->numopsbegin = 0;
     792             :     }
     793             :   }
     794        2959 :   PetscCall(BVRestoreColumn(bv,j,&z));
     795        2959 :   PetscFunctionReturn(PETSC_SUCCESS);
     796             : }
     797             : 
     798             : /*@
     799             :    BVNormalize - Normalize all columns (starting from the leading ones).
     800             : 
     801             :    Collective
     802             : 
     803             :    Input Parameters:
     804             : +  bv   - basis vectors
     805             : -  eigi - (optional) imaginary parts of eigenvalues
     806             : 
     807             :    Notes:
     808             :    On output, all columns will have unit norm. The normalization is done with
     809             :    respect to the 2-norm, or to the B-norm if a non-standard inner product has
     810             :    been specified with BVSetMatrix(), see BVNormColumn().
     811             : 
     812             :    If the optional argument eigi is passed (taken into account only in real
     813             :    scalars) it is interpreted as the imaginary parts of the eigenvalues and
     814             :    the BV is supposed to contain the corresponding eigenvectors. Suppose the
     815             :    first three values are eigi = { 0, alpha, -alpha }, then the first column
     816             :    is normalized as usual, but the second and third ones are normalized assuming
     817             :    that they contain the real and imaginary parts of a complex conjugate pair of
     818             :    eigenvectors.
     819             : 
     820             :    If eigi is passed, the inner-product matrix is ignored.
     821             : 
     822             :    If there are leading columns, they are not modified (are assumed to be already
     823             :    normalized).
     824             : 
     825             :    Level: intermediate
     826             : 
     827             : .seealso: BVNormColumn()
     828             : @*/
     829        3968 : PetscErrorCode BVNormalize(BV bv,PetscScalar *eigi)
     830             : {
     831        3968 :   PetscReal      norm;
     832        3968 :   PetscInt       i;
     833             : 
     834        3968 :   PetscFunctionBegin;
     835        3968 :   PetscValidHeaderSpecific(bv,BV_CLASSID,1);
     836        3968 :   PetscValidType(bv,1);
     837        3968 :   BVCheckSizes(bv,1);
     838             : 
     839        3968 :   PetscCall(PetscLogEventBegin(BV_Normalize,bv,0,0,0));
     840        3968 :   if (bv->matrix && !eigi) {
     841        1158 :     for (i=bv->l;i<bv->k;i++) {
     842        1087 :       PetscCall(BVNormColumn(bv,i,NORM_2,&norm));
     843        1087 :       PetscCall(BVScaleColumn(bv,i,1.0/norm));
     844             :     }
     845        3897 :   } else PetscTryTypeMethod(bv,normalize,eigi);
     846        3968 :   PetscCall(PetscLogEventEnd(BV_Normalize,bv,0,0,0));
     847        3968 :   PetscCall(PetscObjectStateIncrease((PetscObject)bv));
     848        3968 :   PetscFunctionReturn(PETSC_SUCCESS);
     849             : }
     850             : 
     851             : /*
     852             :   Compute Y^H*A*X: right part column by column (with MatMult) and bottom
     853             :   part row by row (with MatMultHermitianTranspose); result placed in marray[*,ldm]
     854             : */
     855          18 : static inline PetscErrorCode BVMatProject_Vec(BV X,Mat A,BV Y,PetscScalar *marray,PetscInt ldm,PetscBool symm)
     856             : {
     857          18 :   PetscInt       i,j,lx,ly,kx,ky,ulim;
     858          18 :   Vec            z,f;
     859             : 
     860          18 :   PetscFunctionBegin;
     861          18 :   lx = X->l; kx = X->k;
     862          18 :   ly = Y->l; ky = Y->k;
     863          18 :   PetscCall(BVCreateVec(X,&f));
     864          18 :   BVCheckOp(Y,3,dotvec);
     865          72 :   for (j=lx;j<kx;j++) {
     866          54 :     PetscCall(BVGetColumn(X,j,&z));
     867          54 :     PetscCall(MatMult(A,z,f));
     868          54 :     PetscCall(BVRestoreColumn(X,j,&z));
     869          54 :     ulim = PetscMin(ly+(j-lx)+1,ky);
     870          54 :     Y->l = 0; Y->k = ulim;
     871          54 :     PetscUseTypeMethod(Y,dotvec,f,marray+j*ldm);
     872          54 :     if (symm) {
     873          90 :       for (i=0;i<j;i++) marray[j+i*ldm] = PetscConj(marray[i+j*ldm]);
     874             :     }
     875             :   }
     876          18 :   if (!symm) {
     877          12 :     BVCheckOp(X,1,dotvec);
     878          12 :     PetscCall(BV_AllocateCoeffs(Y));
     879          48 :     for (j=ly;j<ky;j++) {
     880          36 :       PetscCall(BVGetColumn(Y,j,&z));
     881          36 :       PetscCall(MatMultHermitianTranspose(A,z,f));
     882          36 :       PetscCall(BVRestoreColumn(Y,j,&z));
     883          36 :       ulim = PetscMin(lx+(j-ly),kx);
     884          36 :       X->l = 0; X->k = ulim;
     885          36 :       PetscUseTypeMethod(X,dotvec,f,Y->h);
     886         180 :       for (i=0;i<ulim;i++) marray[j+i*ldm] = PetscConj(Y->h[i]);
     887             :     }
     888             :   }
     889          18 :   PetscCall(VecDestroy(&f));
     890          18 :   X->l = lx; X->k = kx;
     891          18 :   Y->l = ly; Y->k = ky;
     892          18 :   PetscFunctionReturn(PETSC_SUCCESS);
     893             : }
     894             : 
     895             : /*
     896             :   Compute Y^H*A*X= [   --   | Y0'*W1 ]
     897             :                    [ Y1'*W0 | Y1'*W1 ]
     898             :   Allocates auxiliary BV to store the result of A*X, then one BVDot
     899             :   call for top-right part and another one for bottom part;
     900             :   result placed in marray[*,ldm]
     901             : */
     902        2076 : static inline PetscErrorCode BVMatProject_MatMult(BV X,Mat A,BV Y,PetscScalar *marray,PetscInt ldm)
     903             : {
     904        2076 :   PetscInt          j,lx,ly,kx,ky;
     905        2076 :   const PetscScalar *harray;
     906        2076 :   Mat               H;
     907        2076 :   BV                W;
     908             : 
     909        2076 :   PetscFunctionBegin;
     910        2076 :   lx = X->l; kx = X->k;
     911        2076 :   ly = Y->l; ky = Y->k;
     912        2076 :   PetscCall(BVDuplicate(X,&W));
     913        2076 :   X->l = 0; X->k = kx;
     914        2076 :   W->l = 0; W->k = kx;
     915        2076 :   PetscCall(BVMatMult(X,A,W));
     916             : 
     917             :   /* top-right part, Y0'*AX1 */
     918        2076 :   if (ly>0 && lx<kx) {
     919           0 :     PetscCall(MatCreateSeqDense(PETSC_COMM_SELF,ly,kx,NULL,&H));
     920           0 :     W->l = lx; W->k = kx;
     921           0 :     Y->l = 0;  Y->k = ly;
     922           0 :     PetscCall(BVDot(W,Y,H));
     923           0 :     PetscCall(MatDenseGetArrayRead(H,&harray));
     924           0 :     for (j=lx;j<kx;j++) PetscCall(PetscArraycpy(marray+j*ldm,harray+j*ly,ly));
     925           0 :     PetscCall(MatDenseRestoreArrayRead(H,&harray));
     926           0 :     PetscCall(MatDestroy(&H));
     927             :   }
     928             : 
     929             :   /* bottom part, Y1'*AX */
     930        2076 :   if (kx>0 && ly<ky) {
     931        2076 :     PetscCall(MatCreateSeqDense(PETSC_COMM_SELF,ky,kx,NULL,&H));
     932        2076 :     W->l = 0;  W->k = kx;
     933        2076 :     Y->l = ly; Y->k = ky;
     934        2076 :     PetscCall(BVDot(W,Y,H));
     935        2076 :     PetscCall(MatDenseGetArrayRead(H,&harray));
     936       23766 :     for (j=0;j<kx;j++) PetscCall(PetscArraycpy(marray+j*ldm+ly,harray+j*ky+ly,ky-ly));
     937        2076 :     PetscCall(MatDenseRestoreArrayRead(H,&harray));
     938        2076 :     PetscCall(MatDestroy(&H));
     939             :   }
     940        2076 :   PetscCall(BVDestroy(&W));
     941        2076 :   X->l = lx; X->k = kx;
     942        2076 :   Y->l = ly; Y->k = ky;
     943        2076 :   PetscFunctionReturn(PETSC_SUCCESS);
     944             : }
     945             : 
     946             : /*
     947             :   Compute Y^H*A*X= [   --   | Y0'*W1 ]
     948             :                    [ Y1'*W0 | Y1'*W1 ]
     949             :   First stage: allocate auxiliary BV to store A*X1, one BVDot for right part;
     950             :   Second stage: resize BV to accommodate A'*Y1, then call BVDot for transpose of
     951             :   bottom-left part; result placed in marray[*,ldm]
     952             : */
     953         570 : static inline PetscErrorCode BVMatProject_MatMult_2(BV X,Mat A,BV Y,PetscScalar *marray,PetscInt ldm,PetscBool symm)
     954             : {
     955         570 :   PetscInt          i,j,lx,ly,kx,ky;
     956         570 :   const PetscScalar *harray;
     957         570 :   Mat               H;
     958         570 :   BV                W;
     959             : 
     960         570 :   PetscFunctionBegin;
     961         570 :   lx = X->l; kx = X->k;
     962         570 :   ly = Y->l; ky = Y->k;
     963             : 
     964             :   /* right part, Y'*AX1 */
     965         570 :   PetscCall(BVDuplicateResize(X,kx-lx,&W));
     966         570 :   if (ky>0 && lx<kx) {
     967         570 :     PetscCall(BVMatMult(X,A,W));
     968         570 :     PetscCall(MatCreateSeqDense(PETSC_COMM_SELF,ky,kx-lx,NULL,&H));
     969         570 :     Y->l = 0; Y->k = ky;
     970         570 :     PetscCall(BVDot(W,Y,H));
     971         570 :     PetscCall(MatDenseGetArrayRead(H,&harray));
     972        1654 :     for (j=lx;j<kx;j++) PetscCall(PetscArraycpy(marray+j*ldm,harray+(j-lx)*ky,ky));
     973         570 :     PetscCall(MatDenseRestoreArrayRead(H,&harray));
     974         570 :     PetscCall(MatDestroy(&H));
     975             :   }
     976             : 
     977             :   /* bottom-left part, Y1'*AX0 */
     978         570 :   if (lx>0 && ly<ky) {
     979         442 :     if (symm) {
     980             :       /* do not compute, just copy symmetric elements */
     981         762 :       for (i=ly;i<ky;i++) {
     982        1764 :         for (j=0;j<lx;j++) marray[i+j*ldm] = PetscConj(marray[j+i*ldm]);
     983             :       }
     984             :     } else {
     985          70 :       PetscCall(BVResize(W,ky-ly,PETSC_FALSE));
     986          70 :       Y->l = ly; Y->k = ky;
     987          70 :       PetscCall(BVMatMultHermitianTranspose(Y,A,W));
     988          70 :       PetscCall(MatCreateSeqDense(PETSC_COMM_SELF,lx,ky-ly,NULL,&H));
     989          70 :       X->l = 0; X->k = lx;
     990          70 :       PetscCall(BVDot(W,X,H));
     991          70 :       PetscCall(MatDenseGetArrayRead(H,&harray));
     992         180 :       for (i=0;i<ky-ly;i++) {
     993         452 :         for (j=0;j<lx;j++) {
     994         342 :           marray[i+j*ldm+ly] = PetscConj(harray[j+i*(ky-ly)]);
     995             :         }
     996             :       }
     997          70 :       PetscCall(MatDenseRestoreArrayRead(H,&harray));
     998          70 :       PetscCall(MatDestroy(&H));
     999             :     }
    1000             :   }
    1001         570 :   PetscCall(BVDestroy(&W));
    1002         570 :   X->l = lx; X->k = kx;
    1003         570 :   Y->l = ly; Y->k = ky;
    1004         570 :   PetscFunctionReturn(PETSC_SUCCESS);
    1005             : }
    1006             : 
    1007             : /*
    1008             :   Compute Y^H*X = [   --   | Y0'*X1 ]     (X contains A*X):
    1009             :                   [ Y1'*X0 | Y1'*X1 ]
    1010             :   one BVDot call for top-right part and another one for bottom part;
    1011             :   result placed in marray[*,ldm]
    1012             : */
    1013       16544 : static inline PetscErrorCode BVMatProject_Dot(BV X,BV Y,PetscScalar *marray,PetscInt ldm)
    1014             : {
    1015       16544 :   PetscInt          j,lx,ly,kx,ky;
    1016       16544 :   const PetscScalar *harray;
    1017       16544 :   Mat               H;
    1018             : 
    1019       16544 :   PetscFunctionBegin;
    1020       16544 :   lx = X->l; kx = X->k;
    1021       16544 :   ly = Y->l; ky = Y->k;
    1022             : 
    1023             :   /* top-right part, Y0'*X1 */
    1024       16544 :   if (ly>0 && lx<kx) {
    1025        9532 :     PetscCall(MatCreateSeqDense(PETSC_COMM_SELF,ly,kx,NULL,&H));
    1026        9532 :     X->l = lx; X->k = kx;
    1027        9532 :     Y->l = 0;  Y->k = ly;
    1028        9532 :     PetscCall(BVDot(X,Y,H));
    1029        9532 :     PetscCall(MatDenseGetArrayRead(H,&harray));
    1030       22928 :     for (j=lx;j<kx;j++) PetscCall(PetscArraycpy(marray+j*ldm,harray+j*ly,ly));
    1031        9532 :     PetscCall(MatDenseRestoreArrayRead(H,&harray));
    1032        9532 :     PetscCall(MatDestroy(&H));
    1033             :   }
    1034             : 
    1035             :   /* bottom part, Y1'*X */
    1036       16544 :   if (kx>0 && ly<ky) {
    1037       16544 :     PetscCall(MatCreateSeqDense(PETSC_COMM_SELF,ky,kx,NULL,&H));
    1038       16544 :     X->l = 0;  X->k = kx;
    1039       16544 :     Y->l = ly; Y->k = ky;
    1040       16544 :     PetscCall(BVDot(X,Y,H));
    1041       16544 :     PetscCall(MatDenseGetArrayRead(H,&harray));
    1042      158529 :     for (j=0;j<kx;j++) PetscCall(PetscArraycpy(marray+j*ldm+ly,harray+j*ky+ly,ky-ly));
    1043       16544 :     PetscCall(MatDenseRestoreArrayRead(H,&harray));
    1044       16544 :     PetscCall(MatDestroy(&H));
    1045             :   }
    1046       16544 :   X->l = lx; X->k = kx;
    1047       16544 :   Y->l = ly; Y->k = ky;
    1048       16544 :   PetscFunctionReturn(PETSC_SUCCESS);
    1049             : }
    1050             : 
    1051             : /*@
    1052             :    BVMatProject - Computes the projection of a matrix onto a subspace.
    1053             : 
    1054             :    Collective
    1055             : 
    1056             :    Input Parameters:
    1057             : +  X - basis vectors
    1058             : .  A - (optional) matrix to be projected
    1059             : -  Y - left basis vectors, can be equal to X
    1060             : 
    1061             :    Output Parameter:
    1062             : .  M - the resulting matrix
    1063             : 
    1064             :    Notes:
    1065             :    If A=NULL, then it is assumed that X already contains A*X.
    1066             : 
    1067             :    This operation is similar to BVDot(), with important differences.
    1068             :    The goal is to compute the matrix resulting from the orthogonal projection
    1069             :    of A onto the subspace spanned by the columns of X, M = X^H*A*X, or the
    1070             :    oblique projection onto X along Y, M = Y^H*A*X.
    1071             : 
    1072             :    A difference with respect to BVDot() is that the standard inner product
    1073             :    is always used, regardless of a non-standard inner product being specified
    1074             :    with BVSetMatrix().
    1075             : 
    1076             :    On entry, M must be a sequential dense Mat with dimensions ky,kx at least,
    1077             :    where ky (resp. kx) is the number of active columns of Y (resp. X).
    1078             :    Another difference with respect to BVDot() is that all entries of M are
    1079             :    computed except the leading ly,lx part, where ly (resp. lx) is the
    1080             :    number of leading columns of Y (resp. X). Hence, the leading columns of
    1081             :    X and Y participate in the computation, as opposed to BVDot().
    1082             :    The leading part of M is assumed to be already available from previous
    1083             :    computations.
    1084             : 
    1085             :    In the orthogonal projection case, Y=X, some computation can be saved if
    1086             :    A is real symmetric (or complex Hermitian). In order to exploit this
    1087             :    property, the symmetry flag of A must be set with MatSetOption().
    1088             : 
    1089             :    Level: intermediate
    1090             : 
    1091             : .seealso: BVDot(), BVSetActiveColumns(), BVSetMatrix()
    1092             : @*/
    1093       19208 : PetscErrorCode BVMatProject(BV X,Mat A,BV Y,Mat M)
    1094             : {
    1095       19208 :   PetscBool      set,flg,symm=PETSC_FALSE;
    1096       19208 :   PetscInt       m,n,ldm;
    1097       19208 :   PetscScalar    *marray;
    1098       19208 :   Mat            Xmatrix,Ymatrix;
    1099       19208 :   PetscObjectId  idx,idy;
    1100             : 
    1101       19208 :   PetscFunctionBegin;
    1102       19208 :   PetscValidHeaderSpecific(X,BV_CLASSID,1);
    1103       19208 :   if (A) PetscValidHeaderSpecific(A,MAT_CLASSID,2);
    1104       19208 :   PetscValidHeaderSpecific(Y,BV_CLASSID,3);
    1105       19208 :   PetscValidHeaderSpecific(M,MAT_CLASSID,4);
    1106       19208 :   PetscValidType(X,1);
    1107       19208 :   BVCheckSizes(X,1);
    1108       19208 :   if (A) {
    1109        2664 :     PetscValidType(A,2);
    1110        2664 :     PetscCheckSameComm(X,1,A,2);
    1111             :   }
    1112       19208 :   PetscValidType(Y,3);
    1113       19208 :   BVCheckSizes(Y,3);
    1114       19208 :   PetscValidType(M,4);
    1115       19208 :   PetscCheckSameTypeAndComm(X,1,Y,3);
    1116       19208 :   SlepcMatCheckSeq(M);
    1117             : 
    1118       19208 :   PetscCall(MatGetSize(M,&m,&n));
    1119       19208 :   PetscCall(MatDenseGetLDA(M,&ldm));
    1120       19208 :   PetscCheck(m>=Y->k,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_SIZ,"Matrix M has %" PetscInt_FMT " rows, should have at least %" PetscInt_FMT,m,Y->k);
    1121       19208 :   PetscCheck(n>=X->k,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_SIZ,"Matrix M has %" PetscInt_FMT " columns, should have at least %" PetscInt_FMT,n,X->k);
    1122       19208 :   PetscCheck(X->n==Y->n,PetscObjectComm((PetscObject)X),PETSC_ERR_ARG_INCOMP,"Mismatching local dimension X %" PetscInt_FMT ", Y %" PetscInt_FMT,X->n,Y->n);
    1123             : 
    1124       19208 :   PetscCall(PetscLogEventBegin(BV_MatProject,X,A,Y,0));
    1125             :   /* temporarily set standard inner product */
    1126       19208 :   Xmatrix = X->matrix;
    1127       19208 :   Ymatrix = Y->matrix;
    1128       19208 :   X->matrix = Y->matrix = NULL;
    1129             : 
    1130       19208 :   PetscCall(PetscObjectGetId((PetscObject)X,&idx));
    1131       19208 :   PetscCall(PetscObjectGetId((PetscObject)Y,&idy));
    1132       19208 :   if (A && idx==idy) { /* check symmetry of M=X'AX */
    1133        2649 :     PetscCall(MatIsHermitianKnown(A,&set,&flg));
    1134        2649 :     symm = set? flg: PETSC_FALSE;
    1135             :   }
    1136             : 
    1137       19208 :   PetscCall(MatDenseGetArray(M,&marray));
    1138             : 
    1139       19208 :   if (A) {
    1140        2664 :     if (X->vmm==BV_MATMULT_VECS) {
    1141             :       /* perform computation column by column */
    1142          18 :       PetscCall(BVMatProject_Vec(X,A,Y,marray,ldm,symm));
    1143             :     } else {
    1144             :       /* use BVMatMult, then BVDot */
    1145        2646 :       PetscCall(MatHasOperation(A,MATOP_MULT_TRANSPOSE,&flg));
    1146        2646 :       if (symm || (flg && X->l>=X->k/2 && Y->l>=Y->k/2)) PetscCall(BVMatProject_MatMult_2(X,A,Y,marray,ldm,symm));
    1147        2076 :       else PetscCall(BVMatProject_MatMult(X,A,Y,marray,ldm));
    1148             :     }
    1149             :   } else {
    1150             :     /* use BVDot on subblocks */
    1151       16544 :     PetscCall(BVMatProject_Dot(X,Y,marray,ldm));
    1152             :   }
    1153             : 
    1154       19208 :   PetscCall(MatDenseRestoreArray(M,&marray));
    1155       19208 :   PetscCall(PetscLogEventEnd(BV_MatProject,X,A,Y,0));
    1156             :   /* restore non-standard inner product */
    1157       19208 :   X->matrix = Xmatrix;
    1158       19208 :   Y->matrix = Ymatrix;
    1159       19208 :   PetscFunctionReturn(PETSC_SUCCESS);
    1160             : }

Generated by: LCOV version 1.14