diff --git a/lib/qcd/QCD.h b/lib/qcd/QCD.h index f154fa6a..ddb0e217 100644 --- a/lib/qcd/QCD.h +++ b/lib/qcd/QCD.h @@ -484,6 +484,8 @@ namespace QCD { } //namespace QCD } // Grid + + #include #include @@ -494,12 +496,12 @@ namespace QCD { #include #include +#include + #include #include -#include - #include #include #include diff --git a/lib/qcd/action/ActionBase.h b/lib/qcd/action/ActionBase.h index 83bb3ff7..c79416b8 100644 --- a/lib/qcd/action/ActionBase.h +++ b/lib/qcd/action/ActionBase.h @@ -1,49 +1,52 @@ - /************************************************************************************* +/************************************************************************************* - Grid physics library, www.github.com/paboyle/Grid +Grid physics library, www.github.com/paboyle/Grid - Source file: ./lib/qcd/action/ActionBase.h +Source file: ./lib/qcd/action/ActionBase.h - Copyright (C) 2015 +Copyright (C) 2015 Author: Peter Boyle Author: neo - This program is free software; you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation; either version 2 of the License, or - (at your option) any later version. +This program is free software; you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation; either version 2 of the License, or +(at your option) any later version. - This program 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 General Public License for more details. +This program 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 General Public License for more details. - You should have received a copy of the GNU General Public License along - with this program; if not, write to the Free Software Foundation, Inc., - 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +You should have received a copy of the GNU General Public License along +with this program; if not, write to the Free Software Foundation, Inc., +51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. - See the full license in the file "LICENSE" in the top level distribution directory - *************************************************************************************/ - /* END LEGAL */ +See the full license in the file "LICENSE" in the top level distribution +directory +*************************************************************************************/ +/* END LEGAL */ #ifndef QCD_ACTION_BASE #define QCD_ACTION_BASE namespace Grid { -namespace QCD{ - -template -class Action { +namespace QCD { +template +class Action { public: bool is_smeared = false; // Boundary conditions? // Heatbath? - virtual void refresh(const GaugeField &U, GridParallelRNG& pRNG) = 0;// refresh pseudofermions - virtual RealD S (const GaugeField &U) = 0; // evaluate the action - virtual void deriv(const GaugeField &U,GaugeField & dSdU ) = 0; // evaluate the action derivative - virtual ~Action() {}; + virtual void refresh(const GaugeField& U, + GridParallelRNG& pRNG) = 0; // refresh pseudofermions + virtual RealD S(const GaugeField& U) = 0; // evaluate the action + virtual void deriv(const GaugeField& U, + GaugeField& dSdU) = 0; // evaluate the action derivative + virtual ~Action(){}; }; -// Could derive PseudoFermion action with a PF field, FermionField, and a Grid; implement refresh +// Could derive PseudoFermion action with a PF field, FermionField, and a Grid; +// implement refresh /* template class PseudoFermionAction : public Action { @@ -52,7 +55,8 @@ class PseudoFermionAction : public Action { GridParallelRNG &pRNG; GridBase &Grid; - PseudoFermionAction(GridBase &_Grid,GridParallelRNG &_pRNG) : Grid(_Grid), Phi(&_Grid), pRNG(_pRNG) { + PseudoFermionAction(GridBase &_Grid,GridParallelRNG &_pRNG) : Grid(_Grid), +Phi(&_Grid), pRNG(_pRNG) { }; virtual void refresh(const GaugeField &gauge) { @@ -62,26 +66,90 @@ class PseudoFermionAction : public Action { }; */ -template struct ActionLevel{ -public: - - typedef Action* ActPtr; // now force the same colours as the rest of the code - +template +struct ActionLevel { + public: + typedef Action* + ActPtr; // now force the same colours as the rest of the code + + //Add supported representations here + + unsigned int multiplier; std::vector actions; ActionLevel(unsigned int mul = 1) : actions(0), multiplier(mul) { - assert (mul >= 1); + assert(mul >= 1); }; - - void push_back(ActPtr ptr){ - actions.push_back(ptr); - } + + void push_back(ActPtr ptr) { actions.push_back(ptr); } }; -template using ActionSet = std::vector >; +template +struct ActionLevelHirep { + public: + unsigned int multiplier; + + // Fundamental repr actions separated because of the smearing + typedef Action* ActPtr; + //std::vector actions; + // construct a tuple of vectors of the actions for the corresponding higher + // representation fields + typename AccessTypes::VectorCollection actions_hirep; + typedef typename AccessTypes::ClassCollection actions_hirep_ptrs_type; + + std::vector& actions; + + // Temporary conversion between ActionLevel and ActionLevelHirep + ActionLevelHirep(ActionLevel& AL ):actions(AL.actions), multiplier(AL.multiplier){} + + + + ActionLevelHirep(unsigned int mul = 1) : actions(std::get<0>(actions_hirep)), multiplier(mul) { + // initialize the hirep vectors to zero. + //apply(&ActionLevelHirep::resize, actions_hirep, 0); //need a working resize + assert(mul >= 1); + }; + + void push_back(ActPtr ptr) { actions.push_back(ptr); } + +// SFINAE construct, check + template + void push_back(actionpointer ptr, decltype(std::tuple_element::value)* = 0) { + //insert only in the correct vector + std::get(actions_hirep).push_back(ptr); + }; + + template < class ActPtr> + static void resize(ActPtr ap, unsigned int n){ + ap->resize(n); + + } + + + + // Loop on tuple for a callable function + template + inline typename std::enable_if<(I == std::tuple_size::value), void>::type apply( + Callable&, Tuple& , Args...) {} + + template + inline typename std::enable_if<(I < std::tuple_size::value), void>::type apply( + Callable& fn, Tuple& T, Args... arguments) { + fn(std::get(T), arguments...); + apply(T, fn, arguments...); + } + +}; + + +template +using ActionSet = std::vector >; + +template +using ActionSetHirep = std::vector >; } } diff --git a/lib/qcd/hmc/HmcRunner.h b/lib/qcd/hmc/HmcRunner.h index cd7acd56..d434970c 100644 --- a/lib/qcd/hmc/HmcRunner.h +++ b/lib/qcd/hmc/HmcRunner.h @@ -32,14 +32,14 @@ directory namespace Grid { namespace QCD { -template +template class NerscHmcRunnerTemplate { public: INHERIT_GIMPL_TYPES(Gimpl); enum StartType_t { ColdStart, HotStart, TepidStart, CheckpointStart }; - ActionSet TheAction; + ActionSetHirep TheAction; GridCartesian *UGrid; GridCartesian *FGrid; @@ -111,7 +111,7 @@ class NerscHmcRunnerTemplate { SmearedConfiguration SmearingPolicy(UGrid, Nsmear, Stout); std::cout << GridLogDebug << " done\n"; ////////////// - typedef MinimumNorm2 > + typedef MinimumNorm2, RepresentationsPolicy > IntegratorType; // change here to change the algorithm IntegratorParameters MDpar(20); IntegratorType MDynamics(UGrid, MDpar, TheAction, SmearingPolicy); @@ -177,6 +177,12 @@ typedef NerscHmcRunnerTemplate PeriodicNerscHmcRunnerD; typedef NerscHmcRunnerTemplate ConjugateNerscHmcRunner; typedef NerscHmcRunnerTemplate ConjugateNerscHmcRunnerF; typedef NerscHmcRunnerTemplate ConjugateNerscHmcRunnerD; + +template +using NerscHmcRunnerHirep = NerscHmcRunnerTemplate; + + + } } #endif diff --git a/lib/qcd/hmc/integrators/Integrator.h b/lib/qcd/hmc/integrators/Integrator.h index e9f0c0c7..55eaf818 100644 --- a/lib/qcd/hmc/integrators/Integrator.h +++ b/lib/qcd/hmc/integrators/Integrator.h @@ -71,7 +71,7 @@ class Integrator { IntegratorParameters Params; - const ActionSet as; + const ActionSetHirep as; int levels; // double t_U; // Track time passing on each level and for U and for P @@ -113,7 +113,7 @@ class Integrator { // to be used by the actionlevel class to iterate // over the representations template - void update_P_core(Level repr_level, GaugeField& Mom, GaugeField& U, + void update_P_hireps(Level repr_level, GaugeField& Mom, GaugeField& U, double ep) { typedef typename Level::LatticeField FieldType; FieldType Ur = repr_level->getRepresentation();// update U is better @@ -128,10 +128,10 @@ class Integrator { Mom -= force * ep; } } - // Add the specialized class for the fundamental case void update_P(GaugeField& Mom, GaugeField& U, int level, double ep) { // input U actually not used in the fundamental case + // Fundamental updates, include smearing for (int a = 0; a < as[level].actions.size(); ++a) { GaugeField force(U._grid); GaugeField& Us = Smearer.get_U(as[level].actions.at(a)->is_smeared); @@ -148,7 +148,7 @@ class Integrator { Mom -= force * ep; } // Add here the other representations - // as[level].apply(update_P_hireps, Args...) + //apply(update_P_hireps, as[level], Args...) } void update_U(GaugeField& U, double ep) { @@ -179,7 +179,7 @@ class Integrator { public: Integrator(GridBase* grid, IntegratorParameters Par, - ActionSet& Aset, SmearingPolicy& Sm) + ActionSetHirep& Aset, SmearingPolicy& Sm) : Params(Par), as(Aset), P(grid), levels(Aset.size()), Smearer(Sm), Representations(grid) { t_P.resize(levels, 0.0); t_U = 0.0; diff --git a/lib/qcd/hmc/integrators/Integrator_algorithm.h b/lib/qcd/hmc/integrators/Integrator_algorithm.h index 9e12c0c3..3c89a14d 100644 --- a/lib/qcd/hmc/integrators/Integrator_algorithm.h +++ b/lib/qcd/hmc/integrators/Integrator_algorithm.h @@ -151,7 +151,7 @@ namespace Grid{ MinimumNorm2(GridBase* grid, IntegratorParameters Par, - ActionSet & Aset, + ActionSetHirep & Aset, SmearingPolicy& Sm): Integrator(grid,Par,Aset,Sm) {}; diff --git a/lib/qcd/representations/adjoint.h b/lib/qcd/representations/adjoint.h index b211e122..9e960952 100644 --- a/lib/qcd/representations/adjoint.h +++ b/lib/qcd/representations/adjoint.h @@ -21,14 +21,15 @@ class AdjointRep { public: // typdef to be used by the Representations class in HMC to get the // types for the higher representation fields - typedef typename SU_Adjoint::LatticeAdjMatrix LatticeField; + typedef typename SU_Adjoint::LatticeAdjMatrix LatticeMatrix; + typedef typename SU_Adjoint::LatticeAdjField LatticeField; const int Dimension = ncolour * ncolour - 1; LatticeField U; explicit AdjointRep(GridBase* grid) : U(grid) {} - LatticeField update_representation(const LatticeGaugeField& Uin) { + void update_representation(const LatticeGaugeField& Uin) { // Uin is in the fundamental representation // get the U in AdjointRep // (U_adj)_B = tr[e^a U e^b U^dag] diff --git a/lib/qcd/representations/fundamental.h b/lib/qcd/representations/fundamental.h index 886dd399..0e9470a7 100644 --- a/lib/qcd/representations/fundamental.h +++ b/lib/qcd/representations/fundamental.h @@ -22,7 +22,8 @@ class FundamentalRep { // typdef to be used by the Representations class in HMC to get the // types for the higher representation fields - typedef typename SU::LatticeMatrix LatticeField; + typedef typename SU::LatticeMatrix LatticeMatrix; + typedef LatticeGaugeField LatticeField; explicit FundamentalRep(GridBase* grid) {} //do nothing void update_representation(const LatticeGaugeField& Uin) {} // do nothing diff --git a/lib/qcd/representations/hmc_types.h b/lib/qcd/representations/hmc_types.h index 4a281e75..a8d3ed7c 100644 --- a/lib/qcd/representations/hmc_types.h +++ b/lib/qcd/representations/hmc_types.h @@ -1,61 +1,90 @@ #ifndef HMC_TYPES_H #define HMC_TYPES_H -#include -#include #include #include +#include +#include namespace Grid { namespace QCD { - // Supported types -//enum {Fundamental, Adjoint} repr_type; +// enum {Fundamental, Adjoint} repr_type; -// Utility to add support to the HMC for representations other than the fundamental -template -class Representations{ -public: +// Utility to add support to the HMC for representations other than the +// fundamental +template +class Representations { + public: typedef std::tuple Representation_type; + // Size of the tuple, known at compile time + static const int tuple_size = sizeof...(Reptypes); + // The collection of types for the gauge fields + typedef std::tuple Representation_Fields; + // To access the Reptypes (FundamentalRepresentation, AdjointRepresentation) template - using repr_type = typename std::tuple_element::type; + using repr_type = typename std::tuple_element::type; // in order to get the typename of the field use // type repr_type::LatticeField Representation_type rep; // Multiple types constructor - explicit Representations(GridBase *grid):rep(Reptypes(grid)...){}; + explicit Representations(GridBase* grid) : rep(Reptypes(grid)...){}; - int size(){ - return std::tuple_size< Representation_type >::value; - } + int size() { return tuple_size; } // update the fields template - inline typename std::enable_if< I == sizeof...(Reptypes), void >::type update(LatticeGaugeField& U) {} + inline typename std::enable_if<(I == tuple_size), void>::type update( + LatticeGaugeField& U) {} template - inline typename std::enable_if < - I::type update(LatticeGaugeField& U) { + inline typename std::enable_if<(I < tuple_size), void>::type update( + LatticeGaugeField& U) { std::get(rep).update_representation(U); update(U); - } + } + + + }; +typedef Representations NoHirep; -typedef Representations JustTheFundamental; +// Helper classes to access the elements +// Strips the first N parameters from the tuple +// sequence of classes to obtain the S sequence +// Creates a type that is a tuple of vectors of the template type A +template