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 : }
|