/usr/include/trilinos/AnasaziStatusTestWithOrdering.hpp is in libtrilinos-dev 10.4.0.dfsg-1ubuntu2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 | // @HEADER
// ***********************************************************************
//
// Anasazi: Block Eigensolvers Package
// Copyright (2004) Sandia Corporation
//
// Under terms of Contract DE-AC04-94AL85000, there is a non-exclusive
// license for use of this work by or on behalf of the U.S. Government.
//
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation; either version 2.1 of the
// License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
// USA
// Questions? Contact Michael A. Heroux (maherou@sandia.gov)
//
// ***********************************************************************
// @HEADER
//
#ifndef ANASAZI_STATUS_TEST_ORDEREDRESNORM_HPP
#define ANASAZI_STATUS_TEST_ORDEREDRESNORM_HPP
/*!
\file AnasaziStatusTestWithOrdering.hpp
\brief A status test for testing the norm of the eigenvectors residuals along with a
set of auxiliary eigenvalues.
*/
#include "AnasaziStatusTest.hpp"
#include "Teuchos_ScalarTraits.hpp"
#include "Teuchos_LAPACK.hpp"
/*!
\class Anasazi::StatusTestWithOrdering
\brief A status test for testing the norm of the eigenvectors residuals
along with a set of auxiliary eigenvalues.
The test evaluates to ::Passed when then the most significant of the
eigenvalues all have a residual below a certain threshhold. The purpose of
the test is to not only test convergence for some number of eigenvalues,
but to test convergence for the correct ones.
In addition to specifying the tolerance, the user may specify:
<ul>
<li> the norm to be used: 2-norm or OrthoManager::norm() or getRitzRes2Norms()
<li> the scale: absolute or relative to magnitude of Ritz value
<li> the quorum: the number of vectors required for the test to
evaluate as ::Passed.
</ul>
Finally, the user must specify the Anasazi::SortManager used for deciding
significance.
*/
namespace Anasazi {
template <class ScalarType, class MV, class OP>
class StatusTestWithOrdering : public StatusTest<ScalarType,MV,OP> {
private:
typedef typename Teuchos::ScalarTraits<ScalarType>::magnitudeType MagnitudeType;
typedef Teuchos::ScalarTraits<MagnitudeType> MT;
public:
//! @name Constructors/destructors
//@{
//! Constructor
StatusTestWithOrdering(Teuchos::RCP<StatusTest<ScalarType,MV,OP> > test, Teuchos::RCP<SortManager<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> > sorter, int quorum = -1);
//! Destructor
virtual ~StatusTestWithOrdering() {};
//@}
//! @name Status methods
//@{
/*! Check status as defined by test.
\return TestStatus indicating whether the test passed or failed.
*/
TestStatus checkStatus( Eigensolver<ScalarType,MV,OP>* solver );
//! Return the result of the most recent checkStatus call, or undefined if it has not been run.
TestStatus getStatus() const { return state_; }
//! Get the indices for the vectors that passed the test.
/*!
* Non-negative indices correspond to passing vectors from the constituent status test.
* Negative entries correspond to auxilliary values, where the first auxilliary value
* is indexed by -NumAuxVals, the second by -NumAuxVals+1, and so forth.
*/
std::vector<int> whichVecs() const {
return ind_;
}
//! Get the number of vectors that passed the test.
int howMany() const {
return ind_.size();
}
//@}
//! @name Accessor methods
//@{
/*! \brief Set quorum.
*
* Setting quorum to -1 signifies that all residuals from the solver must meet the tolerance.
* This also resets the test status to ::Undefined.
*/
void setQuorum(int quorum) {
state_ = Undefined;
quorum_ = quorum;
}
/*! \brief Get quorum.
*/
int getQuorum() const {
return quorum_;
}
//@}
//! @name Reset methods
//@{
//! Informs the status test that it should reset its internal configuration to the uninitialized state.
/*! This is necessary for the case when the status test is being reused by another solver or for another
eigenvalue problem. The status test may have information that pertains to a particular problem or solver
state. The internal information will be reset back to the uninitialized state. The user specified information
that the convergence test uses will remain.
*/
void reset() {
ind_.resize(0);
state_ = Undefined;
test_->reset();
}
//! Clears the results of the last status test.
/*! This should be distinguished from the reset() method, as it only clears the cached result from the last
* status test, so that a call to getStatus() will return ::Undefined. This is necessary for the SEQOR and SEQAND
* tests in the StatusTestCombo class, which may short circuit and not evaluate all of the StatusTests contained
* in them.
*/
void clearStatus() {
ind_.resize(0);
state_ = Undefined;
test_->clearStatus();
}
/*! \brief Set the auxiliary eigenvalues.
*
* This routine sets only the real part of the auxiliary eigenvalues; the imaginary part is set to zero. This routine also resets the state to ::Undefined.
*/
void setAuxVals(const std::vector<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> &vals) {
rvals_ = vals;
ivals_.resize(rvals_.size(),MT::zero());
state_ = Undefined;
}
/*! \brief Set the auxiliary eigenvalues.
*
* This routine sets both the real and imaginary parts of the auxiliary eigenvalues. This routine also resets the state to ::Undefined.
*/
void setAuxVals(const std::vector<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> &rvals, const std::vector<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> &ivals) {
rvals_ = rvals;
ivals_ = ivals;
state_ = Undefined;
}
/*! \brief Get the auxiliary eigenvalues.
*
* This routine gets the real and imaginary parts of the auxiliary eigenvalues.
*/
void getAuxVals(std::vector<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> &rvals, std::vector<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> &ivals) const {
rvals = rvals_;
ivals = ivals_;
}
//@}
//! @name Print methods
//@{
//! Output formatted description of stopping test to output stream.
std::ostream& print(std::ostream& os, int indent = 0) const;
//@}
private:
TestStatus state_;
std::vector<int> ind_;
int quorum_;
std::vector<MagnitudeType> rvals_, ivals_;
Teuchos::RCP<SortManager<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> > sorter_;
Teuchos::RCP<StatusTest<ScalarType,MV,OP> > test_;
};
template <class ScalarType, class MV, class OP>
StatusTestWithOrdering<ScalarType,MV,OP>::StatusTestWithOrdering(Teuchos::RCP<StatusTest<ScalarType,MV,OP> > test, Teuchos::RCP<SortManager<typename Teuchos::ScalarTraits<ScalarType>::magnitudeType> > sorter, int quorum)
: state_(Undefined), ind_(0), quorum_(quorum), rvals_(0), ivals_(0), sorter_(sorter), test_(test)
{
TEST_FOR_EXCEPTION(sorter_ == Teuchos::null, StatusTestError, "StatusTestWithOrdering::constructor() was passed null pointer for constituent SortManager.");
TEST_FOR_EXCEPTION(test_ == Teuchos::null, StatusTestError, "StatusTestWithOrdering::constructor() was passed null pointer for constituent StatusTest.");
}
template <class ScalarType, class MV, class OP>
TestStatus StatusTestWithOrdering<ScalarType,MV,OP>::checkStatus( Eigensolver<ScalarType,MV,OP>* solver ) {
// Algorithm
// we PASS iff the "most signficant" values of "all values" PASS
// "most significant" is measured by sorter
// auxilliary values are automatically PASSED
// constituent status test determines who else passes
// "all values" mean {auxilliary values} UNION {solver's ritz values}
//
// test_->checkStatus() calls the constituent status test
// cwhch = test_->whichVecs() gets the passing indices from the constituent test
// solval = solver->getRitzValues() gets the solver's ritz values
// allval = {solval auxval} contains all values
// sorter_->sort(allval,perm) sort all values (we just need the perm vector)
//
// allpass = {cwhch numsolval+1 ... numsolval+numAux}
// mostsig = {perm[1] ... perm[quorum]}
// whichVecs = mostsig INTERSECT allpass
// if size(whichVecs) >= quorum,
// PASS
// else
// FAIL
//
// finish: this needs to be better tested and revisited for efficiency.
// call the constituent solver manager
test_->checkStatus(solver);
std::vector<int> cwhch( test_->whichVecs() );
// get the ritzvalues from solver
std::vector<Value<ScalarType> > solval = solver->getRitzValues();
int numsolval = solval.size();
int numauxval = rvals_.size();
int numallval = numsolval + numauxval;
if (numallval == 0) {
ind_.resize(0);
return Failed;
}
// make space for all values, real and imaginary parts
std::vector<MagnitudeType> allvalr(numallval), allvali(numallval);
// separate the real and imaginary parts of solver ritz values
for (int i=0; i<numsolval; ++i) {
allvalr[i] = solval[i].realpart;
allvali[i] = solval[i].imagpart;
}
// put the auxiliary values at the end of the solver ritz values
std::copy(rvals_.begin(),rvals_.end(),allvalr.begin()+numsolval);
std::copy(ivals_.begin(),ivals_.end(),allvali.begin()+numsolval);
// sort all values
std::vector<int> perm(numallval);
sorter_->sort(allvalr,allvali,Teuchos::rcpFromRef(perm),numallval);
// make the set of passing values: allpass = {cwhch -1 ... -numauxval}
std::vector<int> allpass(cwhch.size() + numauxval);
std::copy(cwhch.begin(),cwhch.end(),allpass.begin());
for (int i=0; i<numauxval; i++) {
allpass[cwhch.size()+i] = -(i+1);
}
// make list, with length quorum, of most significant values, if there are that many
// int numsig = quorum_ < numallval ? quorum_ : numallval;
int numsig = cwhch.size() + numauxval;
std::vector<int> mostsig(numsig);
for (int i=0; i<numsig; ++i) {
mostsig[i] = perm[i];
// if perm[i] >= numsolval, then it corresponds to the perm[i]-numsolval aux val
// the first aux val gets index -numauxval, the seonc -numauxval+1, and so forth
if (mostsig[i] >= numsolval) {
mostsig[i] = mostsig[i]-numsolval-numauxval;
}
}
// who passed?
// to use set_intersection, ind_ must have room for the result, which will have size() <= min( allpass.size(), mostsig.size() )
// also, allpass and mostsig must be in ascending order; neither are
// lastly, the return from set_intersection points to the last element in the intersection, which tells us how big the intersection is
ind_.resize(numsig);
std::vector<int>::iterator end;
std::sort(mostsig.begin(),mostsig.end());
std::sort(allpass.begin(),allpass.end());
end = std::set_intersection(mostsig.begin(),mostsig.end(),allpass.begin(),allpass.end(),ind_.begin());
ind_.resize(end - ind_.begin());
// did we pass, overall
if (ind_.size() >= (unsigned int)quorum_) {
state_ = Passed;
}
else {
state_ = Failed;
}
return state_;
}
template <class ScalarType, class MV, class OP>
std::ostream& StatusTestWithOrdering<ScalarType,MV,OP>::print(std::ostream& os, int indent) const {
// build indent string
std::string ind(indent,' ');
// print header
os << ind << "- StatusTestWithOrdering: ";
switch (state_) {
case Passed:
os << "Passed" << std::endl;
break;
case Failed:
os << "Failed" << std::endl;
break;
case Undefined:
os << "Undefined" << std::endl;
break;
}
// print parameters, namely, quorum
os << ind << " Quorum: " << quorum_ << std::endl;
// print any auxilliary values
os << ind << " Auxiliary values: ";
if (rvals_.size() > 0) {
for (unsigned int i=0; i<rvals_.size(); i++) {
os << "(" << rvals_[i] << ", " << ivals_[i] << ") ";
}
os << std::endl;
}
else {
os << "[empty]" << std::endl;
}
// print which vectors have passed
if (state_ != Undefined) {
os << ind << " Which vectors: ";
if (ind_.size() > 0) {
for (unsigned int i=0; i<ind_.size(); i++) os << ind_[i] << " ";
os << std::endl;
}
else {
os << "[empty]" << std::endl;
}
}
// print constituent test
test_->print(os,indent+2);
return os;
}
} // end of Anasazi namespace
#endif /* ANASAZI_STATUS_TEST_ORDEREDRESNORM_HPP */
|