From b6c2efa666a4b678a40df4dde25512a8ca96282a Mon Sep 17 00:00:00 2001 From: Andrew Zhen Ning Yong Date: Tue, 7 Jul 2020 10:07:05 +0100 Subject: [PATCH 1/5] 1) fold now requires second argument: modelPar to properly fold correlators with cosh/sinh form. 2) makeSinhModel has the appropriate alternate sign s.t the prefactor remains positive. 3) parameterGuess uses correct analytic form for sinh-type correlator's init(1), corresponding to the prefactor. --- lib/Physics/CorrelatorFitter.cpp | 57 +++++++++++++++++++++++++++----- lib/Physics/CorrelatorFitter.hpp | 2 +- 2 files changed, 50 insertions(+), 9 deletions(-) diff --git a/lib/Physics/CorrelatorFitter.cpp b/lib/Physics/CorrelatorFitter.cpp index 7784f5d..5061a02 100644 --- a/lib/Physics/CorrelatorFitter.cpp +++ b/lib/Physics/CorrelatorFitter.cpp @@ -84,7 +84,7 @@ DoubleModel CorrelatorModels::makeSinhModel(const Index nState, const Index nt) for (unsigned int i = 0; i < nState; ++i) { - res += p[2*i + 1]*(exp(-p[2*i]*x[0]) - exp(-p[2*i]*(nt - x[0]))); + res += p[2*i + 1]*(- exp(-p[2*i]*x[0]) + exp(-p[2*i]*(nt - x[0]))); } return res; @@ -200,8 +200,6 @@ DVec CorrelatorModels::parameterGuess(const DMatSample &corr, LATAN_ERROR(Argument, "correlator type is undefined"); break; case CorrelatorType::exp: - case CorrelatorType::cosh: - case CorrelatorType::sinh: init.resize(2*par.nState); init(0) = log(corr[central](nt/4)/corr[central](nt/4 + 1)); init(1) = corr[central](nt/4)/(exp(-init(0)*nt/4)); @@ -211,6 +209,26 @@ DVec CorrelatorModels::parameterGuess(const DMatSample &corr, init(p + 1) = init(p - 1)/2.; } break; + case CorrelatorType::cosh: + init.resize(2*par.nState); + init(0) = log(corr[central](nt/4)/corr[central](nt/4 + 1)); + init(1) = corr[central](nt/4)/(exp(-init(0)*nt/4) + exp(-init(0)*3*nt/4)); + for (Index p = 2; p < init.size(); p += 2) + { + init(p) = 2*init(p - 2); + init(p + 1) = init(p - 1)/2.; + } + break; + case CorrelatorType::sinh: + init.resize(2*par.nState); + init(0) = log(corr[central](nt/4)/corr[central](nt/4 + 1)); + init(1) = corr[central](nt/4)/( -exp(-init(0)*nt/4) +exp(-init(0)*3*nt/4)); + for (Index p = 2; p < init.size(); p += 2) + { + init(p) = 2*init(p - 2); + init(p + 1) = init(p - 1)/2.; + } + break; case CorrelatorType::linear: init.resize(2); init(0) = corr[central](nt/4) - corr[central](nt/4 + 1, 0); @@ -253,16 +271,39 @@ DMatSample CorrelatorUtils::shift(const DMatSample &c, const Index ts) } } -DMatSample CorrelatorUtils::fold(const DMatSample &c) +DMatSample CorrelatorUtils::fold(const DMatSample &c, const CorrelatorModels::ModelPar &par) { const Index nt = c[central].rows(); DMatSample buf = c; - - FOR_STAT_ARRAY(buf, s) + int sign; + bool fold = false; + + switch (par.type) { - for (Index t = 0; t < nt; ++t) + case CorrelatorType::cosh: + case CorrelatorType::cst: + sign = 1; + fold = true; + break; + case CorrelatorType::sinh: + sign = -1; + fold = true; + break; + case CorrelatorType::linear: + cout << "Linear model is asymmetric: will not fold." << endl; + break; + default: + break; + } + + if (fold) + { + FOR_STAT_ARRAY(buf, s) { - buf[s](t) = 0.5*(c[s](t) + c[s]((nt - t) % nt)); + for (Index t = 0; t < nt; ++t) + { + buf[s](t) = 0.5*(c[s](t) + sign*c[s]((nt - t) % nt)); + } } } diff --git a/lib/Physics/CorrelatorFitter.hpp b/lib/Physics/CorrelatorFitter.hpp index 694fd9b..816c0ca 100644 --- a/lib/Physics/CorrelatorFitter.hpp +++ b/lib/Physics/CorrelatorFitter.hpp @@ -56,7 +56,7 @@ namespace CorrelatorModels namespace CorrelatorUtils { DMatSample shift(const DMatSample &c, const Index ts); - DMatSample fold(const DMatSample &c); + DMatSample fold(const DMatSample &c, const CorrelatorModels::ModelPar &par); DMatSample fourierTransform(const DMatSample &c, FFT &fft, const unsigned int dir = FFT::Forward); }; From 3e3cdf2d69dfd8b04d77f7604cce7999b6b41f75 Mon Sep 17 00:00:00 2001 From: Andrew Zhen Ning Yong Date: Tue, 7 Jul 2020 10:07:48 +0100 Subject: [PATCH 2/5] 2pt-fit.cpp now has correct number of arguments for fold function if --fold is called. --- physics/2pt-fit.cpp | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/physics/2pt-fit.cpp b/physics/2pt-fit.cpp index 2b19b4f..739f18f 100644 --- a/physics/2pt-fit.cpp +++ b/physics/2pt-fit.cpp @@ -110,10 +110,6 @@ int main(int argc, char *argv[]) nt = corr[central].rows(); corr = corr.block(0, 0, nt, 1); corr = CorrelatorUtils::shift(corr, shift); - if (fold) - { - corr = CorrelatorUtils::fold(corr); - } // make model ////////////////////////////////////////////////////////////// CorrelatorFitter fitter(corr); @@ -140,6 +136,11 @@ int main(int argc, char *argv[]) } } + if (fold) + { + corr = CorrelatorUtils::fold(corr,modelPar); + } + // fit ///////////////////////////////////////////////////////////////////// DVec init(nPar); NloptMinimizer globMin(NloptMinimizer::Algorithm::GN_CRS2_LM); From ddee922e72ed9375d37b3c39aa6d9de4a7f8feac Mon Sep 17 00:00:00 2001 From: Andrew Zhen Ning Yong Date: Fri, 10 Jul 2020 08:33:28 +0100 Subject: [PATCH 3/5] 1) restored original definition of sinh in makeSinhModel() and its init in parameterGuess(). 2) fold definition includes sign factor due to cosh/sinh model. --- lib/Physics/CorrelatorFitter.cpp | 22 ++-------------------- 1 file changed, 2 insertions(+), 20 deletions(-) diff --git a/lib/Physics/CorrelatorFitter.cpp b/lib/Physics/CorrelatorFitter.cpp index 5061a02..02cf5eb 100644 --- a/lib/Physics/CorrelatorFitter.cpp +++ b/lib/Physics/CorrelatorFitter.cpp @@ -84,7 +84,7 @@ DoubleModel CorrelatorModels::makeSinhModel(const Index nState, const Index nt) for (unsigned int i = 0; i < nState; ++i) { - res += p[2*i + 1]*(- exp(-p[2*i]*x[0]) + exp(-p[2*i]*(nt - x[0]))); + res += p[2*i + 1]*(exp(-p[2*i]*x[0]) - exp(-p[2*i]*(nt - x[0]))); } return res; @@ -200,29 +200,11 @@ DVec CorrelatorModels::parameterGuess(const DMatSample &corr, LATAN_ERROR(Argument, "correlator type is undefined"); break; case CorrelatorType::exp: - init.resize(2*par.nState); - init(0) = log(corr[central](nt/4)/corr[central](nt/4 + 1)); - init(1) = corr[central](nt/4)/(exp(-init(0)*nt/4)); - for (Index p = 2; p < init.size(); p += 2) - { - init(p) = 2*init(p - 2); - init(p + 1) = init(p - 1)/2.; - } - break; case CorrelatorType::cosh: - init.resize(2*par.nState); - init(0) = log(corr[central](nt/4)/corr[central](nt/4 + 1)); - init(1) = corr[central](nt/4)/(exp(-init(0)*nt/4) + exp(-init(0)*3*nt/4)); - for (Index p = 2; p < init.size(); p += 2) - { - init(p) = 2*init(p - 2); - init(p + 1) = init(p - 1)/2.; - } - break; case CorrelatorType::sinh: init.resize(2*par.nState); init(0) = log(corr[central](nt/4)/corr[central](nt/4 + 1)); - init(1) = corr[central](nt/4)/( -exp(-init(0)*nt/4) +exp(-init(0)*3*nt/4)); + init(1) = corr[central](nt/4)/(exp(-init(0)*nt/4)); for (Index p = 2; p < init.size(); p += 2) { init(p) = 2*init(p - 2); From 1b12f2ca2d5bd9cb6f55a10fd41536840cf26e2f Mon Sep 17 00:00:00 2001 From: Andrew Zhen Ning Yong Date: Fri, 13 Nov 2020 07:56:58 +0000 Subject: [PATCH 4/5] latan-sample-fake now takes option -r/--seed where seed can be specified if fake bootstrap samples with 100% correlation are needed. --- utils/sample-fake.cpp | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/utils/sample-fake.cpp b/utils/sample-fake.cpp index f34921b..5f1816c 100644 --- a/utils/sample-fake.cpp +++ b/utils/sample-fake.cpp @@ -18,30 +18,42 @@ */ #include +#include + using namespace std; using namespace Latan; int main(int argc, char *argv[]) { + OptParser opt; Index nSample; double val, err; string outFileName; - if (argc != 5) + opt.addOption("r", "seed" , OptParser::OptType::value, true, + "random generator seed (default: random)"); + opt.addOption("", "help" , OptParser::OptType::trigger, true, + "show this help message and exit"); + + bool parsed = opt.parse(argc, argv); + if (!parsed or (opt.getArgs().size() != 4) or opt.gotOption("help")) { cerr << "usage: " << argv[0]; cerr << " " << endl; + cerr << endl << "Possible options:" << endl << opt << endl; return EXIT_FAILURE; } + val = strTo(argv[1]); err = strTo(argv[2]); nSample = strTo(argv[3]); outFileName = argv[4]; random_device rd; - mt19937 gen(rd()); + SeedType seed = (opt.gotOption("r")) ? opt.optionValue("r") : rd(); + mt19937 gen(seed); normal_distribution<> dis(val, err); DSample res(nSample); @@ -59,4 +71,4 @@ int main(int argc, char *argv[]) Io::save(res, outFileName); return EXIT_SUCCESS; -} +} \ No newline at end of file From 9e8d53463581fb917bc506d6de73d561fc01b59b Mon Sep 17 00:00:00 2001 From: Andrew Zhen Ning Yong Date: Thu, 20 May 2021 11:04:21 +0100 Subject: [PATCH 5/5] New methods in XYSampleData class to skip fits that are non-converging/large chi2PerDof. This is tracked by boolean goodFit_. --- lib/Statistics/XYSampleData.cpp | 55 ++++++++++++++++++++++++--------- lib/Statistics/XYSampleData.hpp | 3 ++ 2 files changed, 43 insertions(+), 15 deletions(-) diff --git a/lib/Statistics/XYSampleData.cpp b/lib/Statistics/XYSampleData.cpp index 6e2d2d6..19342c9 100644 --- a/lib/Statistics/XYSampleData.cpp +++ b/lib/Statistics/XYSampleData.cpp @@ -234,6 +234,21 @@ DVec XYSampleData::getYError(const Index j) return data_.getYError(j); } +bool XYSampleData::checkFit() +{ + return goodFit_; +} + +void XYSampleData::checkChi2PerDof(double Chi2PerDof) +{ + if(Chi2PerDof >= 2 or Chi2PerDof < 0 or isnan(Chi2PerDof)) + { + goodFit_ = false; + cerr << "chi2PerDof = " << Chi2PerDof << ". Aborting fit now." << endl; + } + +} + // get total fit variance matrix and its pseudo-inverse //////////////////////// const DMat & XYSampleData::getFitVarMat(void) { @@ -292,24 +307,34 @@ SampleFitResult XYSampleData::fit(std::vector &minimizer, result.resize(nSample_); result.chi2_.resize(nSample_); result.model_.resize(v.size()); + double chi2PerDof; + goodFit_ = true; FOR_STAT_ARRAY(result, s) { - setDataToSample(s); - if (s == central) + if(goodFit_) { - sampleResult = data_.fit(minimizer, initCopy, v); - initCopy = sampleResult.segment(0, initCopy.size()); - } - else - { - sampleResult = data_.fit(*(minimizer.back()), initCopy, v); - } - result[s] = sampleResult; - result.chi2_[s] = sampleResult.getChi2(); - for (unsigned int j = 0; j < v.size(); ++j) - { - result.model_[j].resize(nSample_); - result.model_[j][s] = sampleResult.getModel(j); + setDataToSample(s); + if (s == central) + { + sampleResult = data_.fit(minimizer, initCopy, v); + initCopy = sampleResult.segment(0, initCopy.size()); + chi2PerDof = sampleResult.getChi2PerDof(); + checkChi2PerDof(chi2PerDof); + } + else + { + sampleResult = data_.fit(*(minimizer.back()), initCopy, v); + chi2PerDof = sampleResult.getChi2PerDof(); + checkChi2PerDof(chi2PerDof); + } + result[s] = sampleResult; + result.chi2_[s] = sampleResult.getChi2(); + for (unsigned int j = 0; j < v.size(); ++j) + { + result.model_[j].resize(nSample_); + result.model_[j][s] = sampleResult.getModel(j); + + } } } result.nPar_ = sampleResult.getNPar(); diff --git a/lib/Statistics/XYSampleData.hpp b/lib/Statistics/XYSampleData.hpp index 6cba855..a66c9dd 100644 --- a/lib/Statistics/XYSampleData.hpp +++ b/lib/Statistics/XYSampleData.hpp @@ -91,6 +91,8 @@ public: const DMat & getXYVar(const Index i, const Index j); DVec getXError(const Index i); DVec getYError(const Index j); + bool checkFit(); // check fit candidate based on chi2PerDof + void checkChi2PerDof(double Chi2PerDof); // get total fit variance matrix and its pseudo-inverse const DMat & getFitVarMat(void); const DMat & getFitVarMatPInv(void); @@ -133,6 +135,7 @@ private: Index nSample_, dataSample_{central}; bool initData_{true}, computeVarMat_{true}; bool initXMap_{true}; + bool goodFit_{true}; // used to break minimisation if central sample chi2PerDof is bad }; /******************************************************************************