Line data Source code
1 : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
2 : Copyright (c) 2011-2023 The plumed team
3 : (see the PEOPLE file at the root of the distribution for a list of names)
4 :
5 : See http://www.plumed.org for more information.
6 :
7 : This file is part of plumed, version 2.
8 :
9 : plumed is free software: you can redistribute it and/or modify
10 : it under the terms of the GNU Lesser General Public License as published by
11 : the Free Software Foundation, either version 3 of the License, or
12 : (at your option) any later version.
13 :
14 : plumed is distributed in the hope that it will be useful,
15 : but WITHOUT ANY WARRANTY; without even the implied warranty of
16 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 : GNU Lesser General Public License for more details.
18 :
19 : You should have received a copy of the GNU Lesser General Public License
20 : along with plumed. If not, see <http://www.gnu.org/licenses/>.
21 : +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
22 : #ifndef __PLUMED_tools_Matrix_h
23 : #define __PLUMED_tools_Matrix_h
24 : #include <vector>
25 : #include <string>
26 : #include <set>
27 : #include <cmath>
28 : #include "Exception.h"
29 : #include "MatrixSquareBracketsAccess.h"
30 : #include "Tools.h"
31 : #include "Log.h"
32 : #include "lapack/lapack.h"
33 :
34 : namespace PLMD {
35 :
36 : /// Calculate the dot product between two vectors
37 : template <typename T> T dotProduct( const std::vector<T>& A, const std::vector<T>& B ) {
38 : plumed_assert( A.size()==B.size() );
39 : T val;
40 : for(unsigned i=0; i<A.size(); ++i) {
41 : val+=A[i]*B[i];
42 : }
43 : return val;
44 : }
45 :
46 : /// Calculate the dot product between a vector and itself
47 : template <typename T> T norm( const std::vector<T>& A ) {
48 : T val;
49 : for(unsigned i=0; i<A.size(); ++i) {
50 : val+=A[i]*A[i];
51 : }
52 : return val;
53 : }
54 :
55 : /// This class stores a full matrix and allows one to do some simple matrix operations
56 : template <typename T>
57 36907669 : class Matrix:
58 : public MatrixSquareBracketsAccess<Matrix<T>,T> {
59 : /// Multiply matrix by scalar
60 : template <typename U> friend Matrix<U> operator*(U&, const Matrix<U>& );
61 : /// Matrix matrix multiply
62 : template <typename U> friend void mult( const Matrix<U>&, const Matrix<U>&, Matrix<U>& );
63 : /// Matrix times a std::vector
64 : template <typename U> friend void mult( const Matrix<U>&, const std::vector<U>&, std::vector<U>& );
65 : /// std::vector times a Matrix
66 : template <typename U> friend void mult( const std::vector<U>&, const Matrix<U>&, std::vector<U>& );
67 : /// Matrix transpose
68 : template <typename U> friend void transpose( const Matrix<U>&, Matrix<U>& );
69 : /// Output the entire matrix on a single line
70 : template <typename U> friend Log& operator<<(Log&, const Matrix<U>& );
71 : /// Output the Matrix in matrix form
72 : template <typename U> friend void matrixOut( Log&, const Matrix<U>& );
73 : /// Diagonalize a symmetric matrix - returns zero if diagonalization worked
74 : template <typename U> friend int diagMat( const Matrix<U>&, std::vector<double>&, Matrix<double>& );
75 : /// Calculate the Moore-Penrose Pseudoinverse of a matrix
76 : template <typename U> friend int pseudoInvert( const Matrix<U>&, Matrix<double>& );
77 : /// Calculate the logarithm of the determinant of a symmetric matrix - returns zero if succesfull
78 : template <typename U> friend int logdet( const Matrix<U>&, double& );
79 : /// Invert a matrix (works for both symmetric and asymmetric matrices) - returns zero if sucesfull
80 : template <typename U> friend int Invert( const Matrix<U>&, Matrix<double>& );
81 : /// Do a cholesky decomposition of a matrix
82 : template <typename U> friend void cholesky( const Matrix<U>&, Matrix<U>& );
83 : /// Solve a system of equations using the cholesky decomposition
84 : template <typename U> friend void chol_elsolve( const Matrix<U>&, const std::vector<U>&, std::vector<U>& );
85 : private:
86 : /// Number of elements in matrix (nrows*ncols)
87 : unsigned sz;
88 : /// Number of rows in matrix
89 : unsigned rw;
90 : /// Number of columns in matrix
91 : unsigned cl;
92 : /// The data in the matrix
93 : std::vector<T> data;
94 : public:
95 36904334 : explicit Matrix(const unsigned nr=0, const unsigned nc=0 ) : sz(nr*nc), rw(nr), cl(nc), data(nr*nc) {}
96 272 : Matrix(const Matrix<T>& t) : sz(t.sz), rw(t.rw), cl(t.cl), data(t.data) {}
97 : /// Resize the matrix
98 : void resize( const unsigned nr, const unsigned nc ) {
99 971 : rw=nr;
100 971 : cl=nc;
101 971 : sz=nr*nc;
102 960 : data.resize(sz);
103 944 : }
104 : /// Return the number of rows
105 : inline unsigned nrows() const {
106 3019033 : return rw;
107 : }
108 : /// Return the number of columns
109 : inline unsigned ncols() const {
110 11458232 : return cl;
111 : }
112 : /// Return the contents of the matrix as a vector of length rw*cl
113 : inline std::vector<T>& getVector() {
114 : return data;
115 : }
116 : /// Set the matrix from a vector input
117 : inline void setFromVector( const std::vector<T>& vecin ) {
118 : plumed_assert( vecin.size()==sz );
119 : for(unsigned i=0; i<sz; ++i) {
120 : data[i]=vecin[i];
121 : }
122 : }
123 : /// Return element i,j of the matrix
124 : inline const T& operator () (const unsigned& i, const unsigned& j) const {
125 2015393449 : return data[j+i*cl];
126 : }
127 : /// Return a referenre to element i,j of the matrix
128 : inline T& operator () (const unsigned& i, const unsigned& j) {
129 605118188 : return data[j+i*cl];
130 : }
131 : /// Set all elements of the matrix equal to the value of v
132 : Matrix<T>& operator=(const T& v) {
133 4829318 : for(unsigned i=0; i<sz; ++i) {
134 4811759 : data[i]=v;
135 : }
136 : return *this;
137 : }
138 : /// Set the Matrix equal to another Matrix
139 : Matrix<T>& operator=(const Matrix<T>& m) {
140 56904 : sz=m.sz;
141 56904 : rw=m.rw;
142 56904 : cl=m.cl;
143 56904 : data=m.data;
144 56904 : return *this;
145 : }
146 : /// Set the Matrix equal to the value of a standard vector - used for readin
147 : Matrix<T>& operator=(const std::vector<T>& v) {
148 : plumed_dbg_assert( v.size()==sz );
149 : for(unsigned i=0; i<sz; ++i) {
150 : data[i]=v[i];
151 : }
152 : return *this;
153 : }
154 : /// Add v to all elements of the Matrix
155 : Matrix<T> operator+=(const T& v) {
156 : for(unsigned i=0; i<sz; ++i) {
157 : data[i]+=v;
158 : }
159 : return *this;
160 : }
161 : /// Multiply all elements by v
162 125 : Matrix<T> operator*=(const T& v) {
163 55038 : for(unsigned i=0; i<sz; ++i) {
164 54913 : data[i]*=v;
165 : }
166 125 : return *this;
167 : }
168 : /// Matrix addition
169 : Matrix<T>& operator+=(const Matrix<T>& m) {
170 : plumed_dbg_assert( m.rw==rw && m.cl==cl );
171 : data+=m.data;
172 : return *this;
173 : }
174 : /// Subtract v from all elements of the Matrix
175 : Matrix<T> operator-=(const T& v) {
176 : for(unsigned i=0; i<sz; ++i) {
177 : data-=v;
178 : }
179 : return *this;
180 : }
181 : /// Matrix subtraction
182 : Matrix<T>& operator-=(const Matrix<T>& m) {
183 : plumed_dbg_assert( m.rw==rw && m.cl==cl );
184 : data-=m.data;
185 : return *this;
186 : }
187 : /// Test if the matrix is symmetric or not
188 40203 : unsigned isSymmetric() const {
189 40203 : if (rw!=cl) {
190 : return 0;
191 : }
192 : unsigned sym=1;
193 81477 : for(unsigned i=1; i<rw; ++i)
194 251075 : for(unsigned j=0; j<i; ++j)
195 226313 : if( std::fabs(data[i+j*cl]-data[j+i*cl])>1.e-10 ) {
196 : sym=0;
197 : break;
198 : }
199 : return sym;
200 : }
201 : };
202 :
203 : /// Multiply matrix by scalar
204 : template <typename T> Matrix<T> operator*(T& v, const Matrix<T>& m ) {
205 : Matrix<T> new_m(m);
206 : new_m*=v;
207 : return new_m;
208 : }
209 :
210 16490 : template <typename T> void mult( const Matrix<T>& A, const Matrix<T>& B, Matrix<T>& C ) {
211 16490 : plumed_assert(A.cl==B.rw);
212 16490 : if( A.rw !=C.rw || B.cl !=C.cl ) {
213 0 : C.resize( A.rw, B.cl );
214 : }
215 : C=static_cast<T>( 0 );
216 96121 : for(unsigned i=0; i<A.rw; ++i)
217 4260408 : for(unsigned j=0; j<B.cl; ++j)
218 2012765358 : for (unsigned k=0; k<A.cl; ++k) {
219 2008584581 : C(i,j)+=A(i,k)*B(k,j);
220 : }
221 16490 : }
222 :
223 18594 : template <typename T> void mult( const Matrix<T>& A, const std::vector<T>& B, std::vector<T>& C) {
224 18594 : plumed_assert( A.cl==B.size() );
225 18594 : if( C.size()!=A.rw ) {
226 94 : C.resize(A.rw);
227 : }
228 86682 : for(unsigned i=0; i<A.rw; ++i) {
229 68088 : C[i]= static_cast<T>( 0 );
230 : }
231 86682 : for(unsigned i=0; i<A.rw; ++i)
232 778036 : for(unsigned k=0; k<A.cl; ++k) {
233 709948 : C[i]+=A(i,k)*B[k] ;
234 : }
235 18594 : }
236 :
237 : template <typename T> void mult( const std::vector<T>& A, const Matrix<T>& B, std::vector<T>& C) {
238 : plumed_assert( B.rw==A.size() );
239 : if( C.size()!=B.cl ) {
240 : C.resize( B.cl );
241 : }
242 : for(unsigned i=0; i<B.cl; ++i) {
243 : C[i]=static_cast<T>( 0 );
244 : }
245 : for(unsigned i=0; i<B.cl; ++i)
246 : for(unsigned k=0; k<B.rw; ++k) {
247 : C[i]+=A[k]*B(k,i);
248 : }
249 : }
250 :
251 730 : template <typename T> void transpose( const Matrix<T>& A, Matrix<T>& AT ) {
252 730 : if( A.rw!=AT.cl || A.cl!=AT.rw ) {
253 0 : AT.resize( A.cl, A.rw );
254 : }
255 2920 : for(unsigned i=0; i<A.cl; ++i)
256 52260 : for(unsigned j=0; j<A.rw; ++j) {
257 50070 : AT(i,j)=A(j,i);
258 : }
259 730 : }
260 :
261 : template <typename T> Log& operator<<(Log& ostr, const Matrix<T>& mat) {
262 : for(unsigned i=0; i<mat.sz; ++i) {
263 : ostr<<mat.data[i]<<" ";
264 : }
265 : return ostr;
266 : }
267 :
268 : template <typename T> void matrixOut( Log& ostr, const Matrix<T>& mat) {
269 : for(unsigned i=0; i<mat.rw; ++i) {
270 : for(unsigned j=0; j<mat.cl; ++j) {
271 : ostr<<mat(i,j)<<" ";
272 : }
273 : ostr<<"\n";
274 : }
275 : return;
276 : }
277 :
278 14055 : template <typename T> int diagMat( const Matrix<T>& A, std::vector<double>& eigenvals, Matrix<double>& eigenvecs ) {
279 :
280 : // Check matrix is square and symmetric
281 14055 : plumed_assert( A.rw==A.cl );
282 14055 : plumed_assert( A.isSymmetric()==1 );
283 14055 : std::vector<double> da(A.sz);
284 : unsigned k=0;
285 14055 : std::vector<double> evals(A.cl);
286 : // Transfer the matrix to the local array
287 43814 : for (unsigned i=0; i<A.cl; ++i)
288 460998 : for (unsigned j=0; j<A.rw; ++j) {
289 431239 : da[k++]=static_cast<double>( A(j,i) );
290 : }
291 :
292 14055 : int n=A.cl;
293 14055 : int lwork=-1, liwork=-1, m, info, one=1;
294 14055 : std::vector<double> work(A.cl);
295 14055 : std::vector<int> iwork(A.cl);
296 14055 : double vl, vu, abstol=0.0;
297 14055 : std::vector<int> isup(2*A.cl);
298 14055 : std::vector<double> evecs(A.sz);
299 :
300 14055 : plumed_lapack_dsyevr("V", "I", "U", &n, da.data(), &n, &vl, &vu, &one, &n,
301 : &abstol, &m, evals.data(), evecs.data(), &n,
302 : isup.data(), work.data(), &lwork, iwork.data(), &liwork, &info);
303 14055 : if (info!=0) {
304 : return info;
305 : }
306 :
307 : // Retrieve correct sizes for work and iwork then reallocate
308 14055 : liwork=iwork[0];
309 14055 : iwork.resize(liwork);
310 14055 : lwork=static_cast<int>( work[0] );
311 14055 : work.resize(lwork);
312 :
313 14055 : plumed_lapack_dsyevr("V", "I", "U", &n, da.data(), &n, &vl, &vu, &one, &n,
314 : &abstol, &m, evals.data(), evecs.data(), &n,
315 : isup.data(), work.data(), &lwork, iwork.data(), &liwork, &info);
316 14055 : if (info!=0) {
317 : return info;
318 : }
319 :
320 14055 : if( eigenvals.size()!=A.cl ) {
321 0 : eigenvals.resize( A.cl );
322 : }
323 14055 : if( eigenvecs.rw!=A.rw || eigenvecs.cl!=A.cl ) {
324 0 : eigenvecs.resize( A.rw, A.cl );
325 : }
326 : k=0;
327 43814 : for(unsigned i=0; i<A.cl; ++i) {
328 29759 : eigenvals[i]=evals[i];
329 : // N.B. For ease of producing projectors we store the eigenvectors
330 : // ROW-WISE in the eigenvectors matrix. The first index is the
331 : // eigenvector number and the second the component
332 460998 : for(unsigned j=0; j<A.rw; ++j) {
333 431239 : eigenvecs(i,j)=evecs[k++];
334 : }
335 : }
336 :
337 : // This changes eigenvectors so that the first non-null element
338 : // of each of them is positive
339 : // We can do it because the phase is arbitrary, and helps making
340 : // the result reproducible
341 43814 : for(int i=0; i<n; ++i) {
342 : int j;
343 30984 : for(j=0; j<n; j++)
344 30984 : if(eigenvecs(i,j)*eigenvecs(i,j)>1e-14) {
345 : break;
346 : }
347 29759 : if(j<n)
348 29759 : if(eigenvecs(i,j)<0.0)
349 192549 : for(j=0; j<n; j++) {
350 187295 : eigenvecs(i,j)*=-1;
351 : }
352 : }
353 : return 0;
354 : }
355 :
356 1 : template <typename T> int pseudoInvert( const Matrix<T>& A, Matrix<double>& pseudoinverse ) {
357 1 : std::vector<double> da(A.sz);
358 : unsigned k=0;
359 : // Transfer the matrix to the local array
360 501 : for (unsigned i=0; i<A.cl; ++i)
361 250500 : for (unsigned j=0; j<A.rw; ++j) {
362 250000 : da[k++]=static_cast<double>( A(j,i) );
363 : }
364 :
365 1 : int nsv, info, nrows=A.rw, ncols=A.cl;
366 1 : if(A.rw>A.cl) {
367 : nsv=A.cl;
368 : } else {
369 : nsv=A.rw;
370 : }
371 :
372 : // Create some containers for stuff from single value decomposition
373 1 : std::vector<double> S(nsv);
374 1 : std::vector<double> U(nrows*nrows);
375 1 : std::vector<double> VT(ncols*ncols);
376 1 : std::vector<int> iwork(8*nsv);
377 :
378 : // This optimizes the size of the work array used in lapack singular value decomposition
379 1 : int lwork=-1;
380 1 : std::vector<double> work(1);
381 1 : plumed_lapack_dgesdd( "A", &nrows, &ncols, da.data(), &nrows, S.data(), U.data(), &nrows, VT.data(), &ncols, work.data(), &lwork, iwork.data(), &info );
382 1 : if(info!=0) {
383 : return info;
384 : }
385 :
386 : // Retrieve correct sizes for work and rellocate
387 1 : lwork=(int) work[0];
388 1 : work.resize(lwork);
389 :
390 : // This does the singular value decomposition
391 1 : plumed_lapack_dgesdd( "A", &nrows, &ncols, da.data(), &nrows, S.data(), U.data(), &nrows, VT.data(), &ncols, work.data(), &lwork, iwork.data(), &info );
392 1 : if(info!=0) {
393 : return info;
394 : }
395 :
396 : // Compute the tolerance on the singular values ( machine epsilon * number of singular values * maximum singular value )
397 : double tol;
398 1 : tol=S[0];
399 500 : for(int i=1; i<nsv; ++i) {
400 499 : if( S[i]>tol ) {
401 : tol=S[i];
402 : }
403 : }
404 1 : tol*=nsv*epsilon;
405 :
406 : // Get the inverses of the singlular values
407 1 : Matrix<double> Si( ncols, nrows );
408 : Si=0.0;
409 501 : for(int i=0; i<nsv; ++i) {
410 500 : if( S[i]>tol ) {
411 499 : Si(i,i)=1./S[i];
412 : } else {
413 1 : Si(i,i)=0.0;
414 : }
415 : }
416 :
417 : // Now extract matrices for pseudoinverse
418 1 : Matrix<double> V( ncols, ncols ), UT( nrows, nrows ), tmp( ncols, nrows );
419 : k=0;
420 501 : for(int i=0; i<nrows; ++i) {
421 250500 : for(int j=0; j<nrows; ++j) {
422 250000 : UT(i,j)=U[k++];
423 : }
424 : }
425 : k=0;
426 501 : for(int i=0; i<ncols; ++i) {
427 250500 : for(int j=0; j<ncols; ++j) {
428 250000 : V(i,j)=VT[k++];
429 : }
430 : }
431 :
432 : // And do matrix algebra to construct the pseudoinverse
433 1 : if( pseudoinverse.rw!=ncols || pseudoinverse.cl!=nrows ) {
434 0 : pseudoinverse.resize( ncols, nrows );
435 : }
436 1 : mult( V, Si, tmp );
437 1 : mult( tmp, UT, pseudoinverse );
438 :
439 : return 0;
440 : }
441 :
442 25702 : template <typename T> int Invert( const Matrix<T>& A, Matrix<double>& inverse ) {
443 :
444 25702 : if( A.isSymmetric()==1 ) {
445 : // GAT -- I only ever use symmetric matrices so I can invert them like this.
446 : // I choose to do this as I have had problems with the more general way of doing this that
447 : // is implemented below.
448 9191 : std::vector<double> eval(A.rw);
449 9191 : Matrix<double> evec(A.rw,A.cl), tevec(A.rw,A.cl);
450 : int err;
451 9191 : err=diagMat( A, eval, evec );
452 9191 : if(err!=0) {
453 : return err;
454 : }
455 27283 : for (unsigned i=0; i<A.rw; ++i)
456 53988 : for (unsigned j=0; j<A.cl; ++j) {
457 35896 : tevec(i,j)=evec(j,i)/eval[j];
458 : }
459 9191 : mult(tevec,evec,inverse);
460 : } else {
461 16511 : std::vector<double> da(A.sz);
462 16511 : std::vector<int> ipiv(A.cl);
463 : unsigned k=0;
464 16511 : int n=A.rw, info;
465 49536 : for(unsigned i=0; i<A.cl; ++i)
466 99090 : for(unsigned j=0; j<A.rw; ++j) {
467 66065 : da[k++]=static_cast<double>( A(j,i) );
468 : }
469 :
470 16511 : plumed_lapack_dgetrf(&n,&n,da.data(),&n,ipiv.data(),&info);
471 16511 : if(info!=0) {
472 : return info;
473 : }
474 :
475 16511 : int lwork=-1;
476 16511 : std::vector<double> work(A.cl);
477 16511 : plumed_lapack_dgetri(&n,da.data(),&n,ipiv.data(),work.data(),&lwork,&info);
478 16511 : if(info!=0) {
479 : return info;
480 : }
481 :
482 16511 : lwork=static_cast<int>( work[0] );
483 16511 : work.resize(lwork);
484 16511 : plumed_lapack_dgetri(&n,da.data(),&n,ipiv.data(),work.data(),&lwork,&info);
485 16511 : if(info!=0) {
486 : return info;
487 : }
488 :
489 16511 : if( inverse.cl!=A.cl || inverse.rw!=A.rw ) {
490 0 : inverse.resize(A.rw,A.cl);
491 : }
492 : k=0;
493 49536 : for(unsigned i=0; i<A.rw; ++i)
494 99090 : for(unsigned j=0; j<A.cl; ++j) {
495 66065 : inverse(j,i)=da[k++];
496 : }
497 : }
498 :
499 : return 0;
500 : }
501 :
502 446 : template <typename T> void cholesky( const Matrix<T>& A, Matrix<T>& B ) {
503 :
504 446 : plumed_assert( A.rw==A.cl && A.isSymmetric() );
505 : Matrix<T> L(A.rw,A.cl);
506 : L=0.;
507 446 : std::vector<T> D(A.rw,0.);
508 1047 : for(unsigned i=0; i<A.rw; ++i) {
509 601 : L(i,i)=static_cast<T>( 1 );
510 756 : for (unsigned j=0; j<i; ++j) {
511 155 : L(i,j)=A(i,j);
512 155 : for (unsigned k=0; k<j; ++k) {
513 0 : L(i,j)-=L(i,k)*L(j,k)*D[k];
514 : }
515 155 : if (D[j]!=0.) {
516 155 : L(i,j)/=D[j];
517 : } else {
518 0 : L(i,j)=static_cast<T>( 0 );
519 : }
520 : }
521 601 : D[i]=A(i,i);
522 756 : for (unsigned k=0; k<i; ++k) {
523 155 : D[i]-=L(i,k)*L(i,k)*D[k];
524 : }
525 : }
526 :
527 1047 : for(unsigned i=0; i<A.rw; ++i) {
528 601 : D[i]=(D[i]>0.?std::sqrt(D[i]):0.);
529 : }
530 446 : if( B.rw!=A.rw || B.cl!=A.cl ) {
531 0 : B.resize( A.rw, A.cl);
532 : }
533 : B=0.;
534 1047 : for(unsigned i=0; i<A.rw; ++i)
535 1357 : for(unsigned j=0; j<=i; ++j) {
536 756 : B(i,j)+=L(i,j)*D[j];
537 : }
538 446 : }
539 :
540 : template <typename T> void chol_elsolve( const Matrix<T>& M, const std::vector<T>& b, std::vector<T>& y ) {
541 :
542 : plumed_assert( M.rw==M.cl && M(0,1)==0.0 && b.size()==M.rw );
543 : if( y.size()!=M.rw ) {
544 : y.resize( M.rw );
545 : }
546 : for(unsigned i=0; i<M.rw; ++i) {
547 : y[i]=b[i];
548 : for(unsigned j=0; j<i; ++j) {
549 : y[i]-=M(i,j)*y[j];
550 : }
551 : y[i]*=1.0/M(i,i);
552 : }
553 : }
554 :
555 0 : template <typename T> int logdet( const Matrix<T>& M, double& ldet ) {
556 : // Check matrix is square and symmetric
557 0 : plumed_assert( M.rw==M.cl || M.isSymmetric() );
558 :
559 0 : std::vector<double> da(M.sz);
560 : unsigned k=0;
561 0 : std::vector<double> evals(M.cl);
562 : // Transfer the matrix to the local array
563 0 : for (unsigned i=0; i<M.rw; ++i)
564 0 : for (unsigned j=0; j<M.cl; ++j) {
565 0 : da[k++]=static_cast<double>( M(j,i) );
566 : }
567 :
568 0 : int n=M.cl;
569 0 : int lwork=-1, liwork=-1, info, m, one=1;
570 0 : std::vector<double> work(M.rw);
571 0 : std::vector<int> iwork(M.rw);
572 0 : double vl, vu, abstol=0.0;
573 0 : std::vector<int> isup(2*M.rw);
574 0 : std::vector<double> evecs(M.sz);
575 0 : plumed_lapack_dsyevr("N", "I", "U", &n, da.data(), &n, &vl, &vu, &one, &n,
576 : &abstol, &m, evals.data(), evecs.data(), &n,
577 : isup.data(), work.data(), &lwork, iwork.data(), &liwork, &info);
578 0 : if (info!=0) {
579 : return info;
580 : }
581 :
582 : // Retrieve correct sizes for work and iwork then reallocate
583 0 : lwork=static_cast<int>( work[0] );
584 0 : work.resize(lwork);
585 0 : liwork=iwork[0];
586 0 : iwork.resize(liwork);
587 :
588 0 : plumed_lapack_dsyevr("N", "I", "U", &n, da.data(), &n, &vl, &vu, &one, &n,
589 : &abstol, &m, evals.data(), evecs.data(), &n,
590 : isup.data(), work.data(), &lwork, iwork.data(), &liwork, &info);
591 0 : if (info!=0) {
592 : return info;
593 : }
594 :
595 : // Transfer the eigenvalues and eigenvectors to the output
596 0 : ldet=0;
597 0 : for(unsigned i=0; i<M.cl; i++) {
598 0 : ldet+=log(evals[i]);
599 : }
600 :
601 : return 0;
602 : }
603 :
604 :
605 :
606 : }
607 : #endif
|