Subversion Repositories slepc-dev

Rev

Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
  SLEPc eigensolver: "davidson"

  Step: test for convergence

   - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
   SLEPc - Scalable Library for Eigenvalue Problem Computations
   Copyright (c) 2002-2009, Universidad Politecnica de Valencia, Spain

   This file is part of SLEPc.
     
   SLEPc is free software: you can redistribute it and/or modify it under  the
   terms of version 3 of the GNU Lesser General Public License as published by
   the Free Software Foundation.

   SLEPc  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 SLEPc. If not, see <http://www.gnu.org/licenses/>.
   - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*/


#include "davidson.h"

PetscTruth dvd_testconv_basic_0(dvdDashboard *d, PetscScalar eigvr,
                                PetscScalar eigvi, PetscReal r,
                                PetscReal *err);
PetscTruth dvd_testconv_slepc_0(dvdDashboard *d, PetscScalar eigvr,
                                PetscScalar eigvi, PetscReal r,
                                PetscReal *err);

#undef __FUNCT__  
#define __FUNCT__ "dvd_testconv_basic"
PetscErrorCode dvd_testconv_basic(dvdDashboard *d, dvdBlackboard *b)
{
  PetscErrorCode  ierr;

  PetscFunctionBegin;

  /* Setup the step */
  if (b->state >= DVD_STATE_CONF) {
    if (d->testConv_data) {
      ierr = PetscFree(d->testConv_data); CHKERRQ(ierr);
    }
    d->testConv_data = PETSC_NULL;
    d->testConv = dvd_testconv_basic_0;
  }

  PetscFunctionReturn(0);
}

#undef __FUNCT__  
#define __FUNCT__ "dvd_testconv_basic_0"
PetscTruth dvd_testconv_basic_0(dvdDashboard *d, PetscScalar eigvr,
                                PetscScalar eigvi, PetscReal r,
                                PetscReal *err)
{
  PetscTruth      conv;
  PetscReal       eig_norm, errest;

  PetscFunctionBegin;

  eig_norm = SlepcAbsEigenvalue(eigvr, eigvi);
  //errest = (r < eig_norm) ? r/eig_norm : r;
  errest = r/eig_norm;
  conv = (errest <= d->tol) ? PETSC_TRUE : PETSC_FALSE;
  if (err) *err = errest;

  PetscFunctionReturn(conv);
}

#undef __FUNCT__  
#define __FUNCT__ "dvd_testconv_slepc"
PetscErrorCode dvd_testconv_slepc(dvdDashboard *d, dvdBlackboard *b)
{
  PetscErrorCode  ierr;

  PetscFunctionBegin;

  /* Setup the step */
  if (b->state >= DVD_STATE_CONF) {
    if (d->testConv_data) {
      ierr = PetscFree(d->testConv_data); CHKERRQ(ierr);
    }
    d->testConv_data = PETSC_NULL;
    d->testConv = dvd_testconv_slepc_0;
  }

  PetscFunctionReturn(0);
}

#undef __FUNCT__  
#define __FUNCT__ "dvd_testconv_slepc_0"
PetscTruth dvd_testconv_slepc_0(dvdDashboard *d, PetscScalar eigvr,
                                PetscScalar eigvi, PetscReal r,
                                PetscReal *err)
{
  PetscErrorCode  ierr;

  PetscFunctionBegin;

  ierr = (*d->eps->conv_func)(d->eps, eigvr, eigvi, r, err,
                              d->eps->conv_ctx);
  CHKERRABORT(((PetscObject)d->eps)->comm, ierr);

  PetscFunctionReturn(*err<d->eps->tol);
}