From ac4f2d9798a20a8152cfa8f3da77656256ea2bfc Mon Sep 17 00:00:00 2001 From: Christopher Kelly Date: Wed, 9 Jun 2021 09:08:37 -0400 Subject: [PATCH] Fixed EOFA approx test square rooting the result inappropriately thus failing when it shouldn't To MDWF+ID GPBC evol main program, added routine to compute the lower bound of the EOFA using the power method with a command line toggle --- .../pseudofermion/ExactOneFlavourRatio.h | 9 +++- HMC/Mobius2p1fIDSDRGparityEOFA.cc | 41 +++++++++++++++++-- 2 files changed, 44 insertions(+), 6 deletions(-) diff --git a/Grid/qcd/action/pseudofermion/ExactOneFlavourRatio.h b/Grid/qcd/action/pseudofermion/ExactOneFlavourRatio.h index 622ebc21..f3bb4f1b 100644 --- a/Grid/qcd/action/pseudofermion/ExactOneFlavourRatio.h +++ b/Grid/qcd/action/pseudofermion/ExactOneFlavourRatio.h @@ -394,10 +394,15 @@ NAMESPACE_BEGIN(Grid); //S_init = eta^dag M^{-1/2} M M^{-1/2} eta //S_init - eta^dag eta = eta^dag ( M^{-1/2} M M^{-1/2} - 1 ) eta - RealD test = sqrt(fabs(diff)/norm2_eta); //test the quality of the rational approx + + //If approximate solution + //S_init - eta^dag eta = eta^dag ( [M^{-1/2}+\delta M^{-1/2}] M [M^{-1/2}+\delta M^{-1/2}] - 1 ) eta + // \approx eta^dag ( \delta M^{-1/2} M^{1/2} + M^{1/2}\delta M^{-1/2} ) eta + // We divide out |eta|^2 to remove source scaling but the tolerance on this check should still be somewhat higher than the actual approx tolerance + RealD test = fabs(diff)/norm2_eta; //test the quality of the rational approx std::cout << GridLogMessage << action_name() << " initial action " << action << " expect " << norm2_eta << "; diff " << diff << std::endl; - std::cout << GridLogMessage << action_name() << " sqrt( eta^dag ( M^{-1/2} M M^{-1/2} - 1 ) eta )/sqrt( eta^dag eta ) = " << test << " expect 0 (tol " << param.BoundsCheckTol << ")" << std::endl; + std::cout << GridLogMessage << action_name() << "[ eta^dag ( M^{-1/2} M M^{-1/2} - 1 ) eta ]/|eta^2| = " << test << " expect 0 (tol " << param.BoundsCheckTol << ")" << std::endl; assert( ( test < param.BoundsCheckTol ) && " Initial action check failed" ); initial_action = false; diff --git a/HMC/Mobius2p1fIDSDRGparityEOFA.cc b/HMC/Mobius2p1fIDSDRGparityEOFA.cc index c818eb4f..f86dee4d 100644 --- a/HMC/Mobius2p1fIDSDRGparityEOFA.cc +++ b/HMC/Mobius2p1fIDSDRGparityEOFA.cc @@ -312,9 +312,37 @@ void upperBoundEOFA(ExactOneFlavourRatioPseudoFermionAction & auto lambda_max = power_method(linop,eta); std::cout << GridLogMessage << "Upper bound of EOFA operator " << lambda_max << std::endl; } - +//Applications of M^{-1} cost the same as M for EOFA! +template +class EOFAinvLinop: public LinearOperatorBase{ + ExactOneFlavourRatioPseudoFermionAction &EOFA; + LatticeGaugeFieldD &U; +public: + EOFAinvLinop(ExactOneFlavourRatioPseudoFermionAction &EOFA, LatticeGaugeFieldD &U): EOFA(EOFA), U(U){} + typedef typename FermionImplPolicy::FermionField Field; + void OpDiag (const Field &in, Field &out){ assert(0); } + void OpDir (const Field &in, Field &out,int dir,int disp){ assert(0); } + void OpDirAll (const Field &in, std::vector &out){ assert(0); } + + void Op (const Field &in, Field &out){ assert(0); } + void AdjOp (const Field &in, Field &out){ assert(0); } + void HermOpAndNorm(const Field &in, Field &out,RealD &n1,RealD &n2){ assert(0); } + void HermOp(const Field &in, Field &out){ EOFA.MeofaInv(U, in, out); } +}; + +template +void lowerBoundEOFA(ExactOneFlavourRatioPseudoFermionAction &EOFA, + GridCartesian* FGrid, GridParallelRNG &rng, LatticeGaugeFieldD &latt){ + std::cout << GridLogMessage << "Starting EOFA lower bound compute using power method on M^{-1}. Inverse of highest eigenvalue is the lowest eigenvalue of M" << std::endl; + EOFAinvLinop linop(EOFA, latt); + typename FermionImplPolicy::FermionField eta(FGrid); + gaussian(rng,eta); + PowerMethod power_method; + auto lambda_max = power_method(linop,eta); + std::cout << GridLogMessage << "Lower bound of EOFA operator " << 1./lambda_max << std::endl; +} NAMESPACE_BEGIN(Grid); @@ -667,9 +695,12 @@ int main(int argc, char **argv) { //Action tuning - bool tune_rhmc_s=false, eigenrange_s=false, + bool + tune_rhmc_s=false, eigenrange_s=false, tune_rhmc_DSDR=false, eigenrange_DSDR=false, - check_eofa=false, upper_bound_eofa=false; + check_eofa=false, + upper_bound_eofa=false, lower_bound_eofa(false); + std::string lanc_params_s; std::string lanc_params_DSDR; int tune_rhmc_s_action_or_md; @@ -699,8 +730,9 @@ int main(int argc, char **argv) { } else if(sarg == "--check_eofa") check_eofa = true; else if(sarg == "--upper_bound_eofa") upper_bound_eofa = true; + else if(sarg == "--lower_bound_eofa") lower_bound_eofa = true; } - if(tune_rhmc_s || eigenrange_s || tune_rhmc_DSDR || eigenrange_DSDR ||check_eofa || upper_bound_eofa) { + if(tune_rhmc_s || eigenrange_s || tune_rhmc_DSDR || eigenrange_DSDR ||check_eofa || upper_bound_eofa || lower_bound_eofa) { std::cout << GridLogMessage << "Running checks" << std::endl; TheHMC.initializeGaugeFieldAndRNGs(Ud); @@ -710,6 +742,7 @@ int main(int argc, char **argv) { if(check_eofa) checkEOFA(EOFA, FGridD, TheHMC.Resources.GetParallelRNG(), Ud); if(upper_bound_eofa) upperBoundEOFA(EOFA, FGridD, TheHMC.Resources.GetParallelRNG(), Ud); + if(lower_bound_eofa) lowerBoundEOFA(EOFA, FGridD, TheHMC.Resources.GetParallelRNG(), Ud); if(eigenrange_s) computeEigenvalues(lanc_params_s, FGridD, FrbGridD, Ud, Numerator_sD, TheHMC.Resources.GetParallelRNG()); if(tune_rhmc_s) checkRHMC(FGridD, FrbGridD, Ud, Numerator_sD, Denominator_sD, Quotient_s, TheHMC.Resources.GetParallelRNG(), 4, "strange", tune_rhmc_s_action_or_md); if(eigenrange_DSDR) computeEigenvalues(lanc_params_DSDR, UGridD, UrbGridD, Ud, Numerator_DSDR_D, TheHMC.Resources.GetParallelRNG());