mirror of
				https://github.com/paboyle/Grid.git
				synced 2025-10-25 18:19:34 +01:00 
			
		
		
		
	Compare commits
	
		
			28 Commits
		
	
	
		
			feature/sc
			...
			feature/dw
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | a5645a7efe | ||
|  | 8b3e738269 | ||
|  | 013ea4e8d1 | ||
|  | 7fbbb31a50 | ||
|  | 0e127b1fc7 | ||
|  | 68c028b0a6 | ||
| a61e0df54b | |||
| f871fb0c6d | |||
|  | 9ada378e38 | ||
|  | 587bfcc0f4 | ||
|  | 4f4181c54a | ||
|  | b35169f1dd | ||
|  | 441ad7498d | ||
|  | 96272f3841 | ||
|  | 5c936d88a0 | ||
|  | 1c64ee926e | ||
|  | 2cbb72a81c | ||
|  | 31d83ee046 | ||
|  | a9e8758a01 | ||
|  | 3e125c5b61 | ||
|  | eac6ec4b5e | ||
|  | 213f8db6a2 | ||
|  | 80302e95a8 | ||
|  | b938202081 | ||
|  | 0f468e2179 | ||
|  | 97b9c6f03d | ||
|  | 63982819c6 | ||
|  | 24162c9ead | 
							
								
								
									
										17
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -19,6 +19,8 @@ before_install: | ||||
|     - if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then brew install libmpc; fi | ||||
|      | ||||
| install: | ||||
|     - export CWD=`pwd` | ||||
|     - echo $CWD | ||||
|     - export CC=$CC$VERSION | ||||
|     - export CXX=$CXX$VERSION | ||||
|     - echo $PATH | ||||
| @@ -36,11 +38,22 @@ script: | ||||
|     - ./bootstrap.sh | ||||
|     - mkdir build | ||||
|     - cd build | ||||
|     - ../configure --enable-precision=single --enable-simd=SSE4 --enable-comms=none | ||||
|     - mkdir lime | ||||
|     - cd lime | ||||
|     - mkdir build | ||||
|     - cd build | ||||
|     - wget http://usqcd-software.github.io/downloads/c-lime/lime-1.3.2.tar.gz | ||||
|     - tar xf lime-1.3.2.tar.gz | ||||
|     - cd lime-1.3.2 | ||||
|     - ./configure --prefix=$CWD/build/lime/install | ||||
|     - make -j4 | ||||
|     - make install | ||||
|     - cd $CWD/build | ||||
|     - ../configure --enable-precision=single --enable-simd=SSE4 --enable-comms=none --with-lime=$CWD/build/lime/install | ||||
|     - make -j4  | ||||
|     - ./benchmarks/Benchmark_dwf --threads 1 --debug-signals | ||||
|     - echo make clean | ||||
|     - ../configure --enable-precision=double --enable-simd=SSE4 --enable-comms=none | ||||
|     - ../configure --enable-precision=double --enable-simd=SSE4 --enable-comms=none --with-lime=$CWD/build/lime/install | ||||
|     - make -j4 | ||||
|     - ./benchmarks/Benchmark_dwf --threads 1 --debug-signals | ||||
|     - make check | ||||
|   | ||||
| @@ -49,6 +49,7 @@ Author: Peter Boyle <paboyle@ph.ed.ac.uk> | ||||
| #include <Grid/algorithms/iterative/BlockConjugateGradient.h> | ||||
| #include <Grid/algorithms/iterative/ConjugateGradientReliableUpdate.h> | ||||
| #include <Grid/algorithms/iterative/ImplicitlyRestartedLanczos.h> | ||||
| #include <Grid/algorithms/iterative/InexactPrecConjugateGradient.h> | ||||
| #include <Grid/algorithms/CoarsenedMatrix.h> | ||||
| #include <Grid/algorithms/FFT.h> | ||||
|  | ||||
|   | ||||
| @@ -309,36 +309,59 @@ namespace Grid { | ||||
|       class SchurStaggeredOperator :  public SchurOperatorBase<Field> { | ||||
|     protected: | ||||
|       Matrix &_Mat; | ||||
|       Field tmp; | ||||
|       RealD mass; | ||||
|       double tMpc; | ||||
|       double tIP; | ||||
|       double tMeo; | ||||
|       double taxpby_norm; | ||||
|       uint64_t ncall; | ||||
|     public: | ||||
|       SchurStaggeredOperator (Matrix &Mat): _Mat(Mat){}; | ||||
|       void Report(void) | ||||
|       { | ||||
| 	std::cout << GridLogMessage << " HermOpAndNorm.Mpc "<< tMpc/ncall<<" usec "<<std::endl; | ||||
| 	std::cout << GridLogMessage << " HermOpAndNorm.IP  "<< tIP /ncall<<" usec "<<std::endl; | ||||
| 	std::cout << GridLogMessage << " Mpc.MeoMoe        "<< tMeo/ncall<<" usec "<<std::endl; | ||||
| 	std::cout << GridLogMessage << " Mpc.axpby_norm    "<< taxpby_norm/ncall<<" usec "<<std::endl; | ||||
|       } | ||||
|       SchurStaggeredOperator (Matrix &Mat): _Mat(Mat), tmp(_Mat.RedBlackGrid())  | ||||
|       {  | ||||
| 	assert( _Mat.isTrivialEE() ); | ||||
| 	mass = _Mat.Mass(); | ||||
| 	tMpc=0; | ||||
| 	tIP =0; | ||||
|         tMeo=0; | ||||
|         taxpby_norm=0; | ||||
| 	ncall=0; | ||||
|       } | ||||
|       virtual void HermOpAndNorm(const Field &in, Field &out,RealD &n1,RealD &n2){ | ||||
| 	GridLogIterative.TimingMode(1); | ||||
| 	std::cout << GridLogIterative << " HermOpAndNorm "<<std::endl; | ||||
| 	ncall++; | ||||
| 	tMpc-=usecond(); | ||||
| 	n2 = Mpc(in,out); | ||||
| 	std::cout << GridLogIterative << " HermOpAndNorm.Mpc "<<std::endl; | ||||
| 	tMpc+=usecond(); | ||||
| 	tIP-=usecond(); | ||||
| 	ComplexD dot= innerProduct(in,out); | ||||
| 	std::cout << GridLogIterative << " HermOpAndNorm.innerProduct "<<std::endl; | ||||
| 	tIP+=usecond(); | ||||
| 	n1 = real(dot); | ||||
|       } | ||||
|       virtual void HermOp(const Field &in, Field &out){ | ||||
| 	std::cout << GridLogIterative << " HermOp "<<std::endl; | ||||
| 	Mpc(in,out); | ||||
| 	ncall++; | ||||
| 	tMpc-=usecond(); | ||||
| 	_Mat.Meooe(in,out); | ||||
| 	_Mat.Meooe(out,tmp); | ||||
| 	tMpc+=usecond(); | ||||
| 	taxpby_norm-=usecond(); | ||||
| 	axpby(out,-1.0,mass*mass,tmp,in); | ||||
| 	taxpby_norm+=usecond(); | ||||
|       } | ||||
|       virtual  RealD Mpc      (const Field &in, Field &out) { | ||||
| 	Field tmp(in._grid); | ||||
| 	Field tmp2(in._grid); | ||||
|  | ||||
| 	std::cout << GridLogIterative << " HermOp.Mpc "<<std::endl; | ||||
| 	_Mat.Mooee(in,out); | ||||
| 	_Mat.Mooee(out,tmp); | ||||
| 	std::cout << GridLogIterative << " HermOp.MooeeMooee "<<std::endl; | ||||
|  | ||||
| 	tMeo-=usecond(); | ||||
| 	_Mat.Meooe(in,out); | ||||
| 	_Mat.Meooe(out,tmp2); | ||||
| 	std::cout << GridLogIterative << " HermOp.MeooeMeooe "<<std::endl; | ||||
|  | ||||
| 	RealD nn=axpy_norm(out,-1.0,tmp2,tmp); | ||||
| 	std::cout << GridLogIterative << " HermOp.axpy_norm "<<std::endl; | ||||
| 	_Mat.Meooe(out,tmp); | ||||
| 	tMeo+=usecond(); | ||||
| 	taxpby_norm-=usecond(); | ||||
| 	RealD nn=axpby_norm(out,-1.0,mass*mass,tmp,in); | ||||
| 	taxpby_norm+=usecond(); | ||||
| 	return nn; | ||||
|       } | ||||
|       virtual  RealD MpcDag   (const Field &in, Field &out){ | ||||
|   | ||||
| @@ -53,4 +53,10 @@ void MultiShiftFunction::csv(std::ostream &out) | ||||
|   } | ||||
|   return; | ||||
| } | ||||
|  | ||||
| RealD __InverseApproximation(RealD x) | ||||
| { | ||||
|   return 1.0/x; | ||||
| } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -63,5 +63,7 @@ public: | ||||
|   } | ||||
|  | ||||
| }; | ||||
| RealD __InverseApproximation(RealD x); | ||||
|  | ||||
| } | ||||
| #endif | ||||
|   | ||||
| @@ -54,6 +54,7 @@ class ConjugateGradient : public OperatorFunction<Field> { | ||||
|  | ||||
|   void operator()(LinearOperatorBase<Field> &Linop, const Field &src, Field &psi) { | ||||
|  | ||||
|  | ||||
|     psi.checkerboard = src.checkerboard; | ||||
|     conformable(psi, src); | ||||
|  | ||||
| @@ -70,7 +71,6 @@ class ConjugateGradient : public OperatorFunction<Field> { | ||||
|      | ||||
|     Linop.HermOpAndNorm(psi, mmp, d, b); | ||||
|  | ||||
|  | ||||
|     r = src - mmp; | ||||
|     p = r; | ||||
|  | ||||
| @@ -96,38 +96,44 @@ class ConjugateGradient : public OperatorFunction<Field> { | ||||
|               << "ConjugateGradient: k=0 residual " << cp << " target " << rsq << std::endl; | ||||
|  | ||||
|     GridStopWatch LinalgTimer; | ||||
|     GridStopWatch InnerTimer; | ||||
|     GridStopWatch AxpyNormTimer; | ||||
|     GridStopWatch LinearCombTimer; | ||||
|     GridStopWatch MatrixTimer; | ||||
|     GridStopWatch SolverTimer; | ||||
|  | ||||
|     SolverTimer.Start(); | ||||
|     int k; | ||||
|     for (k = 1; k <= MaxIterations; k++) { | ||||
|     for (k = 1; k <= MaxIterations*1000; k++) { | ||||
|       c = cp; | ||||
|  | ||||
|       MatrixTimer.Start(); | ||||
|       Linop.HermOpAndNorm(p, mmp, d, qq); | ||||
|       Linop.HermOp(p, mmp); | ||||
|       MatrixTimer.Stop(); | ||||
|  | ||||
|       LinalgTimer.Start(); | ||||
|       //  RealD    qqck = norm2(mmp); | ||||
|       //  ComplexD dck  = innerProduct(p,mmp); | ||||
|  | ||||
|       InnerTimer.Start(); | ||||
|       ComplexD dc  = innerProduct(p,mmp); | ||||
|       InnerTimer.Stop(); | ||||
|       d = dc.real(); | ||||
|       a = c / d; | ||||
|       b_pred = a * (a * qq - d) / c; | ||||
|  | ||||
|       AxpyNormTimer.Start(); | ||||
|       cp = axpy_norm(r, -a, mmp, r); | ||||
|       AxpyNormTimer.Stop(); | ||||
|       b = cp / c; | ||||
|  | ||||
|       // Fuse these loops ; should be really easy | ||||
|       psi = a * p + psi; | ||||
|       p = p * b + r; | ||||
|  | ||||
|       LinearCombTimer.Start(); | ||||
|       parallel_for(int ss=0;ss<src._grid->oSites();ss++){ | ||||
| 	vstream(psi[ss], a      *  p[ss] + psi[ss]); | ||||
| 	vstream(p  [ss], b      *  p[ss] + r[ss]); | ||||
|       } | ||||
|       LinearCombTimer.Stop(); | ||||
|       LinalgTimer.Stop(); | ||||
|  | ||||
|       std::cout << GridLogIterative << "ConjugateGradient: Iteration " << k | ||||
|                 << " residual " << cp << " target " << rsq << std::endl; | ||||
|       std::cout << GridLogDebug << "a = "<< a << " b_pred = "<< b_pred << "  b = "<< b << std::endl; | ||||
|       std::cout << GridLogDebug << "qq = "<< qq << " d = "<< d << "  c = "<< c << std::endl; | ||||
|  | ||||
|       // Stopping condition | ||||
|       if (cp <= rsq) { | ||||
| @@ -148,6 +154,9 @@ class ConjugateGradient : public OperatorFunction<Field> { | ||||
| 	std::cout << GridLogMessage << "\tElapsed    " << SolverTimer.Elapsed() <<std::endl; | ||||
| 	std::cout << GridLogMessage << "\tMatrix     " << MatrixTimer.Elapsed() <<std::endl; | ||||
| 	std::cout << GridLogMessage << "\tLinalg     " << LinalgTimer.Elapsed() <<std::endl; | ||||
| 	std::cout << GridLogMessage << "\tInner      " << InnerTimer.Elapsed() <<std::endl; | ||||
| 	std::cout << GridLogMessage << "\tAxpyNorm   " << AxpyNormTimer.Elapsed() <<std::endl; | ||||
| 	std::cout << GridLogMessage << "\tLinearComb " << LinearCombTimer.Elapsed() <<std::endl; | ||||
|  | ||||
|         if (ErrorOnNoConverge) assert(true_residual / Tolerance < 10000.0); | ||||
|  | ||||
|   | ||||
| @@ -43,6 +43,7 @@ namespace Grid { | ||||
| public:                                                 | ||||
|     RealD   Tolerance; | ||||
|     Integer MaxIterations; | ||||
|     Integer IterationsToComplete; //Number of iterations the CG took to finish. Filled in upon completion | ||||
|     int verbose; | ||||
|     MultiShiftFunction shifts; | ||||
|  | ||||
| @@ -164,6 +165,15 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
|     axpby(psi[s],0.,-bs[s]*alpha[s],src,src); | ||||
|   } | ||||
|   | ||||
|   /////////////////////////////////////// | ||||
|   // Timers | ||||
|   /////////////////////////////////////// | ||||
|   GridStopWatch AXPYTimer; | ||||
|   GridStopWatch ShiftTimer; | ||||
|   GridStopWatch QRTimer; | ||||
|   GridStopWatch MatrixTimer; | ||||
|   GridStopWatch SolverTimer; | ||||
|   SolverTimer.Start(); | ||||
|    | ||||
|   // Iteration loop | ||||
|   int k; | ||||
| @@ -171,7 +181,9 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
|   for (k=1;k<=MaxIterations;k++){ | ||||
|      | ||||
|     a = c /cp; | ||||
|     AXPYTimer.Start(); | ||||
|     axpy(p,a,p,r); | ||||
|     AXPYTimer.Stop(); | ||||
|      | ||||
|     // Note to self - direction ps is iterated seperately | ||||
|     // for each shift. Does not appear to have any scope | ||||
| @@ -180,6 +192,7 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
|     // However SAME r is used. Could load "r" and update | ||||
|     // ALL ps[s]. 2/3 Bandwidth saving | ||||
|     // New Kernel: Load r, vector of coeffs, vector of pointers ps | ||||
|     AXPYTimer.Start(); | ||||
|     for(int s=0;s<nshift;s++){ | ||||
|       if ( ! converged[s] ) {  | ||||
| 	if (s==0){ | ||||
| @@ -190,22 +203,34 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
| 	} | ||||
|       } | ||||
|     } | ||||
|     AXPYTimer.Stop(); | ||||
|      | ||||
|     cp=c; | ||||
|     MatrixTimer.Start();   | ||||
|     //Linop.HermOpAndNorm(p,mmp,d,qq); // d is used | ||||
|     // The below is faster on KNL | ||||
|     Linop.HermOp(p,mmp);  | ||||
|     d=real(innerProduct(p,mmp)); | ||||
|      | ||||
|     Linop.HermOpAndNorm(p,mmp,d,qq); | ||||
|     MatrixTimer.Stop();   | ||||
|  | ||||
|     AXPYTimer.Start(); | ||||
|     axpy(mmp,mass[0],p,mmp); | ||||
|     AXPYTimer.Stop(); | ||||
|     RealD rn = norm2(p); | ||||
|     d += rn*mass[0]; | ||||
|      | ||||
|     bp=b; | ||||
|     b=-cp/d; | ||||
|      | ||||
|     AXPYTimer.Start(); | ||||
|     c=axpy_norm(r,b,mmp,r); | ||||
|     AXPYTimer.Stop(); | ||||
|  | ||||
|     // Toggle the recurrence history | ||||
|     bs[0] = b; | ||||
|     iz = 1-iz; | ||||
|     ShiftTimer.Start(); | ||||
|     for(int s=1;s<nshift;s++){ | ||||
|       if((!converged[s])){ | ||||
| 	RealD z0 = z[s][1-iz]; | ||||
| @@ -215,6 +240,7 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
| 	bs[s] = b*z[s][iz]/z0; // NB sign  rel to Mike | ||||
|       } | ||||
|     } | ||||
|     ShiftTimer.Stop(); | ||||
|      | ||||
|     for(int s=0;s<nshift;s++){ | ||||
|       int ss = s; | ||||
| @@ -257,6 +283,9 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
|      | ||||
|     if ( all_converged ){ | ||||
|  | ||||
|     SolverTimer.Stop(); | ||||
|  | ||||
|  | ||||
|       std::cout<<GridLogMessage<< "CGMultiShift: All shifts have converged iteration "<<k<<std::endl; | ||||
|       std::cout<<GridLogMessage<< "CGMultiShift: Checking solutions"<<std::endl; | ||||
|        | ||||
| @@ -269,8 +298,19 @@ void operator() (LinearOperatorBase<Field> &Linop, const Field &src, std::vector | ||||
| 	RealD cn = norm2(src); | ||||
| 	std::cout<<GridLogMessage<<"CGMultiShift: shift["<<s<<"] true residual "<<std::sqrt(rn/cn)<<std::endl; | ||||
|       } | ||||
|  | ||||
|       std::cout << GridLogMessage << "Time Breakdown "<<std::endl; | ||||
|       std::cout << GridLogMessage << "\tElapsed    " << SolverTimer.Elapsed()     <<std::endl; | ||||
|       std::cout << GridLogMessage << "\tAXPY    " << AXPYTimer.Elapsed()     <<std::endl; | ||||
|       std::cout << GridLogMessage << "\tMarix    " << MatrixTimer.Elapsed()     <<std::endl; | ||||
|       std::cout << GridLogMessage << "\tShift    " << ShiftTimer.Elapsed()     <<std::endl; | ||||
|  | ||||
|       IterationsToComplete = k;	 | ||||
|  | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     | ||||
|   } | ||||
|   // ugly hack | ||||
|   std::cout<<GridLogMessage<<"CG multi shift did not converge"<<std::endl; | ||||
|   | ||||
							
								
								
									
										278
									
								
								lib/algorithms/iterative/InexactPrecConjugateGradient.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										278
									
								
								lib/algorithms/iterative/InexactPrecConjugateGradient.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,278 @@ | ||||
|     /************************************************************************************* | ||||
|  | ||||
|     Grid physics library, www.github.com/paboyle/Grid  | ||||
|  | ||||
|     Source file: ./lib/algorithms/iterative/InexactPrecConjugateGradient.h | ||||
|  | ||||
|     Copyright (C) 2015 | ||||
|  | ||||
| Author: Christopher Kelly <ckelly@phys.columbia.edu> | ||||
|  | ||||
|     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. | ||||
|  | ||||
|     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 */ | ||||
| #ifndef GRID_INEXACT_PREC_CONJUGATE_GRADIENT_H_ | ||||
| #define GRID_INEXACT_PREC_CONJUGATE_GRADIENT_H_ | ||||
|  | ||||
| namespace Grid { | ||||
|  | ||||
| //Inexact preconditioned CG based on Golub, Ye, SIAM J. Sci. Comput., 21(4), 13051320.  | ||||
| //(https://pdfs.semanticscholar.org/d2a9/d5bab02146a7fe3a244677432d21e33a2d98.pdf) | ||||
| template <class Field> | ||||
| class InexactPreconditionedConjugateGradient : public OperatorFunction<Field> { | ||||
|  public: | ||||
|   bool ErrorOnNoConverge;  // throw an assert when the CG fails to converge. | ||||
|                            // Defaults true. | ||||
|   RealD Tolerance; | ||||
|   Integer MaxIterations; | ||||
|   Integer IterationsToComplete; //Number of iterations the CG took to finish. Filled in upon completion | ||||
|    | ||||
|   LinearOperatorBase<Field> &Prec; | ||||
|  | ||||
|   InexactPreconditionedConjugateGradient(LinearOperatorBase<Field> &_Prec, RealD tol, Integer maxit, bool err_on_no_conv = true) | ||||
|     : Prec(_Prec), | ||||
|     Tolerance(tol), | ||||
|     MaxIterations(maxit), | ||||
|     ErrorOnNoConverge(err_on_no_conv){}; | ||||
|  | ||||
|   void operator()(LinearOperatorBase<Field> &Linop, const Field &src, Field &psi) { | ||||
|  | ||||
|     psi.checkerboard = src.checkerboard; | ||||
|     conformable(psi, src); | ||||
|  | ||||
|     Real ssq = norm2(src); | ||||
|     RealD rsq = Tolerance * Tolerance * ssq; //inner stopping condition | ||||
|  | ||||
|     Field p(src); | ||||
|     Field r(src); | ||||
|     Field rnm1(src); | ||||
|     Field mmp(src); | ||||
|     Field z(src); | ||||
|  | ||||
|     //Initialize variables | ||||
|     Linop.HermOp(psi, mmp);    | ||||
|     r = src - mmp; | ||||
|      | ||||
|     Real cp = norm2(r); | ||||
|      | ||||
|     p = zero; | ||||
|     Real alpha = 0, beta = 0; | ||||
|      | ||||
|     Real z_nm1_dot_r_nm1; | ||||
|  | ||||
|     int n; | ||||
|     for(n=1; n <= MaxIterations; n++) { | ||||
|       //Check stopping condition | ||||
|       if (cp <= rsq) { | ||||
|         Linop.HermOp(psi, mmp); | ||||
|         r = mmp - src; | ||||
|  | ||||
|         RealD srcnorm = sqrt(norm2(src)); | ||||
|         RealD resnorm = sqrt(norm2(r)); | ||||
|         RealD true_residual = resnorm / srcnorm; | ||||
|  | ||||
|         std::cout << GridLogMessage << "InexactPreconditionedConjugateGradient Converged on iteration " << n << std::endl; | ||||
|         std::cout << GridLogMessage << "\tComputed residual " << sqrt(cp / ssq)<<std::endl; | ||||
|   	std::cout << GridLogMessage << "\tTrue residual " << true_residual<<std::endl; | ||||
|   	std::cout << GridLogMessage << "\tTarget " << Tolerance << std::endl; | ||||
|  | ||||
|         if (ErrorOnNoConverge) assert(true_residual / Tolerance < 10000.0); | ||||
|  | ||||
|   	IterationsToComplete = n;	 | ||||
|  | ||||
|         return; | ||||
|       } | ||||
|        | ||||
|       std::cout << GridLogIterative << std::setprecision(8) | ||||
| 		<< "InexactPreconditionedConjugateGradient: n=" << n << " residual " << cp << " target " << rsq << std::endl; | ||||
|  | ||||
|       //Apply preconditioner to current residual | ||||
|       Prec.HermOp(r, z); | ||||
|  | ||||
|       //Update beta and store appropriate variables for next iteration | ||||
|       Real z_n_dot_r_n  = sqrt(norm(innerProduct(z,r))); | ||||
|  | ||||
|       if(n>1){ | ||||
| 	//  z^T_n ( r_n - r_{n-1} ) | ||||
| 	//  ----------------------- | ||||
| 	//     z^T_{n-1} r_{n-1} | ||||
|  | ||||
| 	Real z_n_dot_r_nm1 = sqrt(norm(innerProduct(z,rnm1))); | ||||
| 	beta = ( z_n_dot_r_n - z_n_dot_r_nm1 ) / z_nm1_dot_r_nm1; | ||||
| 	std::cout << GridLogIterative << "beta " << beta << std::endl; | ||||
|       } | ||||
|  | ||||
|       z_nm1_dot_r_nm1 = z_n_dot_r_n; //for next iteration | ||||
|       rnm1 = r; | ||||
|  | ||||
|       axpy(p, beta, p, z); //p = beta * p + z | ||||
|  | ||||
|       //Compute alpha | ||||
|       Linop.HermOp(p, mmp);       | ||||
|       alpha = z_n_dot_r_n / sqrt(norm(innerProduct(p, mmp))); | ||||
|  | ||||
|       std::cout << GridLogIterative << "alpha " << alpha << std::endl; | ||||
|  | ||||
|       //Update residual and solution | ||||
|       cp = axpy_norm(r, -alpha, mmp, r); | ||||
|       axpy(psi, alpha, p, psi); | ||||
|     } | ||||
|     std::cout << GridLogMessage << "InexactPreconditionedConjugateGradient did NOT converge" | ||||
|               << std::endl; | ||||
|  | ||||
|     if (ErrorOnNoConverge) assert(0); | ||||
|     IterationsToComplete = n; | ||||
|   } | ||||
| }; | ||||
|  | ||||
|  | ||||
| template<class Field> | ||||
| class PolynomialPreconditioner :  public LinearOperatorBase<Field> { | ||||
|   Chebyshev<Field> Cheby; | ||||
|   LinearOperatorBase<Field> &linop; | ||||
| public: | ||||
|   int InnerIterations; | ||||
|   int order; | ||||
|   PolynomialPreconditioner(LinearOperatorBase<Field> &_linop,RealD lo, RealD hi, int _order)  | ||||
|     : linop(_linop), Cheby(lo,hi,_order,__InverseApproximation)  | ||||
|     { | ||||
|       InnerIterations=0; | ||||
|       order = _order; | ||||
|     }; | ||||
|  | ||||
|   void OpDiag (const Field &in, Field &out){ assert(0); } | ||||
|   void OpDir  (const Field &in, Field &out,int dir,int disp){ 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){  | ||||
|     HermOp(in,out); | ||||
|     n1 = 0; n2 = norm2(out);  | ||||
|   } | ||||
|   void HermOp(const Field &in, Field &out){  | ||||
|     Cheby(linop,in,out);  | ||||
|     InnerIterations+=order; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| template<class Field> | ||||
| class DoNothingLinearOperator :  public LinearOperatorBase<Field> { | ||||
| public: | ||||
|   void OpDiag (const Field &in, Field &out){ assert(0); } | ||||
|   void OpDir  (const Field &in, Field &out,int dir,int disp){ 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){ out = in; n1 = 0; n2 = norm2(out); } | ||||
|   void HermOp(const Field &in, Field &out){ out = in; } | ||||
| }; | ||||
|  | ||||
| template<class Field> | ||||
| class FixedIterConjugateGradientPreconditioner :  public LinearOperatorBase<Field> { | ||||
| public: | ||||
|   LinearOperatorBase<Field> &linop; | ||||
|   ConjugateGradient<Field> CG; | ||||
|  | ||||
|   FixedIterConjugateGradientPreconditioner (LinearOperatorBase<Field> &_linop, Integer _iter): linop(_linop), CG(1e-20, _iter){ | ||||
|     CG.ErrorOnNoConverge = false; | ||||
|   } | ||||
|  | ||||
|   void OpDiag (const Field &in, Field &out){ assert(0); } | ||||
|   void OpDir  (const Field &in, Field &out,int dir,int disp){ 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){  | ||||
|     out = zero; | ||||
|     CG(linop,in,out);     | ||||
|     n2 = norm2(out); | ||||
|   } | ||||
|   void HermOp(const Field &in, Field &out){ | ||||
|     out = zero; | ||||
|     CG(linop,in,out);     | ||||
|   } | ||||
| }; | ||||
|  | ||||
| template<class Field> | ||||
| class SloppyConjugateGradientPreconditioner :  public LinearOperatorBase<Field> { | ||||
| public: | ||||
|   LinearOperatorBase<Field> &linop; | ||||
|   ConjugateGradient<Field> CG; | ||||
|   int InnerIterations; | ||||
|  | ||||
|   SloppyConjugateGradientPreconditioner (LinearOperatorBase<Field> &_linop, Real _resid, Integer max_iter): linop(_linop), CG(_resid, max_iter), InnerIterations(0){ | ||||
|   } | ||||
|  | ||||
|   void ResetCounters(){ InnerIterations = 0; } | ||||
|  | ||||
|   void OpDiag (const Field &in, Field &out){ assert(0); } | ||||
|   void OpDir  (const Field &in, Field &out,int dir,int disp){ 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){  | ||||
|     out = zero; | ||||
|     CG(linop,in,out);     | ||||
|     InnerIterations += CG.IterationsToComplete; | ||||
|     n2 = norm2(out); | ||||
|   } | ||||
|   void HermOp(const Field &in, Field &out){ | ||||
|     out = zero; | ||||
|     CG(linop,in,out); | ||||
|     InnerIterations += CG.IterationsToComplete; | ||||
|   } | ||||
| }; | ||||
|  | ||||
|  | ||||
| template<class FieldH, class FieldL> | ||||
| class SloppyConjugateGradientLowerPrecPreconditioner :  public LinearOperatorBase<FieldH> { | ||||
| public: | ||||
|   LinearOperatorBase<FieldL> &linop; | ||||
|   ConjugateGradient<FieldL> CG; | ||||
|   GridBase* L_grid; //lower-prec Grid | ||||
|   int InnerIterations; | ||||
|   FieldL tmp_l1; | ||||
|   FieldL tmp_l2; | ||||
|  | ||||
|   SloppyConjugateGradientLowerPrecPreconditioner (LinearOperatorBase<FieldL> &_linop, GridBase* _L_grid, Real _resid, Integer max_iter): linop(_linop), CG(_resid, max_iter), InnerIterations(0), L_grid(_L_grid), tmp_l1(_L_grid), tmp_l2(_L_grid){ | ||||
|     CG.ErrorOnNoConverge = false; | ||||
|   } | ||||
|  | ||||
|   void ResetCounters(){ InnerIterations = 0; } | ||||
|  | ||||
|   void OpDiag (const FieldH &in, FieldH &out){ assert(0); } | ||||
|   void OpDir  (const FieldH &in, FieldH &out,int dir,int disp){ assert(0); } | ||||
|   void Op     (const FieldH &in, FieldH &out){ assert(0); } | ||||
|   void AdjOp  (const FieldH &in, FieldH &out){ assert(0); } | ||||
|   void HermOpAndNorm(const FieldH &in, FieldH &out,RealD &n1,RealD &n2){  | ||||
|     precisionChange(tmp_l1, in); | ||||
|     tmp_l2 = zero; | ||||
|     CG(linop,tmp_l1,tmp_l2);     | ||||
|     InnerIterations += CG.IterationsToComplete; | ||||
|     precisionChange(out, tmp_l2); | ||||
|     n2 = norm2(tmp_l2); | ||||
|   } | ||||
|   void HermOp(const FieldH &in, FieldH &out){ | ||||
|     precisionChange(tmp_l1, in); | ||||
|     tmp_l2 = zero; | ||||
|     CG(linop,tmp_l1,tmp_l2);     | ||||
|     InnerIterations += CG.IterationsToComplete; | ||||
|     precisionChange(out, tmp_l2); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| #endif | ||||
| @@ -244,19 +244,11 @@ namespace Grid { | ||||
|  | ||||
|   template<class sobj,class vobj> strong_inline | ||||
|   RealD axpy_norm(Lattice<vobj> &ret,sobj a,const Lattice<vobj> &x,const Lattice<vobj> &y){ | ||||
|     ret.checkerboard = x.checkerboard; | ||||
|     conformable(ret,x); | ||||
|     conformable(x,y); | ||||
|     axpy(ret,a,x,y); | ||||
|     return norm2(ret); | ||||
|     return axpy_norm_fast(ret,a,x,y); | ||||
|   } | ||||
|   template<class sobj,class vobj> strong_inline | ||||
|   RealD axpby_norm(Lattice<vobj> &ret,sobj a,sobj b,const Lattice<vobj> &x,const Lattice<vobj> &y){ | ||||
|     ret.checkerboard = x.checkerboard; | ||||
|     conformable(ret,x); | ||||
|     conformable(x,y); | ||||
|     axpby(ret,a,b,x,y); | ||||
|     return norm2(ret); // FIXME implement parallel norm in ss loop | ||||
|     return axpby_norm_fast(ret,a,b,x,y); | ||||
|   } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -33,7 +33,7 @@ namespace Grid { | ||||
|   // Deterministic Reduction operations | ||||
|   //////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
| template<class vobj> inline RealD norm2(const Lattice<vobj> &arg){ | ||||
|   ComplexD nrm = innerProduct(arg,arg); | ||||
|   auto nrm = innerProduct(arg,arg); | ||||
|   return std::real(nrm);  | ||||
| } | ||||
|  | ||||
| @@ -43,32 +43,85 @@ inline ComplexD innerProduct(const Lattice<vobj> &left,const Lattice<vobj> &righ | ||||
| { | ||||
|   typedef typename vobj::scalar_type scalar_type; | ||||
|   typedef typename vobj::vector_typeD vector_type; | ||||
|   scalar_type  nrm; | ||||
|    | ||||
|   GridBase *grid = left._grid; | ||||
|   const int pad = 8; | ||||
|  | ||||
|   std::vector<vector_type,alignedAllocator<vector_type> > sumarray(grid->SumArraySize()); | ||||
|   ComplexD  inner; | ||||
|   Vector<ComplexD> sumarray(grid->SumArraySize()*pad); | ||||
|  | ||||
|   parallel_for(int thr=0;thr<grid->SumArraySize();thr++){ | ||||
|     int nwork, mywork, myoff; | ||||
|     GridThread::GetWork(left._grid->oSites(),thr,mywork,myoff); | ||||
|      | ||||
|     decltype(innerProductD(left._odata[0],right._odata[0])) vnrm=zero; // private to thread; sub summation | ||||
|     decltype(innerProductD(left._odata[0],right._odata[0])) vinner=zero; // private to thread; sub summation | ||||
|     for(int ss=myoff;ss<mywork+myoff; ss++){ | ||||
|       vnrm = vnrm + innerProductD(left._odata[ss],right._odata[ss]); | ||||
|       vinner = vinner + innerProductD(left._odata[ss],right._odata[ss]); | ||||
|     } | ||||
|     sumarray[thr]=TensorRemove(vnrm) ; | ||||
|     // All threads sum across SIMD; reduce serial work at end | ||||
|     // one write per cacheline with streaming store | ||||
|     ComplexD tmp = Reduce(TensorRemove(vinner)) ; | ||||
|     vstream(sumarray[thr*pad],tmp); | ||||
|   } | ||||
|    | ||||
|   vector_type vvnrm; vvnrm=zero;  // sum across threads | ||||
|   inner=0.0; | ||||
|   for(int i=0;i<grid->SumArraySize();i++){ | ||||
|     vvnrm = vvnrm+sumarray[i]; | ||||
|     inner = inner+sumarray[i*pad]; | ||||
|   }  | ||||
|   nrm = Reduce(vvnrm);// sum across simd | ||||
|   right._grid->GlobalSum(nrm); | ||||
|   right._grid->GlobalSum(inner); | ||||
|   return inner; | ||||
| } | ||||
|  | ||||
| ///////////////////////// | ||||
| // Fast axpby_norm | ||||
| // z = a x + b y | ||||
| // return norm z | ||||
| ///////////////////////// | ||||
| template<class sobj,class vobj> strong_inline RealD  | ||||
| axpy_norm_fast(Lattice<vobj> &z,sobj a,const Lattice<vobj> &x,const Lattice<vobj> &y)  | ||||
| { | ||||
|   sobj one(1.0); | ||||
|   return axpby_norm_fast(z,a,one,x,y); | ||||
| } | ||||
|  | ||||
| template<class sobj,class vobj> strong_inline RealD  | ||||
| axpby_norm_fast(Lattice<vobj> &z,sobj a,sobj b,const Lattice<vobj> &x,const Lattice<vobj> &y)  | ||||
| { | ||||
|   const int pad = 8; | ||||
|   z.checkerboard = x.checkerboard; | ||||
|   conformable(z,x); | ||||
|   conformable(x,y); | ||||
|  | ||||
|   typedef typename vobj::scalar_type scalar_type; | ||||
|   typedef typename vobj::vector_typeD vector_type; | ||||
|   RealD  nrm; | ||||
|    | ||||
|   GridBase *grid = x._grid; | ||||
|    | ||||
|   Vector<RealD> sumarray(grid->SumArraySize()*pad); | ||||
|    | ||||
|   parallel_for(int thr=0;thr<grid->SumArraySize();thr++){ | ||||
|     int nwork, mywork, myoff; | ||||
|     GridThread::GetWork(x._grid->oSites(),thr,mywork,myoff); | ||||
|      | ||||
|     // private to thread; sub summation | ||||
|     decltype(innerProductD(z._odata[0],z._odata[0])) vnrm=zero;  | ||||
|     for(int ss=myoff;ss<mywork+myoff; ss++){ | ||||
|       vobj tmp = a*x._odata[ss]+b*y._odata[ss]; | ||||
|       vnrm = vnrm + innerProductD(tmp,tmp); | ||||
|       vstream(z._odata[ss],tmp); | ||||
|     } | ||||
|     vstream(sumarray[thr*pad],real(Reduce(TensorRemove(vnrm)))) ; | ||||
|   } | ||||
|    | ||||
|   nrm = 0.0; // sum across threads; linear in thread count but fast | ||||
|   for(int i=0;i<grid->SumArraySize();i++){ | ||||
|     nrm = nrm+sumarray[i*pad]; | ||||
|   }  | ||||
|   z._grid->GlobalSum(nrm); | ||||
|   return nrm;  | ||||
| } | ||||
|  | ||||
|   | ||||
| template<class Op,class T1> | ||||
| inline auto sum(const LatticeUnaryExpression<Op,T1> & expr) | ||||
|   ->typename decltype(expr.first.func(eval(0,std::get<0>(expr.second))))::scalar_object | ||||
|   | ||||
| @@ -358,17 +358,10 @@ PARALLEL_CRITICAL | ||||
|  | ||||
|       if ( (control & BINARYIO_LEXICOGRAPHIC) && (nrank > 1) ) { | ||||
| #ifdef USE_MPI_IO | ||||
| 	std::cout<< GridLogMessage<<"IOobject: MPI read I/O "<< file<< " and offset " << offset << std::endl; | ||||
| 	std::cout<< GridLogMessage<<"IOobject: MPI read I/O "<< file<< std::endl; | ||||
| 	ierr=MPI_File_open(grid->communicator,(char *) file.c_str(), MPI_MODE_RDONLY, MPI_INFO_NULL, &fh);    assert(ierr==0); | ||||
| 	ierr=MPI_File_set_view(fh, disp, mpiObject, fileArray, "native", MPI_INFO_NULL);    assert(ierr==0); | ||||
| 	ierr=MPI_File_read_all(fh, &iodata[0], 1, localArray, &status);    assert(ierr==0); | ||||
|  | ||||
| 	MPI_Offset os; | ||||
| 	MPI_File_get_position(fh, &os); | ||||
| 	MPI_File_get_byte_offset(fh, os, &disp); | ||||
| 	offset = disp; | ||||
|  | ||||
|  | ||||
| 	MPI_File_close(&fh); | ||||
| 	MPI_Type_free(&fileArray); | ||||
| 	MPI_Type_free(&localArray); | ||||
| @@ -377,7 +370,7 @@ PARALLEL_CRITICAL | ||||
| #endif | ||||
|       } else { | ||||
| 	std::cout << GridLogMessage <<"IOobject: C++ read I/O " << file << " : " | ||||
|                   << iodata.size() * sizeof(fobj) << " bytes and offset " << offset << std::endl; | ||||
|                   << iodata.size() * sizeof(fobj) << " bytes" << std::endl; | ||||
|         std::ifstream fin; | ||||
| 	fin.open(file, std::ios::binary | std::ios::in); | ||||
|         if (control & BINARYIO_MASTER_APPEND) | ||||
| @@ -389,9 +382,7 @@ PARALLEL_CRITICAL | ||||
|           fin.seekg(offset + myrank * lsites * sizeof(fobj)); | ||||
|         } | ||||
|         fin.read((char *)&iodata[0], iodata.size() * sizeof(fobj)); | ||||
|          | ||||
|         assert(fin.fail() == 0); | ||||
|         offset = fin.tellg(); | ||||
|         fin.close(); | ||||
|       } | ||||
|       timer.Stop(); | ||||
| @@ -647,11 +638,6 @@ PARALLEL_CRITICAL | ||||
|     IOobject(w,grid,iodata,file,offset,format,BINARYIO_READ|BINARYIO_LEXICOGRAPHIC, | ||||
| 	     nersc_csum,scidac_csuma,scidac_csumb); | ||||
|  | ||||
|     std::cout << GridLogMessage << "RNG file nersc_checksum   " << std::hex << nersc_csum << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file scidac_checksuma " << std::hex << scidac_csuma << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file scidac_checksumb " << std::hex << scidac_csumb << std::dec << std::endl; | ||||
|  | ||||
|  | ||||
|     timer.Start(); | ||||
|     parallel_for(uint64_t lidx=0;lidx<lsites;lidx++){ | ||||
|       std::vector<RngStateType> tmp(RngStateCount); | ||||
| @@ -670,11 +656,6 @@ PARALLEL_CRITICAL | ||||
|       serial.SetState(tmp,0); | ||||
|     } | ||||
|  | ||||
|     std::cout << GridLogMessage << "RNG file checksum t " << std::hex << nersc_csum_tmp    << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file checksuma t " << std::hex << scidac_csuma_tmp << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file checksumb t " << std::hex << scidac_csumb_tmp << std::dec << std::endl; | ||||
|  | ||||
|  | ||||
|     nersc_csum   = nersc_csum   + nersc_csum_tmp; | ||||
|     scidac_csuma = scidac_csuma ^ scidac_csuma_tmp; | ||||
|     scidac_csumb = scidac_csumb ^ scidac_csumb_tmp; | ||||
| @@ -725,11 +706,6 @@ PARALLEL_CRITICAL | ||||
|  | ||||
|     IOobject(w,grid,iodata,file,offset,format,BINARYIO_WRITE|BINARYIO_LEXICOGRAPHIC, | ||||
| 	     nersc_csum,scidac_csuma,scidac_csumb); | ||||
|  | ||||
|     std::cout << GridLogMessage << "RNG file checksum " << std::hex << nersc_csum    << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file checksuma " << std::hex << scidac_csuma << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file checksumb " << std::hex << scidac_csumb << std::dec << std::endl; | ||||
|     | ||||
|     iodata.resize(1); | ||||
|     { | ||||
|       std::vector<RngStateType> tmp(RngStateCount); | ||||
| @@ -739,11 +715,6 @@ PARALLEL_CRITICAL | ||||
|     IOobject(w,grid,iodata,file,offset,format,BINARYIO_WRITE|BINARYIO_MASTER_APPEND, | ||||
| 	     nersc_csum_tmp,scidac_csuma_tmp,scidac_csumb_tmp); | ||||
|  | ||||
|     std::cout << GridLogMessage << "RNG file checksum t " << std::hex << nersc_csum_tmp    << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file checksuma t " << std::hex << scidac_csuma_tmp << std::dec << std::endl; | ||||
|     std::cout << GridLogMessage << "RNG file checksumb t " << std::hex << scidac_csumb_tmp << std::dec << std::endl; | ||||
|   | ||||
|  | ||||
|     nersc_csum   = nersc_csum   + nersc_csum_tmp; | ||||
|     scidac_csuma = scidac_csuma ^ scidac_csuma_tmp; | ||||
|     scidac_csumb = scidac_csumb ^ scidac_csumb_tmp; | ||||
|   | ||||
| @@ -182,6 +182,11 @@ class GridLimeReader : public BinaryIO { | ||||
|    { | ||||
|      filename= _filename; | ||||
|      File = fopen(filename.c_str(), "r"); | ||||
|      if (File == nullptr) | ||||
|      { | ||||
|        std::cerr << "cannot open file '" << filename << "'" << std::endl; | ||||
|        abort(); | ||||
|      } | ||||
|      LimeR = limeCreateReader(File); | ||||
|    } | ||||
|    ///////////////////////////////////////////// | ||||
| @@ -238,48 +243,6 @@ class GridLimeReader : public BinaryIO { | ||||
| 	// Verify checksums | ||||
| 	///////////////////////////////////////////// | ||||
| 	assert(scidacChecksumVerify(scidacChecksum_,scidac_csuma,scidac_csumb)==1); | ||||
|   std::cout << GridLogMessage<< " readLimeLatticeBinaryObject checksums match ! " <<std::endl; | ||||
| 	return; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   //////////////////////////////////////////// | ||||
|   // Read an RNG object and verify checksum | ||||
|   //////////////////////////////////////////// | ||||
|   void readLimeRNGObject(GridSerialRNG &sRNG, GridParallelRNG &pRNG,std::string record_name) | ||||
|   { | ||||
|     scidacChecksum scidacChecksum_; | ||||
|     uint32_t nersc_csum,scidac_csuma,scidac_csumb; | ||||
|  | ||||
|     while ( limeReaderNextRecord(LimeR) == LIME_SUCCESS ) {  | ||||
|       uint64_t file_bytes =limeReaderBytes(LimeR); | ||||
|  | ||||
|       //      std::cout << GridLogMessage << limeReaderType(LimeR) << " "<< file_bytes <<" bytes "<<std::endl; | ||||
|       //      std::cout << GridLogMessage<< " readLimeObject seeking "<<  record_name <<" found record :" <<limeReaderType(LimeR) <<std::endl; | ||||
|  | ||||
|       if ( !strncmp(limeReaderType(LimeR), record_name.c_str(),strlen(record_name.c_str()) )  ) { | ||||
|  | ||||
|         const int RngStateCount = GridSerialRNG::RngStateCount; | ||||
|         typedef std::array<typename GridSerialRNG::RngStateType,RngStateCount> RNGstate; | ||||
|  | ||||
| 	      uint64_t PayloadSize = sizeof(RNGstate) * (pRNG._grid->_gsites+1); | ||||
|  | ||||
|     	  assert(PayloadSize == file_bytes);// Must match or user error | ||||
|         uint64_t offset= ftello(File); | ||||
| 	    	std::cout << GridLogDebug << " ReadLatticeObject from offset "<<offset << std::endl; | ||||
| 	      BinaryIO::readRNG(sRNG, pRNG, filename, offset, nersc_csum,scidac_csuma,scidac_csumb); | ||||
|  | ||||
| 	      ///////////////////////////////////////////// | ||||
| 	      // Insist checksum is next record | ||||
| 	      ///////////////////////////////////////////// | ||||
| 	      readLimeObject(scidacChecksum_,std::string("scidacChecksum"),std::string(SCIDAC_CHECKSUM)); | ||||
|  | ||||
| 	      ///////////////////////////////////////////// | ||||
| 	      // Verify checksums | ||||
| 	      ///////////////////////////////////////////// | ||||
| 	      assert(scidacChecksumVerify(scidacChecksum_,scidac_csuma,scidac_csumb)==1); | ||||
|         std::cout << GridLogMessage<< " readLimeRNGObject checksums match ! " <<std::endl; | ||||
| 	return; | ||||
|       } | ||||
|     } | ||||
| @@ -471,78 +434,6 @@ class GridLimeWriter : public BinaryIO | ||||
|       writeLimeObject(0,1,checksum,std::string("scidacChecksum"),std::string(SCIDAC_CHECKSUM)); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  | ||||
|   //////////////////////////////////////////////////// | ||||
|   // Write an rng object and csum | ||||
|   // This routine is Collectively called by all nodes | ||||
|   // in communicator used by the field._grid | ||||
|   //////////////////////////////////////////////////// | ||||
|   void writeLimeRNGObject(GridSerialRNG &sRNG, GridParallelRNG &pRNG, std::string record_name) | ||||
|   { | ||||
|     GridBase *grid = pRNG._grid; | ||||
|     assert(boss_node == pRNG._grid->IsBoss() ); | ||||
|  | ||||
|     const int RngStateCount = GridSerialRNG::RngStateCount; | ||||
|     typedef std::array<typename GridSerialRNG::RngStateType,RngStateCount> RNGstate; | ||||
|  | ||||
|     //////////////////////////////////////////// | ||||
|     // Create record header | ||||
|     //////////////////////////////////////////// | ||||
|     int err; | ||||
|     uint32_t nersc_csum,scidac_csuma,scidac_csumb; | ||||
|     uint64_t PayloadSize = sizeof(RNGstate) * (grid->_gsites+1); | ||||
|     std::cout << GridLogDebug << "Computed payload size " << PayloadSize << std::endl; | ||||
|     if ( boss_node ) { | ||||
|       createLimeRecordHeader(record_name, 0, 0, PayloadSize); | ||||
|       fflush(File); | ||||
|     } | ||||
|      | ||||
|     //////////////////////////////////////////////// | ||||
|     // Check all nodes agree on file position | ||||
|     //////////////////////////////////////////////// | ||||
|     uint64_t offset1; | ||||
|     if ( boss_node ) { | ||||
|       offset1 = ftello(File);     | ||||
|     } | ||||
|     grid->Broadcast(0,(void *)&offset1,sizeof(offset1)); | ||||
|  | ||||
|     /////////////////////////////////////////// | ||||
|     // The above is collective. Write by other means into the binary record | ||||
|     /////////////////////////////////////////// | ||||
|     uint64_t offset = offset1; | ||||
|     BinaryIO::writeRNG(sRNG, pRNG,filename, offset, nersc_csum,scidac_csuma,scidac_csumb); | ||||
|  | ||||
|     /////////////////////////////////////////// | ||||
|     // Wind forward and close the record | ||||
|     /////////////////////////////////////////// | ||||
|     if ( boss_node ) { | ||||
|       fseek(File,0,SEEK_END);              | ||||
|       uint64_t offset2 = ftello(File);     | ||||
|       std::cout << GridLogDebug << " now at offset "<<offset2 << std::endl; | ||||
|       assert( (offset2-offset1) == PayloadSize); | ||||
|     } | ||||
|  | ||||
|     ///////////////////////////////////////////////////////////// | ||||
|     // Check MPI-2 I/O did what we expect to file | ||||
|     ///////////////////////////////////////////////////////////// | ||||
|  | ||||
|     if ( boss_node ) {  | ||||
|       err=limeWriterCloseRecord(LimeW);  assert(err>=0); | ||||
|     } | ||||
|     //////////////////////////////////////// | ||||
|     // Write checksum element, propagaing forward from the BinaryIO | ||||
|     // Always pair a checksum with a binary object, and close message | ||||
|     //////////////////////////////////////// | ||||
|     scidacChecksum checksum; | ||||
|     std::stringstream streama; streama << std::hex << scidac_csuma; | ||||
|     std::stringstream streamb; streamb << std::hex << scidac_csumb; | ||||
|     checksum.suma= streama.str(); | ||||
|     checksum.sumb= streamb.str(); | ||||
|     if ( boss_node ) {  | ||||
|       writeLimeObject(0,1,checksum,std::string("scidacChecksum"),std::string(SCIDAC_CHECKSUM)); | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| class ScidacWriter : public GridLimeWriter { | ||||
| @@ -559,27 +450,6 @@ class ScidacWriter : public GridLimeWriter { | ||||
|       writeLimeObject(0,1,_userFile,_userFile.SerialisableClassName(),std::string(SCIDAC_FILE_XML)); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void writeScidacRNGRecord(GridSerialRNG &sRNG, GridParallelRNG &pRNG)  | ||||
|   { | ||||
|     GridBase *grid = pRNG._grid; | ||||
|     FieldMetaData header; | ||||
|  | ||||
|     header.floating_point = "IEEE64BIG"; | ||||
|     header.checksum = 0x0; // Nersc checksum unused in ILDG, Scidac | ||||
|     GridMetaData(grid,header);  | ||||
|     MachineCharacteristics(header); | ||||
|      | ||||
|     ////////////////////////////////////////////// | ||||
|     // Fill the Lime file record by record | ||||
|     ////////////////////////////////////////////// | ||||
|     if ( this->boss_node ) { | ||||
|       writeLimeObject(1,0,header ,std::string("FieldMetaData"),std::string(GRID_FORMAT)); // Open message  | ||||
|     } | ||||
|     // Collective call | ||||
|     writeLimeRNGObject(sRNG, pRNG,std::string(ILDG_BINARY_DATA));      // Closes message with checksum | ||||
|   } | ||||
|    | ||||
|   //////////////////////////////////////////////// | ||||
|   // Write generic lattice field in scidac format | ||||
|   //////////////////////////////////////////////// | ||||
| @@ -608,7 +478,6 @@ class ScidacWriter : public GridLimeWriter { | ||||
|     // Collective call | ||||
|     writeLimeLatticeBinaryObject(field,std::string(ILDG_BINARY_DATA));      // Closes message with checksum | ||||
|   } | ||||
|    | ||||
| }; | ||||
|  | ||||
|  | ||||
| @@ -622,27 +491,8 @@ class ScidacReader : public GridLimeReader { | ||||
|      readLimeObject(_scidacFile,_scidacFile.SerialisableClassName(),std::string(SCIDAC_PRIVATE_FILE_XML)); | ||||
|      readLimeObject(_userFile,_userFile.SerialisableClassName(),std::string(SCIDAC_FILE_XML)); | ||||
|    } | ||||
|  | ||||
|   //////////////////////////////////////////////// | ||||
|   // Read RNGobject in scidac format | ||||
|   //////////////////////////////////////////////// | ||||
|   void readScidacRNGRecord(GridSerialRNG &sRNG, GridParallelRNG &pRNG)  | ||||
|   { | ||||
|     GridBase * grid = pRNG._grid; | ||||
|  | ||||
|     //////////////////////////////////////// | ||||
|     // fill the Grid header | ||||
|     //////////////////////////////////////// | ||||
|     FieldMetaData header; | ||||
|   | ||||
|     ////////////////////////////////////////////// | ||||
|     // Fill the Lime file record by record | ||||
|     ////////////////////////////////////////////// | ||||
|     readLimeObject(header ,std::string("FieldMetaData"),std::string(GRID_FORMAT)); // Open message  | ||||
|     readLimeRNGObject(sRNG, pRNG, std::string(ILDG_BINARY_DATA)); | ||||
|   } | ||||
|   //////////////////////////////////////////////// | ||||
|   // Read generic lattice field in scidac format | ||||
|   // Write generic lattice field in scidac format | ||||
|   //////////////////////////////////////////////// | ||||
|   template <class vobj, class userRecord> | ||||
|   void readScidacFieldRecord(Lattice<vobj> &field,userRecord &_userRecord)  | ||||
|   | ||||
| @@ -49,7 +49,8 @@ inline double usecond(void) { | ||||
|  | ||||
| typedef  std::chrono::system_clock          GridClock; | ||||
| typedef  std::chrono::time_point<GridClock> GridTimePoint; | ||||
| typedef  std::chrono::milliseconds          GridTime; | ||||
| typedef  std::chrono::milliseconds          GridMillisecs; | ||||
| typedef  std::chrono::microseconds          GridTime; | ||||
| typedef  std::chrono::microseconds          GridUsecs; | ||||
|  | ||||
| inline std::ostream& operator<< (std::ostream & stream, const std::chrono::milliseconds & time) | ||||
| @@ -57,6 +58,11 @@ inline std::ostream& operator<< (std::ostream & stream, const std::chrono::milli | ||||
|   stream << time.count()<<" ms"; | ||||
|   return stream; | ||||
| } | ||||
| inline std::ostream& operator<< (std::ostream & stream, const std::chrono::microseconds & time) | ||||
| { | ||||
|   stream << time.count()<<" usec"; | ||||
|   return stream; | ||||
| } | ||||
|   | ||||
| class GridStopWatch { | ||||
| private: | ||||
|   | ||||
| @@ -63,9 +63,12 @@ namespace Grid { | ||||
|       virtual RealD  M    (const FermionField &in, FermionField &out)=0; | ||||
|       virtual RealD  Mdag (const FermionField &in, FermionField &out)=0; | ||||
|  | ||||
|       // half checkerboard operaions | ||||
|       // Query the even even properties to make algorithmic decisions | ||||
|       virtual int    ConstEE(void) { return 1; }; // clover returns zero as EE depends on gauge field | ||||
|       virtual int    isTrivialEE(void) { return 0; }; | ||||
|       virtual RealD  Mass(void) {return 0.0;}; | ||||
|  | ||||
|       // half checkerboard operaions | ||||
|       virtual void   Meooe       (const FermionField &in, FermionField &out)=0; | ||||
|       virtual void   MeooeDag    (const FermionField &in, FermionField &out)=0; | ||||
|       virtual void   Mooee       (const FermionField &in, FermionField &out)=0; | ||||
|   | ||||
| @@ -765,6 +765,11 @@ class StaggeredImpl : public PeriodicGaugeImpl<GaugeImplTypes<S, Representation: | ||||
|       reg = memory; | ||||
|     } | ||||
|  | ||||
|     inline void InsertGaugeField(DoubledGaugeField &U_ds, | ||||
| 				 const GaugeLinkField &U,int mu) | ||||
|     { | ||||
|       PokeIndex<LorentzIndex>(U_ds, U, mu); | ||||
|     } | ||||
|     inline void DoubleStore(GridBase *GaugeGrid, | ||||
| 			    DoubledGaugeField &UUUds, // for Naik term | ||||
| 			    DoubledGaugeField &Uds, | ||||
| @@ -803,8 +808,10 @@ class StaggeredImpl : public PeriodicGaugeImpl<GaugeImplTypes<S, Representation: | ||||
| 	U    = U    *phases; | ||||
| 	Udag = Udag *phases; | ||||
|  | ||||
| 	PokeIndex<LorentzIndex>(Uds, U, mu); | ||||
| 	PokeIndex<LorentzIndex>(Uds, Udag, mu + 4); | ||||
| 	InsertGaugeField(Uds,U,mu); | ||||
| 	InsertGaugeField(Uds,Udag,mu+4); | ||||
| 	//	PokeIndex<LorentzIndex>(Uds, U, mu); | ||||
| 	//	PokeIndex<LorentzIndex>(Uds, Udag, mu + 4); | ||||
|  | ||||
| 	// 3 hop based on thin links. Crazy huh ? | ||||
| 	U  = PeekIndex<LorentzIndex>(Uthin, mu); | ||||
| @@ -816,8 +823,8 @@ class StaggeredImpl : public PeriodicGaugeImpl<GaugeImplTypes<S, Representation: | ||||
| 	UUU    = UUU    *phases; | ||||
| 	UUUdag = UUUdag *phases; | ||||
|  | ||||
| 	PokeIndex<LorentzIndex>(UUUds, UUU, mu); | ||||
| 	PokeIndex<LorentzIndex>(UUUds, UUUdag, mu+4); | ||||
| 	InsertGaugeField(UUUds,UUU,mu); | ||||
| 	InsertGaugeField(UUUds,UUUdag,mu+4); | ||||
|  | ||||
|       } | ||||
|     } | ||||
| @@ -910,6 +917,23 @@ class StaggeredImpl : public PeriodicGaugeImpl<GaugeImplTypes<S, Representation: | ||||
|       mac(&phi(), &UU(), &chi()); | ||||
|     } | ||||
|        | ||||
|     inline void InsertGaugeField(DoubledGaugeField &U_ds,const GaugeLinkField &U,int mu) | ||||
|     { | ||||
|       GridBase *GaugeGrid = U_ds._grid; | ||||
|       parallel_for (int lidx = 0; lidx < GaugeGrid->lSites(); lidx++) { | ||||
|  | ||||
| 	SiteScalarGaugeLink   ScalarU; | ||||
| 	SiteDoubledGaugeField ScalarUds; | ||||
| 	 | ||||
| 	std::vector<int> lcoor; | ||||
| 	GaugeGrid->LocalIndexToLocalCoor(lidx, lcoor); | ||||
| 	peekLocalSite(ScalarUds, U_ds, lcoor); | ||||
| 	 | ||||
| 	peekLocalSite(ScalarU, U, lcoor); | ||||
| 	ScalarUds(mu) = ScalarU(); | ||||
| 	 | ||||
|       } | ||||
|     } | ||||
|     inline void DoubleStore(GridBase *GaugeGrid, | ||||
| 			    DoubledGaugeField &UUUds, // for Naik term | ||||
| 			    DoubledGaugeField &Uds, | ||||
| @@ -951,23 +975,8 @@ class StaggeredImpl : public PeriodicGaugeImpl<GaugeImplTypes<S, Representation: | ||||
| 	U    = U    *phases; | ||||
| 	Udag = Udag *phases; | ||||
|  | ||||
|  | ||||
| 	for (int lidx = 0; lidx < GaugeGrid->lSites(); lidx++) { | ||||
| 	  SiteScalarGaugeLink   ScalarU; | ||||
| 	  SiteDoubledGaugeField ScalarUds; | ||||
| 	   | ||||
| 	  std::vector<int> lcoor; | ||||
| 	  GaugeGrid->LocalIndexToLocalCoor(lidx, lcoor); | ||||
| 	  peekLocalSite(ScalarUds, Uds, lcoor); | ||||
|  | ||||
| 	  peekLocalSite(ScalarU, U, lcoor); | ||||
| 	  ScalarUds(mu) = ScalarU(); | ||||
|  | ||||
| 	  peekLocalSite(ScalarU, Udag, lcoor); | ||||
| 	  ScalarUds(mu + 4) = ScalarU(); | ||||
|  | ||||
| 	  pokeLocalSite(ScalarUds, Uds, lcoor); | ||||
| 	} | ||||
| 	InsertGaugeField(Uds,U,mu); | ||||
| 	InsertGaugeField(Uds,Udag,mu+4); | ||||
|  | ||||
| 	// 3 hop based on thin links. Crazy huh ? | ||||
| 	U  = PeekIndex<LorentzIndex>(Uthin, mu); | ||||
| @@ -979,24 +988,8 @@ class StaggeredImpl : public PeriodicGaugeImpl<GaugeImplTypes<S, Representation: | ||||
| 	UUU    = UUU    *phases; | ||||
| 	UUUdag = UUUdag *phases; | ||||
|  | ||||
| 	for (int lidx = 0; lidx < GaugeGrid->lSites(); lidx++) { | ||||
|  | ||||
| 	  SiteScalarGaugeLink  ScalarU; | ||||
| 	  SiteDoubledGaugeField ScalarUds; | ||||
| 	   | ||||
| 	  std::vector<int> lcoor; | ||||
| 	  GaugeGrid->LocalIndexToLocalCoor(lidx, lcoor); | ||||
|        | ||||
| 	  peekLocalSite(ScalarUds, UUUds, lcoor); | ||||
|  | ||||
| 	  peekLocalSite(ScalarU, UUU, lcoor); | ||||
| 	  ScalarUds(mu) = ScalarU(); | ||||
|  | ||||
| 	  peekLocalSite(ScalarU, UUUdag, lcoor); | ||||
| 	  ScalarUds(mu + 4) = ScalarU(); | ||||
| 	   | ||||
| 	  pokeLocalSite(ScalarUds, UUUds, lcoor); | ||||
| 	} | ||||
| 	InsertGaugeField(UUUds,UUU,mu); | ||||
| 	InsertGaugeField(UUUds,UUUdag,mu+4); | ||||
|  | ||||
|       } | ||||
|     } | ||||
|   | ||||
| @@ -44,6 +44,7 @@ ImprovedStaggeredFermionStatic::displacements({1, 1, 1, 1, -1, -1, -1, -1, 3, 3, | ||||
| template <class Impl> | ||||
| ImprovedStaggeredFermion<Impl>::ImprovedStaggeredFermion(GridCartesian &Fgrid, GridRedBlackCartesian &Hgrid,  | ||||
| 							 RealD _mass, | ||||
| 							 RealD _c1, RealD _c2,RealD _u0, | ||||
| 							 const ImplParams &p) | ||||
|     : Kernels(p), | ||||
|       _grid(&Fgrid), | ||||
| @@ -62,6 +63,16 @@ ImprovedStaggeredFermion<Impl>::ImprovedStaggeredFermion(GridCartesian &Fgrid, G | ||||
|       UUUmuOdd(&Hgrid) , | ||||
|       _tmp(&Hgrid) | ||||
| { | ||||
|   int vol4; | ||||
|   int LLs=1; | ||||
|   c1=_c1; | ||||
|   c2=_c2; | ||||
|   u0=_u0; | ||||
|   vol4= _grid->oSites(); | ||||
|   Stencil.BuildSurfaceList(LLs,vol4); | ||||
|   vol4= _cbgrid->oSites(); | ||||
|   StencilEven.BuildSurfaceList(LLs,vol4); | ||||
|   StencilOdd.BuildSurfaceList(LLs,vol4); | ||||
| } | ||||
|  | ||||
| template <class Impl> | ||||
| @@ -69,22 +80,10 @@ ImprovedStaggeredFermion<Impl>::ImprovedStaggeredFermion(GaugeField &_Uthin, Gau | ||||
| 							 GridRedBlackCartesian &Hgrid, RealD _mass, | ||||
| 							 RealD _c1, RealD _c2,RealD _u0, | ||||
| 							 const ImplParams &p) | ||||
|   : ImprovedStaggeredFermion(Fgrid,Hgrid,_mass,p) | ||||
|   : ImprovedStaggeredFermion(Fgrid,Hgrid,_mass,_c1,_c2,_u0,p) | ||||
| { | ||||
|   c1=_c1; | ||||
|   c2=_c2; | ||||
|   u0=_u0; | ||||
|   ImportGauge(_Uthin,_Ufat); | ||||
| } | ||||
| template <class Impl> | ||||
| ImprovedStaggeredFermion<Impl>::ImprovedStaggeredFermion(GaugeField &_Uthin,GaugeField &_Utriple, GaugeField &_Ufat, GridCartesian &Fgrid, | ||||
| 							 GridRedBlackCartesian &Hgrid, RealD _mass, | ||||
| 							 const ImplParams &p) | ||||
|   : ImprovedStaggeredFermion(Fgrid,Hgrid,_mass,p) | ||||
| { | ||||
|   ImportGaugeSimple(_Utriple,_Ufat); | ||||
| } | ||||
|  | ||||
|  | ||||
|   //////////////////////////////////////////////////////////// | ||||
|   // Momentum space propagator should be  | ||||
| @@ -98,11 +97,6 @@ ImprovedStaggeredFermion<Impl>::ImprovedStaggeredFermion(GaugeField &_Uthin,Gaug | ||||
|   // of above link to implmement fourier based solver. | ||||
|   //////////////////////////////////////////////////////////// | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::ImportGauge(const GaugeField &_Uthin)  | ||||
| { | ||||
|   ImportGauge(_Uthin,_Uthin); | ||||
| }; | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::ImportGaugeSimple(const GaugeField &_Utriple,const GaugeField &_Ufat)  | ||||
| { | ||||
|   ///////////////////////////////////////////////////////////////// | ||||
| @@ -125,6 +119,20 @@ void ImprovedStaggeredFermion<Impl>::ImportGaugeSimple(const GaugeField &_Utripl | ||||
|     PokeIndex<LorentzIndex>(Umu, -U, mu+4); | ||||
|  | ||||
|   } | ||||
|   CopyGaugeCheckerboards(); | ||||
| } | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::ImportGaugeSimple(const DoubledGaugeField &_UUU,const DoubledGaugeField &_U)  | ||||
| { | ||||
|  | ||||
|   Umu   = _U; | ||||
|   UUUmu = _UUU; | ||||
|   CopyGaugeCheckerboards(); | ||||
| } | ||||
|  | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::CopyGaugeCheckerboards(void) | ||||
| { | ||||
|   pickCheckerboard(Even, UmuEven,  Umu); | ||||
|   pickCheckerboard(Odd,  UmuOdd ,  Umu); | ||||
|   pickCheckerboard(Even, UUUmuEven,UUUmu); | ||||
| @@ -160,10 +168,7 @@ void ImprovedStaggeredFermion<Impl>::ImportGauge(const GaugeField &_Uthin,const | ||||
|     PokeIndex<LorentzIndex>(UUUmu, U*(-0.5*c2/u0/u0/u0), mu+4); | ||||
|   } | ||||
|  | ||||
|   pickCheckerboard(Even, UmuEven, Umu); | ||||
|   pickCheckerboard(Odd,  UmuOdd , Umu); | ||||
|   pickCheckerboard(Even, UUUmuEven, UUUmu); | ||||
|   pickCheckerboard(Odd,   UUUmuOdd, UUUmu); | ||||
|   CopyGaugeCheckerboards(); | ||||
| } | ||||
|  | ||||
| ///////////////////////////// | ||||
| @@ -322,6 +327,7 @@ void ImprovedStaggeredFermion<Impl>::DhopDerivEO(GaugeField &mat, const FermionF | ||||
|  | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::Dhop(const FermionField &in, FermionField &out, int dag) { | ||||
|   DhopCalls+=2; | ||||
|   conformable(in._grid, _grid);  // verifies full grid | ||||
|   conformable(in._grid, out._grid); | ||||
|  | ||||
| @@ -332,6 +338,7 @@ void ImprovedStaggeredFermion<Impl>::Dhop(const FermionField &in, FermionField & | ||||
|  | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::DhopOE(const FermionField &in, FermionField &out, int dag) { | ||||
|   DhopCalls+=1; | ||||
|   conformable(in._grid, _cbgrid);    // verifies half grid | ||||
|   conformable(in._grid, out._grid);  // drops the cb check | ||||
|  | ||||
| @@ -343,6 +350,7 @@ void ImprovedStaggeredFermion<Impl>::DhopOE(const FermionField &in, FermionField | ||||
|  | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::DhopEO(const FermionField &in, FermionField &out, int dag) { | ||||
|   DhopCalls+=1; | ||||
|   conformable(in._grid, _cbgrid);    // verifies half grid | ||||
|   conformable(in._grid, out._grid);  // drops the cb check | ||||
|  | ||||
| @@ -374,25 +382,193 @@ void ImprovedStaggeredFermion<Impl>::DhopInternal(StencilImpl &st, LebesgueOrder | ||||
| 						  DoubledGaugeField &U, | ||||
| 						  DoubledGaugeField &UUU, | ||||
| 						  const FermionField &in, | ||||
| 						  FermionField &out, int dag) { | ||||
| 						  FermionField &out, int dag)  | ||||
| { | ||||
| #ifdef GRID_OMP | ||||
|   if ( StaggeredKernelsStatic::Comms == StaggeredKernelsStatic::CommsAndCompute ) | ||||
|     DhopInternalOverlappedComms(st,lo,U,UUU,in,out,dag); | ||||
|   else | ||||
| #endif | ||||
|     DhopInternalSerialComms(st,lo,U,UUU,in,out,dag); | ||||
| } | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::DhopInternalOverlappedComms(StencilImpl &st, LebesgueOrder &lo, | ||||
| 								 DoubledGaugeField &U, | ||||
| 								 DoubledGaugeField &UUU, | ||||
| 								 const FermionField &in, | ||||
| 								 FermionField &out, int dag)  | ||||
| { | ||||
| #ifdef GRID_OMP | ||||
|   Compressor compressor;  | ||||
|   int len =  U._grid->oSites(); | ||||
|   const int LLs =  1; | ||||
|  | ||||
|   DhopTotalTime   -= usecond(); | ||||
|  | ||||
|   DhopFaceTime    -= usecond(); | ||||
|   st.Prepare(); | ||||
|   st.HaloGather(in,compressor); | ||||
|   st.CommsMergeSHM(compressor); | ||||
|   DhopFaceTime    += usecond(); | ||||
|  | ||||
|   ////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|   // Ugly explicit thread mapping introduced for OPA reasons. | ||||
|   ////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|   DhopComputeTime    -= usecond(); | ||||
| #pragma omp parallel  | ||||
|   { | ||||
|     int tid = omp_get_thread_num(); | ||||
|     int nthreads = omp_get_num_threads(); | ||||
|     int ncomms = CartesianCommunicator::nCommThreads; | ||||
|     if (ncomms == -1) ncomms = 1; | ||||
|     assert(nthreads > ncomms); | ||||
|  | ||||
|     if (tid >= ncomms) { | ||||
|       nthreads -= ncomms; | ||||
|       int ttid  = tid - ncomms; | ||||
|       int n     = len; | ||||
|       int chunk = n / nthreads; | ||||
|       int rem   = n % nthreads; | ||||
|       int myblock, myn; | ||||
|       if (ttid < rem) { | ||||
|         myblock = ttid * chunk + ttid; | ||||
|         myn = chunk+1; | ||||
|       } else { | ||||
|         myblock = ttid*chunk + rem; | ||||
|         myn = chunk; | ||||
|       } | ||||
|  | ||||
|       // do the compute | ||||
|       if (dag == DaggerYes) { | ||||
|         for (int ss = myblock; ss < myblock+myn; ++ss) { | ||||
|           int sU = ss; | ||||
| 	  // Interior = 1; Exterior = 0; must implement for staggered | ||||
|           Kernels::DhopSiteDag(st,lo,U,UUU,st.CommBuf(),1,sU,in,out,1,0);  | ||||
|         } | ||||
|       } else { | ||||
|         for (int ss = myblock; ss < myblock+myn; ++ss) { | ||||
| 	  // Interior = 1; Exterior = 0; | ||||
|           int sU = ss; | ||||
|           Kernels::DhopSite(st,lo,U,UUU,st.CommBuf(),1,sU,in,out,1,0); | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       st.CommunicateThreaded(); | ||||
|     } | ||||
|   } | ||||
|   DhopComputeTime    += usecond(); | ||||
|  | ||||
|   // First to enter, last to leave timing | ||||
|   DhopFaceTime    -= usecond(); | ||||
|   st.CommsMerge(compressor); | ||||
|   DhopFaceTime    -= usecond(); | ||||
|  | ||||
|   DhopComputeTime2    -= usecond(); | ||||
|   if (dag == DaggerYes) { | ||||
|     int sz=st.surface_list.size(); | ||||
|     parallel_for (int ss = 0; ss < sz; ss++) { | ||||
|       int sU = st.surface_list[ss]; | ||||
|       Kernels::DhopSiteDag(st,lo,U,UUU,st.CommBuf(),1,sU,in,out,0,1); | ||||
|     } | ||||
|   } else { | ||||
|     int sz=st.surface_list.size(); | ||||
|     parallel_for (int ss = 0; ss < sz; ss++) { | ||||
|       int sU = st.surface_list[ss]; | ||||
|       Kernels::DhopSite(st,lo,U,UUU,st.CommBuf(),1,sU,in,out,0,1); | ||||
|     } | ||||
|   } | ||||
|   DhopComputeTime2    += usecond(); | ||||
| #else | ||||
|   assert(0); | ||||
| #endif | ||||
| } | ||||
|  | ||||
|  | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::DhopInternalSerialComms(StencilImpl &st, LebesgueOrder &lo, | ||||
| 							     DoubledGaugeField &U, | ||||
| 							     DoubledGaugeField &UUU, | ||||
| 							     const FermionField &in, | ||||
| 							     FermionField &out, int dag)  | ||||
| { | ||||
|   assert((dag == DaggerNo) || (dag == DaggerYes)); | ||||
|  | ||||
|   DhopTotalTime   -= usecond(); | ||||
|  | ||||
|   DhopCommTime    -= usecond(); | ||||
|   Compressor compressor; | ||||
|   st.HaloExchange(in, compressor); | ||||
|   DhopCommTime    += usecond(); | ||||
|  | ||||
|   DhopComputeTime -= usecond(); | ||||
|   if (dag == DaggerYes) { | ||||
|     PARALLEL_FOR_LOOP | ||||
|     for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|     parallel_for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|       Kernels::DhopSiteDag(st, lo, U, UUU, st.CommBuf(), 1, sss, in, out); | ||||
|     } | ||||
|   } else { | ||||
|     PARALLEL_FOR_LOOP | ||||
|     for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|     parallel_for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|       Kernels::DhopSite(st, lo, U, UUU, st.CommBuf(), 1, sss, in, out); | ||||
|     } | ||||
|   } | ||||
|   DhopComputeTime += usecond(); | ||||
|   DhopTotalTime   += usecond(); | ||||
| }; | ||||
|  | ||||
|   //////////////////////////////////////////////////////////////// | ||||
|   // Reporting | ||||
|   //////////////////////////////////////////////////////////////// | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::Report(void)  | ||||
| { | ||||
|   std::vector<int> latt = GridDefaultLatt();           | ||||
|   RealD volume = 1;  for(int mu=0;mu<Nd;mu++) volume=volume*latt[mu]; | ||||
|   RealD NP = _grid->_Nprocessors; | ||||
|   RealD NN = _grid->NodeCount(); | ||||
|  | ||||
|   std::cout << GridLogMessage << "#### Dhop calls report " << std::endl; | ||||
|  | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion Number of DhopEO Calls   : "  | ||||
| 	    << DhopCalls   << std::endl; | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion TotalTime   /Calls       : "  | ||||
| 	    << DhopTotalTime   / DhopCalls << " us" << std::endl; | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion CommTime    /Calls       : "  | ||||
| 	    << DhopCommTime    / DhopCalls << " us" << std::endl; | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion ComputeTime/Calls        : "  | ||||
| 	    << DhopComputeTime / DhopCalls << " us" << std::endl; | ||||
|  | ||||
|   // Average the compute time | ||||
|   _grid->GlobalSum(DhopComputeTime); | ||||
|   DhopComputeTime/=NP; | ||||
|  | ||||
|   RealD mflops = 1154*volume*DhopCalls/DhopComputeTime/2; // 2 for red black counting | ||||
|   std::cout << GridLogMessage << "Average mflops/s per call                : " << mflops << std::endl; | ||||
|   std::cout << GridLogMessage << "Average mflops/s per call per rank       : " << mflops/NP << std::endl; | ||||
|   std::cout << GridLogMessage << "Average mflops/s per call per node       : " << mflops/NN << std::endl; | ||||
|    | ||||
|   RealD Fullmflops = 1154*volume*DhopCalls/(DhopTotalTime)/2; // 2 for red black counting | ||||
|   std::cout << GridLogMessage << "Average mflops/s per call (full)         : " << Fullmflops << std::endl; | ||||
|   std::cout << GridLogMessage << "Average mflops/s per call per rank (full): " << Fullmflops/NP << std::endl; | ||||
|   std::cout << GridLogMessage << "Average mflops/s per call per node (full): " << Fullmflops/NN << std::endl; | ||||
|  | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion Stencil"    <<std::endl;  Stencil.Report(); | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion StencilEven"<<std::endl;  StencilEven.Report(); | ||||
|   std::cout << GridLogMessage << "ImprovedStaggeredFermion StencilOdd" <<std::endl;  StencilOdd.Report(); | ||||
| } | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion<Impl>::ZeroCounters(void)  | ||||
| { | ||||
|   DhopCalls       = 0; | ||||
|   DhopTotalTime   = 0; | ||||
|   DhopCommTime    = 0; | ||||
|   DhopComputeTime = 0; | ||||
|   DhopFaceTime    = 0; | ||||
|  | ||||
|   Stencil.ZeroCounters(); | ||||
|   StencilEven.ZeroCounters(); | ||||
|   StencilOdd.ZeroCounters(); | ||||
| } | ||||
|  | ||||
|  | ||||
| ////////////////////////////////////////////////////////  | ||||
| // Conserved current - not yet implemented. | ||||
| //////////////////////////////////////////////////////// | ||||
|   | ||||
| @@ -49,6 +49,18 @@ class ImprovedStaggeredFermion : public StaggeredKernels<Impl>, public ImprovedS | ||||
|   FermionField _tmp; | ||||
|   FermionField &tmp(void) { return _tmp; } | ||||
|  | ||||
|   //////////////////////////////////////// | ||||
|   // Performance monitoring | ||||
|   //////////////////////////////////////// | ||||
|   void Report(void); | ||||
|   void ZeroCounters(void); | ||||
|   double DhopTotalTime; | ||||
|   double DhopCalls; | ||||
|   double DhopCommTime; | ||||
|   double DhopComputeTime; | ||||
|   double DhopComputeTime2; | ||||
|   double DhopFaceTime; | ||||
|  | ||||
|   /////////////////////////////////////////////////////////////// | ||||
|   // Implement the abstract base | ||||
|   /////////////////////////////////////////////////////////////// | ||||
| @@ -105,25 +117,34 @@ class ImprovedStaggeredFermion : public StaggeredKernels<Impl>, public ImprovedS | ||||
|  | ||||
|   void DhopInternal(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U,DoubledGaugeField &UUU, | ||||
|                     const FermionField &in, FermionField &out, int dag); | ||||
|   void DhopInternalSerialComms(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U,DoubledGaugeField &UUU, | ||||
|                     const FermionField &in, FermionField &out, int dag); | ||||
|   void DhopInternalOverlappedComms(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U,DoubledGaugeField &UUU, | ||||
|                     const FermionField &in, FermionField &out, int dag); | ||||
|  | ||||
|   // Constructor | ||||
|   ////////////////////////////////////////////////////////////////////////// | ||||
|   // Grid own interface Constructor | ||||
|   ////////////////////////////////////////////////////////////////////////// | ||||
|   ImprovedStaggeredFermion(GaugeField &_Uthin, GaugeField &_Ufat, GridCartesian &Fgrid, | ||||
| 			   GridRedBlackCartesian &Hgrid, RealD _mass, | ||||
| 			   RealD _c1=9.0/8.0, RealD _c2=-1.0/24.0,RealD _u0=1.0, | ||||
| 			   const ImplParams &p = ImplParams()); | ||||
|  | ||||
|   ImprovedStaggeredFermion(GaugeField &_Uthin, GaugeField &_Utriple, GaugeField &_Ufat, GridCartesian &Fgrid, | ||||
| 			   GridRedBlackCartesian &Hgrid, RealD _mass, | ||||
| 			   RealD _c1, RealD _c2,RealD _u0, | ||||
| 			   const ImplParams &p = ImplParams()); | ||||
|  | ||||
|   ////////////////////////////////////////////////////////////////////////// | ||||
|   // MILC constructor no gauge fields | ||||
|   ////////////////////////////////////////////////////////////////////////// | ||||
|   ImprovedStaggeredFermion(GridCartesian &Fgrid, GridRedBlackCartesian &Hgrid, RealD _mass, | ||||
| 			   RealD _c1=1.0, RealD _c2=1.0,RealD _u0=1.0, | ||||
| 			   const ImplParams &p = ImplParams()); | ||||
|  | ||||
|  | ||||
|   // DoubleStore impl dependent | ||||
|   void ImportGaugeSimple(const GaugeField &_Utriple, const GaugeField &_Ufat); | ||||
|   void ImportGauge      (const GaugeField &_Uthin ) { assert(0); } | ||||
|   void ImportGauge      (const GaugeField &_Uthin  ,const GaugeField &_Ufat); | ||||
|   void ImportGauge(const GaugeField &_Uthin); | ||||
|   void ImportGaugeSimple(const GaugeField &_UUU    ,const GaugeField &_U); | ||||
|   void ImportGaugeSimple(const DoubledGaugeField &_UUU,const DoubledGaugeField &_U); | ||||
|   DoubledGaugeField &GetU(void)   { return Umu ; } ; | ||||
|   DoubledGaugeField &GetUUU(void) { return UUUmu; }; | ||||
|   void CopyGaugeCheckerboards(void); | ||||
|  | ||||
|   /////////////////////////////////////////////////////////////// | ||||
|   // Data members require to support the functionality | ||||
| @@ -132,7 +153,8 @@ class ImprovedStaggeredFermion : public StaggeredKernels<Impl>, public ImprovedS | ||||
|   //    protected: | ||||
|  public: | ||||
|   // any other parameters of action ??? | ||||
|  | ||||
|   virtual int   isTrivialEE(void) { return 1; }; | ||||
|   virtual RealD Mass(void) { return mass; } | ||||
|   RealD mass; | ||||
|   RealD u0; | ||||
|   RealD c1; | ||||
|   | ||||
| @@ -41,8 +41,7 @@ ImprovedStaggeredFermion5DStatic::displacements({1, 1, 1, 1, -1, -1, -1, -1, 3, | ||||
|  | ||||
|   // 5d lattice for DWF. | ||||
| template<class Impl> | ||||
| ImprovedStaggeredFermion5D<Impl>::ImprovedStaggeredFermion5D(GaugeField &_Uthin,GaugeField &_Ufat, | ||||
| 							     GridCartesian         &FiveDimGrid, | ||||
| ImprovedStaggeredFermion5D<Impl>::ImprovedStaggeredFermion5D(GridCartesian         &FiveDimGrid, | ||||
| 							     GridRedBlackCartesian &FiveDimRedBlackGrid, | ||||
| 							     GridCartesian         &FourDimGrid, | ||||
| 							     GridRedBlackCartesian &FourDimRedBlackGrid, | ||||
| @@ -121,16 +120,74 @@ ImprovedStaggeredFermion5D<Impl>::ImprovedStaggeredFermion5D(GaugeField &_Uthin, | ||||
|     assert(FiveDimGrid._simd_layout[0]        ==1); | ||||
|  | ||||
|   } | ||||
|   int LLs = FiveDimGrid._rdimensions[0]; | ||||
|   int vol4= FourDimGrid.oSites(); | ||||
|   Stencil.BuildSurfaceList(LLs,vol4); | ||||
|  | ||||
|   // Allocate the required comms buffer | ||||
|   vol4=FourDimRedBlackGrid.oSites(); | ||||
|   StencilEven.BuildSurfaceList(LLs,vol4); | ||||
|   StencilOdd.BuildSurfaceList(LLs,vol4); | ||||
| } | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::CopyGaugeCheckerboards(void) | ||||
| { | ||||
|   pickCheckerboard(Even, UmuEven,  Umu); | ||||
|   pickCheckerboard(Odd,  UmuOdd ,  Umu); | ||||
|   pickCheckerboard(Even, UUUmuEven,UUUmu); | ||||
|   pickCheckerboard(Odd,  UUUmuOdd, UUUmu); | ||||
| } | ||||
| template<class Impl> | ||||
| ImprovedStaggeredFermion5D<Impl>::ImprovedStaggeredFermion5D(GaugeField &_Uthin,GaugeField &_Ufat, | ||||
| 							     GridCartesian         &FiveDimGrid, | ||||
| 							     GridRedBlackCartesian &FiveDimRedBlackGrid, | ||||
| 							     GridCartesian         &FourDimGrid, | ||||
| 							     GridRedBlackCartesian &FourDimRedBlackGrid, | ||||
| 							     RealD _mass, | ||||
| 							     RealD _c1,RealD _c2, RealD _u0, | ||||
| 							     const ImplParams &p) : | ||||
|   ImprovedStaggeredFermion5D(FiveDimGrid,FiveDimRedBlackGrid, | ||||
| 			     FourDimGrid,FourDimRedBlackGrid, | ||||
| 			     _mass,_c1,_c2,_u0,p) | ||||
| { | ||||
|   ImportGauge(_Uthin,_Ufat); | ||||
| } | ||||
|  | ||||
| /////////////////////////////////////////////////// | ||||
| // For MILC use; pass three link U's and 1 link U | ||||
| /////////////////////////////////////////////////// | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::ImportGauge(const GaugeField &_Uthin)  | ||||
| void ImprovedStaggeredFermion5D<Impl>::ImportGaugeSimple(const GaugeField &_Utriple,const GaugeField &_Ufat)  | ||||
| { | ||||
|   ImportGauge(_Uthin,_Uthin); | ||||
| }; | ||||
|   ///////////////////////////////////////////////////////////////// | ||||
|   // Trivial import; phases and fattening and such like preapplied | ||||
|   ///////////////////////////////////////////////////////////////// | ||||
|   for (int mu = 0; mu < Nd; mu++) { | ||||
|  | ||||
|     auto U = PeekIndex<LorentzIndex>(_Utriple, mu); | ||||
|     Impl::InsertGaugeField(UUUmu,U,mu); | ||||
|  | ||||
|     U = adj( Cshift(U, mu, -3)); | ||||
|     Impl::InsertGaugeField(UUUmu,-U,mu+4); | ||||
|  | ||||
|     U = PeekIndex<LorentzIndex>(_Ufat, mu); | ||||
|     Impl::InsertGaugeField(Umu,U,mu); | ||||
|  | ||||
|     U = adj( Cshift(U, mu, -1)); | ||||
|     Impl::InsertGaugeField(Umu,-U,mu+4); | ||||
|  | ||||
|   } | ||||
|   CopyGaugeCheckerboards(); | ||||
| } | ||||
| template <class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::ImportGaugeSimple(const DoubledGaugeField &_UUU,const DoubledGaugeField &_U)  | ||||
| { | ||||
|   ///////////////////////////////////////////////////////////////// | ||||
|   // Trivial import; phases and fattening and such like preapplied | ||||
|   ///////////////////////////////////////////////////////////////// | ||||
|   Umu   = _U; | ||||
|   UUUmu = _UUU; | ||||
|   CopyGaugeCheckerboards(); | ||||
| } | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::ImportGauge(const GaugeField &_Uthin,const GaugeField &_Ufat) | ||||
| { | ||||
| @@ -159,10 +216,7 @@ void ImprovedStaggeredFermion5D<Impl>::ImportGauge(const GaugeField &_Uthin,cons | ||||
|     PokeIndex<LorentzIndex>(UUUmu, U*(-0.5*c2/u0/u0/u0), mu+4); | ||||
|   } | ||||
|  | ||||
|   pickCheckerboard(Even, UmuEven, Umu); | ||||
|   pickCheckerboard(Odd,  UmuOdd , Umu); | ||||
|   pickCheckerboard(Even, UUUmuEven, UUUmu); | ||||
|   pickCheckerboard(Odd,  UUUmuOdd, UUUmu); | ||||
|   CopyGaugeCheckerboards(); | ||||
| } | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::DhopDir(const FermionField &in, FermionField &out,int dir5,int disp) | ||||
| @@ -223,6 +277,162 @@ void ImprovedStaggeredFermion5D<Impl>::DhopDerivOE(GaugeField &mat, | ||||
|   assert(0); | ||||
| } | ||||
|  | ||||
| /*CHANGE */ | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::DhopInternal(StencilImpl & st, LebesgueOrder &lo, | ||||
| 						    DoubledGaugeField & U,DoubledGaugeField & UUU, | ||||
| 						    const FermionField &in, FermionField &out,int dag) | ||||
| { | ||||
| #ifdef GRID_OMP | ||||
|   if ( StaggeredKernelsStatic::Comms == StaggeredKernelsStatic::CommsAndCompute ) | ||||
|     DhopInternalOverlappedComms(st,lo,U,UUU,in,out,dag); | ||||
|   else | ||||
| #endif | ||||
|     DhopInternalSerialComms(st,lo,U,UUU,in,out,dag); | ||||
| } | ||||
|  | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::DhopInternalOverlappedComms(StencilImpl & st, LebesgueOrder &lo, | ||||
| 								   DoubledGaugeField & U,DoubledGaugeField & UUU, | ||||
| 								   const FermionField &in, FermionField &out,int dag) | ||||
| { | ||||
| #ifdef GRID_OMP | ||||
|   //  assert((dag==DaggerNo) ||(dag==DaggerYes)); | ||||
|  | ||||
|   Compressor compressor;  | ||||
|  | ||||
|   int LLs = in._grid->_rdimensions[0]; | ||||
|   int len =  U._grid->oSites(); | ||||
|  | ||||
|   DhopFaceTime-=usecond(); | ||||
|   st.Prepare(); | ||||
|   st.HaloGather(in,compressor); | ||||
|   //  st.HaloExchangeOptGather(in,compressor); // Wilson compressor | ||||
|   st.CommsMergeSHM(compressor);// Could do this inside parallel region overlapped with comms | ||||
|   DhopFaceTime+=usecond(); | ||||
|  | ||||
|   double ctime=0; | ||||
|   double ptime=0; | ||||
|  | ||||
|   ////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|   // Ugly explicit thread mapping introduced for OPA reasons. | ||||
|   ////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
| #pragma omp parallel reduction(max:ctime) reduction(max:ptime) | ||||
|   { | ||||
|     int tid = omp_get_thread_num(); | ||||
|     int nthreads = omp_get_num_threads(); | ||||
|     int ncomms = CartesianCommunicator::nCommThreads; | ||||
|     if (ncomms == -1) ncomms = 1; | ||||
|     assert(nthreads > ncomms); | ||||
|     if (tid >= ncomms) { | ||||
|       double start = usecond(); | ||||
|       nthreads -= ncomms; | ||||
|       int ttid  = tid - ncomms; | ||||
|       int n     = U._grid->oSites(); // 4d vol | ||||
|       int chunk = n / nthreads; | ||||
|       int rem   = n % nthreads; | ||||
|       int myblock, myn; | ||||
|       if (ttid < rem) { | ||||
|         myblock = ttid * chunk + ttid; | ||||
|         myn = chunk+1; | ||||
|       } else { | ||||
|         myblock = ttid*chunk + rem; | ||||
|         myn = chunk; | ||||
|       } | ||||
|  | ||||
|       // do the compute | ||||
|       if (dag == DaggerYes) { | ||||
|         for (int ss = myblock; ss < myblock+myn; ++ss) { | ||||
|           int sU = ss; | ||||
| 	  // Interior = 1; Exterior = 0; must implement for staggered | ||||
|           Kernels::DhopSiteDag(st,lo,U,UUU,st.CommBuf(),LLs,sU,in,out,1,0); //<--------- | ||||
|         } | ||||
|       } else { | ||||
|         for (int ss = myblock; ss < myblock+myn; ++ss) { | ||||
| 	  // Interior = 1; Exterior = 0; | ||||
|           int sU = ss; | ||||
|           Kernels::DhopSite(st,lo,U,UUU,st.CommBuf(),LLs,sU,in,out,1,0); //<------------ | ||||
|         } | ||||
|       } | ||||
|         ptime = usecond() - start; | ||||
|     } else { | ||||
|       double start = usecond(); | ||||
|       st.CommunicateThreaded(); | ||||
|       ctime = usecond() - start; | ||||
|     } | ||||
|   } | ||||
|   DhopCommTime += ctime; | ||||
|   DhopComputeTime+=ptime; | ||||
|  | ||||
|   // First to enter, last to leave timing | ||||
|   st.CollateThreads(); | ||||
|  | ||||
|   DhopFaceTime-=usecond(); | ||||
|   st.CommsMerge(compressor); | ||||
|   DhopFaceTime+=usecond(); | ||||
|  | ||||
|   DhopComputeTime2-=usecond(); | ||||
|   if (dag == DaggerYes) { | ||||
|     int sz=st.surface_list.size(); | ||||
|     parallel_for (int ss = 0; ss < sz; ss++) { | ||||
|       int sU = st.surface_list[ss]; | ||||
|       Kernels::DhopSiteDag(st,lo,U,UUU,st.CommBuf(),LLs,sU,in,out,0,1); //<---------- | ||||
|     } | ||||
|   } else { | ||||
|     int sz=st.surface_list.size(); | ||||
|     parallel_for (int ss = 0; ss < sz; ss++) { | ||||
|       int sU = st.surface_list[ss]; | ||||
|       Kernels::DhopSite(st,lo,U,UUU,st.CommBuf(),LLs,sU,in,out,0,1);//<---------- | ||||
|     } | ||||
|   } | ||||
|   DhopComputeTime2+=usecond(); | ||||
| #else | ||||
|   assert(0); | ||||
| #endif | ||||
|  | ||||
| } | ||||
|  | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::DhopInternalSerialComms(StencilImpl & st, LebesgueOrder &lo, | ||||
| 						    DoubledGaugeField & U,DoubledGaugeField & UUU, | ||||
| 						    const FermionField &in, FermionField &out,int dag) | ||||
| { | ||||
|   Compressor compressor; | ||||
|   int LLs = in._grid->_rdimensions[0]; | ||||
|  | ||||
|  | ||||
|  | ||||
|  //double t1=usecond(); | ||||
|   DhopTotalTime -= usecond(); | ||||
|   DhopCommTime -= usecond(); | ||||
|   st.HaloExchange(in,compressor); | ||||
|   DhopCommTime += usecond(); | ||||
|    | ||||
|   DhopComputeTime -= usecond(); | ||||
|   // Dhop takes the 4d grid from U, and makes a 5d index for fermion | ||||
|   if (dag == DaggerYes) { | ||||
|     parallel_for (int ss = 0; ss < U._grid->oSites(); ss++) { | ||||
|       int sU=ss; | ||||
|       Kernels::DhopSiteDag(st, lo, U, UUU, st.CommBuf(), LLs, sU,in, out); | ||||
|     } | ||||
|   } else { | ||||
|     parallel_for (int ss = 0; ss < U._grid->oSites(); ss++) { | ||||
|       int sU=ss; | ||||
|       Kernels::DhopSite(st,lo,U,UUU,st.CommBuf(),LLs,sU,in,out); | ||||
|     } | ||||
|   } | ||||
|   DhopComputeTime += usecond(); | ||||
|   DhopTotalTime   += usecond(); | ||||
|  //double t2=usecond(); | ||||
|  //std::cout << __FILE__ << " " << __func__  << " Total Time " << DhopTotalTime << std::endl; | ||||
|  //std::cout << __FILE__ << " " << __func__  << " Total Time Org " << t2-t1 << std::endl; | ||||
|  //std::cout << __FILE__ << " " << __func__  << " Comml Time " << DhopCommTime << std::endl; | ||||
|  //std::cout << __FILE__ << " " << __func__  << " Compute Time " << DhopComputeTime << std::endl; | ||||
|  | ||||
| } | ||||
| /*CHANGE END*/ | ||||
|  | ||||
| /* ORG | ||||
| template<class Impl> | ||||
| void ImprovedStaggeredFermion5D<Impl>::DhopInternal(StencilImpl & st, LebesgueOrder &lo, | ||||
| 						    DoubledGaugeField & U,DoubledGaugeField & UUU, | ||||
| @@ -254,6 +464,7 @@ void ImprovedStaggeredFermion5D<Impl>::DhopInternal(StencilImpl & st, LebesgueOr | ||||
|   DhopComputeTime += usecond(); | ||||
|   DhopTotalTime   += usecond(); | ||||
| } | ||||
| */ | ||||
|  | ||||
|  | ||||
| template<class Impl> | ||||
| @@ -336,6 +547,9 @@ void ImprovedStaggeredFermion5D<Impl>::ZeroCounters(void) | ||||
|   DhopTotalTime    = 0; | ||||
|   DhopCommTime    = 0; | ||||
|   DhopComputeTime = 0; | ||||
|   DhopFaceTime    = 0; | ||||
|  | ||||
|  | ||||
|   Stencil.ZeroCounters(); | ||||
|   StencilEven.ZeroCounters(); | ||||
|   StencilOdd.ZeroCounters(); | ||||
|   | ||||
| @@ -64,6 +64,8 @@ namespace QCD { | ||||
|       double DhopCalls; | ||||
|       double DhopCommTime; | ||||
|       double DhopComputeTime; | ||||
|       double DhopComputeTime2; | ||||
|       double DhopFaceTime; | ||||
|  | ||||
|       /////////////////////////////////////////////////////////////// | ||||
|       // Implement the abstract base | ||||
| @@ -119,7 +121,27 @@ namespace QCD { | ||||
| 		      FermionField &out, | ||||
| 		      int dag); | ||||
|      | ||||
|     void DhopInternalOverlappedComms(StencilImpl & st, | ||||
| 		      LebesgueOrder &lo, | ||||
| 		      DoubledGaugeField &U, | ||||
| 		      DoubledGaugeField &UUU, | ||||
| 		      const FermionField &in,  | ||||
| 		      FermionField &out, | ||||
| 		      int dag); | ||||
|  | ||||
|     void DhopInternalSerialComms(StencilImpl & st, | ||||
| 		      LebesgueOrder &lo, | ||||
| 		      DoubledGaugeField &U, | ||||
| 		      DoubledGaugeField &UUU, | ||||
| 		      const FermionField &in,  | ||||
| 		      FermionField &out, | ||||
| 		      int dag); | ||||
|      | ||||
|      | ||||
|     // Constructors | ||||
|     //////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|     // Grid internal interface -- Thin link and fat link, with coefficients | ||||
|     //////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|     ImprovedStaggeredFermion5D(GaugeField &_Uthin, | ||||
| 			       GaugeField &_Ufat, | ||||
| 			       GridCartesian         &FiveDimGrid, | ||||
| @@ -127,18 +149,38 @@ namespace QCD { | ||||
| 			       GridCartesian         &FourDimGrid, | ||||
| 			       GridRedBlackCartesian &FourDimRedBlackGrid, | ||||
| 			       double _mass, | ||||
| 			       RealD _c1=9.0/8.0, RealD _c2=-1.0/24.0,RealD _u0=1.0, | ||||
| 			       RealD _c1, RealD _c2,RealD _u0, | ||||
| 			       const ImplParams &p= ImplParams()); | ||||
|     //////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|     // MILC constructor ; triple links, no rescale factors; must be externally pre multiplied | ||||
|     //////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|     ImprovedStaggeredFermion5D(GridCartesian         &FiveDimGrid, | ||||
| 			       GridRedBlackCartesian &FiveDimRedBlackGrid, | ||||
| 			       GridCartesian         &FourDimGrid, | ||||
| 			       GridRedBlackCartesian &FourDimRedBlackGrid, | ||||
| 			       double _mass, | ||||
| 			       RealD _c1=1.0, RealD _c2=1.0,RealD _u0=1.0, | ||||
| 			       const ImplParams &p= ImplParams()); | ||||
|  | ||||
|     // DoubleStore | ||||
|     void ImportGauge(const GaugeField &_U); | ||||
|     // DoubleStore gauge field in operator | ||||
|     void ImportGauge      (const GaugeField &_Uthin ) { assert(0); } | ||||
|     void ImportGauge      (const GaugeField &_Uthin  ,const GaugeField &_Ufat); | ||||
|     void ImportGaugeSimple(const GaugeField &_UUU,const GaugeField &_U); | ||||
|     void ImportGaugeSimple(const DoubledGaugeField &_UUU,const DoubledGaugeField &_U); | ||||
|     // Give a reference; can be used to do an assignment or copy back out after import | ||||
|     // if Carleton wants to cache them and not use the ImportSimple | ||||
|     DoubledGaugeField &GetU(void)   { return Umu ; } ; | ||||
|     DoubledGaugeField &GetUUU(void) { return UUUmu; }; | ||||
|     void CopyGaugeCheckerboards(void); | ||||
|      | ||||
|     /////////////////////////////////////////////////////////////// | ||||
|     // Data members require to support the functionality | ||||
|     /////////////////////////////////////////////////////////////// | ||||
|   public: | ||||
|  | ||||
|     virtual int   isTrivialEE(void) { return 1; }; | ||||
|     virtual RealD Mass(void) { return mass; } | ||||
|      | ||||
|     GridBase *_FourDimGrid; | ||||
|     GridBase *_FourDimRedBlackGrid; | ||||
|     GridBase *_FiveDimGrid; | ||||
|   | ||||
| @@ -32,223 +32,241 @@ namespace Grid { | ||||
| namespace QCD { | ||||
|  | ||||
| int StaggeredKernelsStatic::Opt= StaggeredKernelsStatic::OptGeneric; | ||||
| int StaggeredKernelsStatic::Comms = StaggeredKernelsStatic::CommsAndCompute; | ||||
|  | ||||
| #define GENERIC_STENCIL_LEG(U,Dir,skew,multLink)		\ | ||||
|   SE = st.GetEntry(ptype, Dir+skew, sF);			\ | ||||
|   if (SE->_is_local ) {						\ | ||||
|     if (SE->_permute) {						\ | ||||
|       chi_p = χ						\ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype);		\ | ||||
|     } else {							\ | ||||
|       chi_p = &in._odata[SE->_offset];				\ | ||||
|     }								\ | ||||
|   } else {							\ | ||||
|     chi_p = &buf[SE->_offset];					\ | ||||
|   }								\ | ||||
|   multLink(Uchi, U._odata[sU], *chi_p, Dir);			 | ||||
|  | ||||
| #define GENERIC_STENCIL_LEG_INT(U,Dir,skew,multLink)		\ | ||||
|   SE = st.GetEntry(ptype, Dir+skew, sF);			\ | ||||
|   if (SE->_is_local ) {						\ | ||||
|     if (SE->_permute) {						\ | ||||
|       chi_p = χ						\ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype);		\ | ||||
|     } else {							\ | ||||
|       chi_p = &in._odata[SE->_offset];				\ | ||||
|     }								\ | ||||
|   } else if ( st.same_node[Dir] ) {				\ | ||||
|     chi_p = &buf[SE->_offset];					\ | ||||
|   }								\ | ||||
|   if (SE->_is_local || st.same_node[Dir] ) {			\ | ||||
|     multLink(Uchi, U._odata[sU], *chi_p, Dir);			\ | ||||
|   } | ||||
|  | ||||
| #define GENERIC_STENCIL_LEG_EXT(U,Dir,skew,multLink)		\ | ||||
|   SE = st.GetEntry(ptype, Dir+skew, sF);			\ | ||||
|   if ((!SE->_is_local) && (!st.same_node[Dir]) ) {		\ | ||||
|     nmu++;							\ | ||||
|     chi_p = &buf[SE->_offset];					\ | ||||
|     multLink(Uchi, U._odata[sU], *chi_p, Dir);			\ | ||||
|   } | ||||
|  | ||||
| template <class Impl> | ||||
| StaggeredKernels<Impl>::StaggeredKernels(const ImplParams &p) : Base(p){}; | ||||
|  | ||||
| //////////////////////////////////////////// | ||||
| //////////////////////////////////////////////////////////////////////////////////// | ||||
| // Generic implementation; move to different file? | ||||
| //////////////////////////////////////////// | ||||
|  | ||||
| // Int, Ext, Int+Ext cases for comms overlap | ||||
| //////////////////////////////////////////////////////////////////////////////////// | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteDepth(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, | ||||
| 					   SiteSpinor *buf, int sF, | ||||
| 					   int sU, const FermionField &in, SiteSpinor &out,int threeLink) { | ||||
| void StaggeredKernels<Impl>::DhopSiteGeneric(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 					     DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 					     SiteSpinor *buf, int LLs, int sU,  | ||||
| 					     const FermionField &in, FermionField &out, int dag) { | ||||
|   const SiteSpinor *chi_p; | ||||
|   SiteSpinor chi; | ||||
|   SiteSpinor Uchi; | ||||
|   StencilEntry *SE; | ||||
|   int ptype; | ||||
|   int skew = 0; | ||||
|   if (threeLink) skew=8; | ||||
|   /////////////////////////// | ||||
|   // Xp | ||||
|   /////////////////////////// | ||||
|   int skew; | ||||
|  | ||||
|   SE = st.GetEntry(ptype, Xp+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=LLs*sU+s; | ||||
|     skew = 0; | ||||
|     GENERIC_STENCIL_LEG(U,Xp,skew,Impl::multLink); | ||||
|     GENERIC_STENCIL_LEG(U,Yp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(U,Zp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(U,Tp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(U,Xm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(U,Ym,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(U,Zm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(U,Tm,skew,Impl::multLinkAdd); | ||||
|     skew=8; | ||||
|     GENERIC_STENCIL_LEG(UUU,Xp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Yp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Zp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Tp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Xm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Ym,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Zm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG(UUU,Tm,skew,Impl::multLinkAdd); | ||||
|     if ( dag ) {  | ||||
|       Uchi = - Uchi; | ||||
|     }  | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|     vstream(out._odata[sF], Uchi); | ||||
|   } | ||||
|   Impl::multLink(Uchi, U._odata[sU], *chi_p, Xp); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Yp | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Yp+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Yp); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Zp | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Zp+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Zp); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Tp | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Tp+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Tp); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Xm | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Xm+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Xm); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Ym | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Ym+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Ym); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Zm | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Zm+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Zm); | ||||
|  | ||||
|   /////////////////////////// | ||||
|   // Tm | ||||
|   /////////////////////////// | ||||
|   SE = st.GetEntry(ptype, Tm+skew, sF); | ||||
|   if (SE->_is_local) { | ||||
|     if (SE->_permute) { | ||||
|       chi_p = χ | ||||
|       permute(chi,  in._odata[SE->_offset], ptype); | ||||
|     } else { | ||||
|       chi_p = &in._odata[SE->_offset]; | ||||
|     } | ||||
|   } else { | ||||
|     chi_p = &buf[SE->_offset]; | ||||
|   } | ||||
|   Impl::multLinkAdd(Uchi, U._odata[sU], *chi_p, Tm); | ||||
|  | ||||
|   vstream(out, Uchi); | ||||
| }; | ||||
|  | ||||
|   /////////////////////////////////////////////////// | ||||
|   // Only contributions from interior of our node | ||||
|   /////////////////////////////////////////////////// | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteGenericInt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 						DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 						SiteSpinor *buf, int LLs, int sU,  | ||||
| 						const FermionField &in, FermionField &out,int dag) { | ||||
|   const SiteSpinor *chi_p; | ||||
|   SiteSpinor chi; | ||||
|   SiteSpinor Uchi; | ||||
|   StencilEntry *SE; | ||||
|   int ptype; | ||||
|   int skew ; | ||||
|  | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=LLs*sU+s; | ||||
|     skew = 0; | ||||
|     Uchi=zero; | ||||
|     GENERIC_STENCIL_LEG_INT(U,Xp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Yp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Zp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Tp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Xm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Ym,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Zm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(U,Tm,skew,Impl::multLinkAdd); | ||||
|     skew=8; | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Xp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Yp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Zp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Tp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Xm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Ym,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Zm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_INT(UUU,Tm,skew,Impl::multLinkAdd); | ||||
|     if ( dag ) { | ||||
|       Uchi = - Uchi; | ||||
|     } | ||||
|     vstream(out._odata[sF], Uchi); | ||||
|   } | ||||
| }; | ||||
|  | ||||
|  | ||||
|   /////////////////////////////////////////////////// | ||||
|   // Only contributions from exterior of our node | ||||
|   /////////////////////////////////////////////////// | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteGenericExt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 						DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 						SiteSpinor *buf, int LLs, int sU, | ||||
| 						const FermionField &in, FermionField &out,int dag) { | ||||
|   const SiteSpinor *chi_p; | ||||
|   SiteSpinor chi; | ||||
|   SiteSpinor Uchi; | ||||
|   StencilEntry *SE; | ||||
|   int ptype; | ||||
|   int nmu=0; | ||||
|   int skew ; | ||||
|  | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=LLs*sU+s; | ||||
|     skew = 0; | ||||
|     Uchi=zero; | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Xp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Yp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Zp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Tp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Xm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Ym,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Zm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(U,Tm,skew,Impl::multLinkAdd); | ||||
|     skew=8; | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Xp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Yp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Zp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Tp,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Xm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Ym,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Zm,skew,Impl::multLinkAdd); | ||||
|     GENERIC_STENCIL_LEG_EXT(UUU,Tm,skew,Impl::multLinkAdd); | ||||
|  | ||||
|     if ( nmu ) {  | ||||
|       if ( dag ) {  | ||||
| 	out._odata[sF] = out._odata[sF] - Uchi; | ||||
|       } else {  | ||||
| 	out._odata[sF] = out._odata[sF] + Uchi; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| //////////////////////////////////////////////////////////////////////////////////// | ||||
| // Driving / wrapping routine to select right kernel | ||||
| //////////////////////////////////////////////////////////////////////////////////// | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteDag(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 					 SiteSpinor *buf, int LLs, int sU, | ||||
| 						  const FermionField &in, FermionField &out) { | ||||
|   SiteSpinor naik; | ||||
|   SiteSpinor naive; | ||||
|   int oneLink  =0; | ||||
|   int threeLink=1; | ||||
| 					 const FermionField &in, FermionField &out, | ||||
| 					 int interior,int exterior) | ||||
| { | ||||
|   int dag=1; | ||||
|   switch(Opt) { | ||||
| #ifdef AVX512 | ||||
|   //FIXME; move the sign into the Asm routine | ||||
|   case OptInlineAsm: | ||||
|     DhopSiteAsm(st,lo,U,UUU,buf,LLs,sU,in,out); | ||||
|     for(int s=0;s<LLs;s++) { | ||||
|       int sF=s+LLs*sU; | ||||
|       out._odata[sF]=-out._odata[sF]; | ||||
|     } | ||||
|     break; | ||||
| #endif | ||||
|   case OptHandUnroll: | ||||
|     DhopSiteHand(st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     break; | ||||
|   case OptGeneric: | ||||
|     for(int s=0;s<LLs;s++){ | ||||
|        int sF=s+LLs*sU; | ||||
|        DhopSiteDepth(st,lo,U,buf,sF,sU,in,naive,oneLink); | ||||
|        DhopSiteDepth(st,lo,UUU,buf,sF,sU,in,naik,threeLink); | ||||
|        out._odata[sF] =-naive-naik;  | ||||
|      } | ||||
|     break; | ||||
|   default: | ||||
|     std::cout<<"Oops Opt = "<<Opt<<std::endl; | ||||
|     assert(0); | ||||
|     break; | ||||
|   } | ||||
|   DhopSite(st,lo,U,UUU,buf,LLs,sU,in,out,dag,interior,exterior); | ||||
| }; | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSite(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 				      SiteSpinor *buf, int LLs, int sU, | ||||
| 				      const FermionField &in, FermionField &out, | ||||
| 				      int interior,int exterior) | ||||
| { | ||||
|   int dag=0; | ||||
|   DhopSite(st,lo,U,UUU,buf,LLs,sU,in,out,dag,interior,exterior); | ||||
| }; | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSite(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 				      SiteSpinor *buf, int LLs, | ||||
| 				      int sU, const FermionField &in, FermionField &out)  | ||||
| 				      int sU, const FermionField &in, FermionField &out, | ||||
| 				      int dag,int interior,int exterior)  | ||||
| { | ||||
|   int oneLink  =0; | ||||
|   int threeLink=1; | ||||
|   SiteSpinor naik; | ||||
|   SiteSpinor naive; | ||||
|   int dag=0; | ||||
|   switch(Opt) { | ||||
| #ifdef AVX512 | ||||
|   case OptInlineAsm: | ||||
|     DhopSiteAsm(st,lo,U,UUU,buf,LLs,sU,in,out); | ||||
|     if ( interior && exterior ) { | ||||
|       DhopSiteAsm(st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } else {  | ||||
|       std::cout << GridLogError << "Cannot overlap comms and compute with Staggered assembly"<<std::endl; | ||||
|       assert(0); | ||||
|     } | ||||
|     break; | ||||
| #endif | ||||
|   case OptHandUnroll: | ||||
|     if ( interior && exterior ) { | ||||
|       DhopSiteHand   (st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } else if ( interior ) { | ||||
|       DhopSiteHandInt(st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } else if ( exterior ) { | ||||
|       DhopSiteHandExt(st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } | ||||
|     break; | ||||
|   case OptGeneric: | ||||
|     for(int s=0;s<LLs;s++){ | ||||
|       int sF=LLs*sU+s; | ||||
|       //      assert(sF<in._odata.size()); | ||||
|       //      assert(sU< U._odata.size()); | ||||
|       //      assert(sF>=0);      assert(sU>=0); | ||||
|       DhopSiteDepth(st,lo,U,buf,sF,sU,in,naive,oneLink); | ||||
|       DhopSiteDepth(st,lo,UUU,buf,sF,sU,in,naik,threeLink); | ||||
|       out._odata[sF] =naive+naik; | ||||
|     if ( interior && exterior ) { | ||||
|       DhopSiteGeneric   (st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } else if ( interior ) { | ||||
|       DhopSiteGenericInt(st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } else if ( exterior ) { | ||||
|       DhopSiteGenericExt(st,lo,U,UUU,buf,LLs,sU,in,out,dag); | ||||
|     } | ||||
|     break; | ||||
|   default: | ||||
|   | ||||
| @@ -38,8 +38,9 @@ namespace QCD { | ||||
| class StaggeredKernelsStatic {  | ||||
|  public: | ||||
|   enum { OptGeneric, OptHandUnroll, OptInlineAsm }; | ||||
|   // S-direction is INNERMOST and takes no part in the parity. | ||||
|   static int Opt;  // these are a temporary hack | ||||
|   enum { CommsAndCompute, CommsThenCompute }; | ||||
|   static int Opt; | ||||
|   static int Comms; | ||||
| }; | ||||
|   | ||||
| template<class Impl> class StaggeredKernels : public FermionOperator<Impl> , public StaggeredKernelsStatic {  | ||||
| @@ -53,24 +54,62 @@ public: | ||||
|    void DhopDir(StencilImpl &st, DoubledGaugeField &U, DoubledGaugeField &UUU, SiteSpinor * buf, | ||||
| 		      int sF, int sU, const FermionField &in, FermionField &out, int dir,int disp); | ||||
|  | ||||
|    void DhopSiteDepth(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, SiteSpinor * buf, | ||||
| 		     int sF, int sU, const FermionField &in, SiteSpinor &out,int threeLink); | ||||
|    /////////////////////////////////////////////////////////////////////////////////////// | ||||
|    // Generic Nc kernels | ||||
|    /////////////////////////////////////////////////////////////////////////////////////// | ||||
|    void DhopSiteGeneric(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 			DoubledGaugeField &U, DoubledGaugeField &UUU,  | ||||
| 			SiteSpinor * buf, int LLs, int sU,  | ||||
| 			const FermionField &in, FermionField &out,int dag); | ||||
|    void DhopSiteGenericInt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 			   DoubledGaugeField &U, DoubledGaugeField &UUU,  | ||||
| 			   SiteSpinor * buf, int LLs, int sU,  | ||||
| 			   const FermionField &in, FermionField &out,int dag); | ||||
|    void DhopSiteGenericExt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 			   DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 			   SiteSpinor * buf, int LLs, int sU,  | ||||
| 			   const FermionField &in, FermionField &out,int dag); | ||||
|  | ||||
|    /////////////////////////////////////////////////////////////////////////////////////// | ||||
|    // Nc=3 specific kernels | ||||
|    /////////////////////////////////////////////////////////////////////////////////////// | ||||
|    void DhopSiteHand(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 		     DoubledGaugeField &U,DoubledGaugeField &UUU,  | ||||
| 		     SiteSpinor * buf, int LLs, int sU,  | ||||
| 		     const FermionField &in, FermionField &out,int dag); | ||||
|    void DhopSiteHandInt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 			DoubledGaugeField &U,DoubledGaugeField &UUU,  | ||||
| 			SiteSpinor * buf, int LLs, int sU,  | ||||
| 			const FermionField &in, FermionField &out,int dag); | ||||
|    void DhopSiteHandExt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 			DoubledGaugeField &U,DoubledGaugeField &UUU,  | ||||
| 			SiteSpinor * buf, int LLs, int sU,  | ||||
| 			const FermionField &in, FermionField &out,int dag); | ||||
|  | ||||
|    void DhopSiteDepthHand(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, SiteSpinor * buf, | ||||
| 		     int sF, int sU, const FermionField &in, SiteSpinor&out,int threeLink); | ||||
|    /////////////////////////////////////////////////////////////////////////////////////// | ||||
|    // Asm Nc=3 specific kernels | ||||
|    /////////////////////////////////////////////////////////////////////////////////////// | ||||
|    void DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 		    DoubledGaugeField &U,DoubledGaugeField &UUU,  | ||||
| 		    SiteSpinor * buf, int LLs, int sU,  | ||||
| 		    const FermionField &in, FermionField &out,int dag); | ||||
|    /////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|    // Generic interface; fan out to right routine | ||||
|    /////////////////////////////////////////////////////////////////////////////////////////////////// | ||||
|    void DhopSite(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 		 DoubledGaugeField &U, DoubledGaugeField &UUU,  | ||||
| 		 SiteSpinor * buf, int LLs, int sU, | ||||
| 		 const FermionField &in, FermionField &out, int interior=1,int exterior=1); | ||||
|  | ||||
|    void DhopSiteHand(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, DoubledGaugeField &UUU,SiteSpinor * buf, | ||||
| 		     int LLs, int sU, const FermionField &in, FermionField &out, int dag); | ||||
|    void DhopSiteDag(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 		    DoubledGaugeField &U, DoubledGaugeField &UUU,  | ||||
| 		    SiteSpinor * buf, int LLs, int sU, | ||||
| 		    const FermionField &in, FermionField &out, int interior=1,int exterior=1); | ||||
|  | ||||
|    void DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U,DoubledGaugeField &UUU, SiteSpinor * buf, | ||||
| 			 int LLs, int sU, const FermionField &in, FermionField &out); | ||||
|        | ||||
|    void DhopSite(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, DoubledGaugeField &UUU, SiteSpinor * buf, | ||||
| 		int sF, int sU, const FermionField &in, FermionField &out); | ||||
|  | ||||
|    void DhopSiteDag(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, DoubledGaugeField &UUU, SiteSpinor *buf,  | ||||
|                    int LLs, int sU, const FermionField &in, FermionField &out); | ||||
|    void DhopSite(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 		 DoubledGaugeField &U, DoubledGaugeField &UUU,  | ||||
| 		 SiteSpinor * buf, int LLs, int sU, | ||||
| 		 const FermionField &in, FermionField &out, int dag, int interior,int exterior); | ||||
|    | ||||
| public: | ||||
|  | ||||
|   | ||||
| @@ -560,6 +560,27 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | ||||
|        VSTORE(2,%0,pUChi_02)					\ | ||||
|        : : "r" (out) : "memory" ); | ||||
|  | ||||
| #define nREDUCE(out)							\ | ||||
|   asm (									\ | ||||
|        VADD(UChi_00,UChi_10,UChi_00)					\ | ||||
|        VADD(UChi_01,UChi_11,UChi_01)					\ | ||||
|        VADD(UChi_02,UChi_12,UChi_02)					\ | ||||
|        VADD(UChi_30,UChi_20,UChi_30)					\ | ||||
|        VADD(UChi_31,UChi_21,UChi_31)					\ | ||||
|        VADD(UChi_32,UChi_22,UChi_32)					\ | ||||
|        VADD(UChi_00,UChi_30,UChi_00)					\ | ||||
|        VADD(UChi_01,UChi_31,UChi_01)					\ | ||||
|        VADD(UChi_02,UChi_32,UChi_02)				);	\ | ||||
|   asm (VZERO(Chi_00)							\ | ||||
|        VSUB(UChi_00,Chi_00,UChi_00)					\ | ||||
|        VSUB(UChi_01,Chi_00,UChi_01)					\ | ||||
|        VSUB(UChi_02,Chi_00,UChi_02)				);	\ | ||||
|   asm (								\ | ||||
|        VSTORE(0,%0,pUChi_00)					\ | ||||
|        VSTORE(1,%0,pUChi_01)					\ | ||||
|        VSTORE(2,%0,pUChi_02)					\ | ||||
|        : : "r" (out) : "memory" ); | ||||
|  | ||||
| #define REDUCEa(out)					\ | ||||
|   asm (							\ | ||||
|   VADD(UChi_00,UChi_10,UChi_00)				\ | ||||
| @@ -571,6 +592,22 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | ||||
|        VSTORE(2,%0,pUChi_02)						\ | ||||
|        : : "r" (out) : "memory" ); | ||||
|  | ||||
| // FIXME is sign right in the VSUB ? | ||||
| #define nREDUCEa(out)					\ | ||||
|   asm (							\ | ||||
|   VADD(UChi_00,UChi_10,UChi_00)				\ | ||||
|   VADD(UChi_01,UChi_11,UChi_01)				\ | ||||
|   VADD(UChi_02,UChi_12,UChi_02)	);			\ | ||||
|   asm (VZERO(Chi_00)							\ | ||||
|        VSUB(UChi_00,Chi_00,UChi_00)					\ | ||||
|        VSUB(UChi_01,Chi_00,UChi_01)					\ | ||||
|        VSUB(UChi_02,Chi_00,UChi_02)				);	\ | ||||
|   asm (									\ | ||||
|        VSTORE(0,%0,pUChi_00)				\ | ||||
|        VSTORE(1,%0,pUChi_01)				\ | ||||
|        VSTORE(2,%0,pUChi_02)				\ | ||||
|        : : "r" (out) : "memory" ); | ||||
|  | ||||
| #define PERMUTE_DIR(dir)			\ | ||||
|       permute##dir(Chi_0,Chi_0);\ | ||||
|       permute##dir(Chi_1,Chi_1);\ | ||||
| @@ -581,10 +618,9 @@ namespace QCD { | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 					 DoubledGaugeField &U, | ||||
| 					 DoubledGaugeField &UUU, | ||||
| 					 SiteSpinor *buf, int LLs, | ||||
| 					 int sU, const FermionField &in, FermionField &out)  | ||||
| 					 DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 					 SiteSpinor *buf, int LLs, int sU,  | ||||
| 					 const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
|   assert(0); | ||||
| }; | ||||
| @@ -645,10 +681,9 @@ void StaggeredKernels<Impl>::DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo, | ||||
|   // This is the single precision 5th direction vectorised kernel | ||||
| #include <simd/Intel512single.h> | ||||
| template <> void StaggeredKernels<StaggeredVec5dImplF>::DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 								    DoubledGaugeField &U, | ||||
| 								    DoubledGaugeField &UUU, | ||||
| 								    SiteSpinor *buf, int LLs, | ||||
| 								    int sU, const FermionField &in, FermionField &out)  | ||||
| 								    DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 								    SiteSpinor *buf, int LLs, int sU,  | ||||
| 								    const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
| #ifdef AVX512 | ||||
|   uint64_t gauge0,gauge1,gauge2,gauge3; | ||||
| @@ -685,8 +720,12 @@ template <> void StaggeredKernels<StaggeredVec5dImplF>::DhopSiteAsm(StencilImpl | ||||
|     MULT_ADD_LS(gauge0,gauge1,gauge2,gauge3); | ||||
|  | ||||
|     addr0 = (uint64_t) &out._odata[sF]; | ||||
|     if ( dag ) { | ||||
|       nREDUCE(addr0); | ||||
|     } else {  | ||||
|       REDUCE(addr0); | ||||
|     } | ||||
|    } | ||||
| #else  | ||||
|     assert(0); | ||||
| #endif | ||||
| @@ -695,10 +734,9 @@ template <> void StaggeredKernels<StaggeredVec5dImplF>::DhopSiteAsm(StencilImpl | ||||
|  | ||||
| #include <simd/Intel512double.h> | ||||
| template <> void StaggeredKernels<StaggeredVec5dImplD>::DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 								    DoubledGaugeField &U, | ||||
| 								    DoubledGaugeField &UUU, | ||||
| 								    SiteSpinor *buf, int LLs, | ||||
| 								    int sU, const FermionField &in, FermionField &out)  | ||||
| 								    DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 								    SiteSpinor *buf, int LLs, int sU,  | ||||
| 								    const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
| #ifdef AVX512 | ||||
|   uint64_t gauge0,gauge1,gauge2,gauge3; | ||||
| @@ -734,8 +772,12 @@ template <> void StaggeredKernels<StaggeredVec5dImplD>::DhopSiteAsm(StencilImpl | ||||
|     MULT_ADD_LS(gauge0,gauge1,gauge2,gauge3); | ||||
|  | ||||
|     addr0 = (uint64_t) &out._odata[sF]; | ||||
|     if ( dag ) { | ||||
|       nREDUCE(addr0); | ||||
|     } else {  | ||||
|       REDUCE(addr0); | ||||
|     } | ||||
|   } | ||||
| #else  | ||||
|   assert(0); | ||||
| #endif | ||||
| @@ -776,10 +818,9 @@ template <> void StaggeredKernels<StaggeredVec5dImplD>::DhopSiteAsm(StencilImpl | ||||
|  | ||||
| #include <simd/Intel512single.h> | ||||
| template <> void StaggeredKernels<StaggeredImplF>::DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 								    DoubledGaugeField &U, | ||||
| 								    DoubledGaugeField &UUU, | ||||
| 								    SiteSpinor *buf, int LLs, | ||||
| 								    int sU, const FermionField &in, FermionField &out)  | ||||
| 							       DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 							       SiteSpinor *buf, int LLs, int sU,  | ||||
| 							       const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
| #ifdef AVX512 | ||||
|   uint64_t gauge0,gauge1,gauge2,gauge3; | ||||
| @@ -832,8 +873,12 @@ template <> void StaggeredKernels<StaggeredImplF>::DhopSiteAsm(StencilImpl &st, | ||||
|     MULT_ADD_XYZT(gauge2,gauge3);   | ||||
|  | ||||
|     addr0 = (uint64_t) &out._odata[sF]; | ||||
|     if ( dag ) {  | ||||
|       nREDUCEa(addr0); | ||||
|     } else {  | ||||
|       REDUCEa(addr0); | ||||
|     } | ||||
|   } | ||||
| #else  | ||||
|   assert(0); | ||||
| #endif | ||||
| @@ -841,10 +886,9 @@ template <> void StaggeredKernels<StaggeredImplF>::DhopSiteAsm(StencilImpl &st, | ||||
|  | ||||
| #include <simd/Intel512double.h> | ||||
| template <> void StaggeredKernels<StaggeredImplD>::DhopSiteAsm(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 								    DoubledGaugeField &U, | ||||
| 								    DoubledGaugeField &UUU, | ||||
| 								    SiteSpinor *buf, int LLs, | ||||
| 								    int sU, const FermionField &in, FermionField &out)  | ||||
| 							       DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 							       SiteSpinor *buf, int LLs, int sU,  | ||||
| 							       const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
| #ifdef AVX512 | ||||
|   uint64_t gauge0,gauge1,gauge2,gauge3; | ||||
| @@ -897,8 +941,12 @@ template <> void StaggeredKernels<StaggeredImplD>::DhopSiteAsm(StencilImpl &st, | ||||
|     MULT_ADD_XYZT(gauge2,gauge3);   | ||||
|      | ||||
|     addr0 = (uint64_t) &out._odata[sF]; | ||||
|     if ( dag ) { | ||||
|       nREDUCEa(addr0); | ||||
|     } else {  | ||||
|       REDUCEa(addr0); | ||||
|     } | ||||
|   } | ||||
| #else  | ||||
|   assert(0); | ||||
| #endif | ||||
| @@ -909,7 +957,7 @@ template <> void StaggeredKernels<StaggeredImplD>::DhopSiteAsm(StencilImpl &st, | ||||
| 				  DoubledGaugeField &U,			\ | ||||
| 				  DoubledGaugeField &UUU,		\ | ||||
| 				  SiteSpinor *buf, int LLs,		\ | ||||
| 				  int sU, const FermionField &in, FermionField &out); | ||||
| 				  int sU, const FermionField &in, FermionField &out,int dag); | ||||
|  | ||||
| KERNEL_INSTANTIATE(StaggeredKernels,DhopSiteAsm,StaggeredImplD); | ||||
| KERNEL_INSTANTIATE(StaggeredKernels,DhopSiteAsm,StaggeredImplF); | ||||
|   | ||||
| @@ -28,7 +28,6 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | ||||
|     /*  END LEGAL */ | ||||
| #include <Grid.h> | ||||
|  | ||||
| #define REGISTER | ||||
|  | ||||
| #define LOAD_CHI(b)		\ | ||||
|   const SiteSpinor & ref (b[offset]);	\ | ||||
| @@ -59,7 +58,7 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | ||||
|     UChi ## _1 += U_12*Chi_2;\ | ||||
|     UChi ## _2 += U_22*Chi_2; | ||||
|  | ||||
| #define MULT_ADD(A,UChi)				\ | ||||
| #define MULT_ADD(U,A,UChi)			\ | ||||
|   auto & ref(U._odata[sU](A));			\ | ||||
|    Impl::loadLinkElement(U_00,ref()(0,0));      \ | ||||
|    Impl::loadLinkElement(U_10,ref()(1,0));      \ | ||||
| @@ -86,237 +85,315 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | ||||
|   permute##dir(Chi_1,Chi_1);			\ | ||||
|   permute##dir(Chi_2,Chi_2); | ||||
|  | ||||
|  | ||||
| #define HAND_STENCIL_LEG_BASE(Dir,Perm,skew)	\ | ||||
|   SE=st.GetEntry(ptype,Dir+skew,sF);	\ | ||||
|   offset = SE->_offset;			\ | ||||
|   local  = SE->_is_local;		\ | ||||
|   perm   = SE->_permute;		\ | ||||
|   if ( local ) {						\ | ||||
|     LOAD_CHI(in._odata);					\ | ||||
|     if ( perm) {						\ | ||||
|       PERMUTE_DIR(Perm);					\ | ||||
|     }								\ | ||||
|   } else {							\ | ||||
|     LOAD_CHI(buf);						\ | ||||
|   }								 | ||||
|  | ||||
| #define HAND_STENCIL_LEG_BEGIN(Dir,Perm,skew,even)		\ | ||||
|   HAND_STENCIL_LEG_BASE(Dir,Perm,skew)				\ | ||||
|   {								\ | ||||
|     MULT(Dir,even);						\ | ||||
|   } | ||||
|  | ||||
| #define HAND_STENCIL_LEG(U,Dir,Perm,skew,even)			\ | ||||
|   HAND_STENCIL_LEG_BASE(Dir,Perm,skew)				\ | ||||
|   {								\ | ||||
|     MULT_ADD(U,Dir,even);					\ | ||||
|   } | ||||
|  | ||||
|  | ||||
|  | ||||
| #define HAND_STENCIL_LEG_INT(U,Dir,Perm,skew,even)	\ | ||||
|   SE=st.GetEntry(ptype,Dir+skew,sF);			\ | ||||
|   offset = SE->_offset;					\ | ||||
|   local  = SE->_is_local;				\ | ||||
|   perm   = SE->_permute;				\ | ||||
|   if ( local ) {					\ | ||||
|     LOAD_CHI(in._odata);				\ | ||||
|     if ( perm) {					\ | ||||
|       PERMUTE_DIR(Perm);				\ | ||||
|     }							\ | ||||
|   } else if ( st.same_node[Dir] ) {			\ | ||||
|     LOAD_CHI(buf);					\ | ||||
|   }							\ | ||||
|   if (SE->_is_local || st.same_node[Dir] ) {		\ | ||||
|     MULT_ADD(U,Dir,even);				\ | ||||
|   } | ||||
|  | ||||
| #define HAND_STENCIL_LEG_EXT(U,Dir,Perm,skew,even)	\ | ||||
|   SE=st.GetEntry(ptype,Dir+skew,sF);			\ | ||||
|   offset = SE->_offset;					\ | ||||
|   local  = SE->_is_local;				\ | ||||
|   perm   = SE->_permute;				\ | ||||
|   if ((!SE->_is_local) && (!st.same_node[Dir]) ) {		\ | ||||
|     nmu++;							\ | ||||
|     { LOAD_CHI(buf);	  }					\ | ||||
|     { MULT_ADD(U,Dir,even); }					\ | ||||
|   }								 | ||||
|  | ||||
| namespace Grid { | ||||
| namespace QCD { | ||||
|  | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteHand(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U,DoubledGaugeField &UUU, | ||||
| 					  SiteSpinor *buf, int LLs, | ||||
| 					  int sU, const FermionField &in, FermionField &out, int dag)  | ||||
| { | ||||
|   SiteSpinor naik;  | ||||
|   SiteSpinor naive; | ||||
|   int oneLink  =0; | ||||
|   int threeLink=1; | ||||
|   int skew(0); | ||||
|   Real scale(1.0); | ||||
|    | ||||
|   if(dag) scale = -1.0; | ||||
|    | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=s+LLs*sU; | ||||
|     DhopSiteDepthHand(st,lo,U,buf,sF,sU,in,naive,oneLink); | ||||
|     DhopSiteDepthHand(st,lo,UUU,buf,sF,sU,in,naik,threeLink); | ||||
|     out._odata[sF] =scale*(naive+naik); | ||||
|   } | ||||
| } | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteDepthHand(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, | ||||
| 					       SiteSpinor *buf, int sF, | ||||
| 					       int sU, const FermionField &in, SiteSpinor &out,int threeLink)  | ||||
| void StaggeredKernels<Impl>::DhopSiteHand(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 					  DoubledGaugeField &U,DoubledGaugeField &UUU, | ||||
| 					  SiteSpinor *buf, int LLs, int sU,  | ||||
| 					  const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
|   typedef typename Simd::scalar_type S; | ||||
|   typedef typename Simd::vector_type V; | ||||
|  | ||||
|   REGISTER Simd even_0; // 12 regs on knc | ||||
|   REGISTER Simd even_1; | ||||
|   REGISTER Simd even_2; | ||||
|   REGISTER Simd odd_0; // 12 regs on knc | ||||
|   REGISTER Simd odd_1; | ||||
|   REGISTER Simd odd_2; | ||||
|   Simd even_0; // 12 regs on knc | ||||
|   Simd even_1; | ||||
|   Simd even_2; | ||||
|   Simd odd_0; // 12 regs on knc | ||||
|   Simd odd_1; | ||||
|   Simd odd_2; | ||||
|  | ||||
|   REGISTER Simd Chi_0;    // two spinor; 6 regs | ||||
|   REGISTER Simd Chi_1; | ||||
|   REGISTER Simd Chi_2; | ||||
|   Simd Chi_0;    // two spinor; 6 regs | ||||
|   Simd Chi_1; | ||||
|   Simd Chi_2; | ||||
|    | ||||
|   REGISTER Simd U_00;  // two rows of U matrix | ||||
|   REGISTER Simd U_10; | ||||
|   REGISTER Simd U_20;   | ||||
|   REGISTER Simd U_01; | ||||
|   REGISTER Simd U_11; | ||||
|   REGISTER Simd U_21;  // 2 reg left. | ||||
|   REGISTER Simd U_02; | ||||
|   REGISTER Simd U_12; | ||||
|   REGISTER Simd U_22;  | ||||
|  | ||||
|   int skew = 0; | ||||
|   if (threeLink) skew=8; | ||||
|   Simd U_00;  // two rows of U matrix | ||||
|   Simd U_10; | ||||
|   Simd U_20;   | ||||
|   Simd U_01; | ||||
|   Simd U_11; | ||||
|   Simd U_21;  // 2 reg left. | ||||
|   Simd U_02; | ||||
|   Simd U_12; | ||||
|   Simd U_22;  | ||||
|  | ||||
|   SiteSpinor result; | ||||
|   int offset,local,perm, ptype; | ||||
|  | ||||
|   StencilEntry *SE; | ||||
|   int skew; | ||||
|  | ||||
|   // Xp | ||||
|   SE=st.GetEntry(ptype,Xp+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=s+LLs*sU; | ||||
|  | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(3); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|     skew = 0; | ||||
|     HAND_STENCIL_LEG_BEGIN(Xp,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_BEGIN(Yp,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG      (U,Zp,1,skew,even);   | ||||
|     HAND_STENCIL_LEG      (U,Tp,0,skew,odd);   | ||||
|     HAND_STENCIL_LEG      (U,Xm,3,skew,even);   | ||||
|     HAND_STENCIL_LEG      (U,Ym,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG      (U,Zm,1,skew,even);   | ||||
|     HAND_STENCIL_LEG      (U,Tm,0,skew,odd);   | ||||
|     skew = 8; | ||||
|     HAND_STENCIL_LEG(UUU,Xp,3,skew,even);   | ||||
|     HAND_STENCIL_LEG(UUU,Yp,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG(UUU,Zp,1,skew,even);   | ||||
|     HAND_STENCIL_LEG(UUU,Tp,0,skew,odd);   | ||||
|     HAND_STENCIL_LEG(UUU,Xm,3,skew,even);   | ||||
|     HAND_STENCIL_LEG(UUU,Ym,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG(UUU,Zm,1,skew,even);   | ||||
|     HAND_STENCIL_LEG(UUU,Tm,0,skew,odd);   | ||||
|      | ||||
|     if ( dag ) { | ||||
|       result()()(0) = - even_0 - odd_0; | ||||
|       result()()(1) = - even_1 - odd_1; | ||||
|       result()()(2) = - even_2 - odd_2; | ||||
|     } else {  | ||||
|     LOAD_CHI(buf); | ||||
|       result()()(0) = even_0 + odd_0; | ||||
|       result()()(1) = even_1 + odd_1; | ||||
|       result()()(2) = even_2 + odd_2; | ||||
|     } | ||||
|     vstream(out._odata[sF],result); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteHandInt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 					     DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 					     SiteSpinor *buf, int LLs, int sU,  | ||||
| 					     const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
|     MULT(Xp,even); | ||||
|   } | ||||
|   typedef typename Simd::scalar_type S; | ||||
|   typedef typename Simd::vector_type V; | ||||
|  | ||||
|   // Yp | ||||
|   SE=st.GetEntry(ptype,Yp+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|   Simd even_0; // 12 regs on knc | ||||
|   Simd even_1; | ||||
|   Simd even_2; | ||||
|   Simd odd_0; // 12 regs on knc | ||||
|   Simd odd_1; | ||||
|   Simd odd_2; | ||||
|  | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(2); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   Simd Chi_0;    // two spinor; 6 regs | ||||
|   Simd Chi_1; | ||||
|   Simd Chi_2; | ||||
|    | ||||
|   Simd U_00;  // two rows of U matrix | ||||
|   Simd U_10; | ||||
|   Simd U_20;   | ||||
|   Simd U_01; | ||||
|   Simd U_11; | ||||
|   Simd U_21;  // 2 reg left. | ||||
|   Simd U_02; | ||||
|   Simd U_12; | ||||
|   Simd U_22;  | ||||
|  | ||||
|   SiteSpinor result; | ||||
|   int offset,local,perm, ptype; | ||||
|  | ||||
|   StencilEntry *SE; | ||||
|   int skew; | ||||
|  | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=s+LLs*sU; | ||||
|  | ||||
|     even_0 = zero;    even_1 = zero;    even_2 = zero; | ||||
|      odd_0 = zero;     odd_1 = zero;     odd_2 = zero; | ||||
|  | ||||
|     skew = 0; | ||||
|     HAND_STENCIL_LEG_INT(U,Xp,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(U,Yp,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_INT(U,Zp,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(U,Tp,0,skew,odd);   | ||||
|     HAND_STENCIL_LEG_INT(U,Xm,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(U,Ym,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_INT(U,Zm,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(U,Tm,0,skew,odd);   | ||||
|     skew = 8; | ||||
|     HAND_STENCIL_LEG_INT(UUU,Xp,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(UUU,Yp,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_INT(UUU,Zp,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(UUU,Tp,0,skew,odd);   | ||||
|     HAND_STENCIL_LEG_INT(UUU,Xm,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(UUU,Ym,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_INT(UUU,Zm,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_INT(UUU,Tm,0,skew,odd);   | ||||
|  | ||||
|     // Assume every site must be connected to at least one interior point. No 1^4 subvols. | ||||
|     if ( dag ) { | ||||
|       result()()(0) = - even_0 - odd_0; | ||||
|       result()()(1) = - even_1 - odd_1; | ||||
|       result()()(2) = - even_2 - odd_2; | ||||
|     } else {  | ||||
|     LOAD_CHI(buf); | ||||
|       result()()(0) = even_0 + odd_0; | ||||
|       result()()(1) = even_1 + odd_1; | ||||
|       result()()(2) = even_2 + odd_2; | ||||
|     } | ||||
|     vstream(out._odata[sF],result); | ||||
|   } | ||||
| } | ||||
|  | ||||
|  | ||||
| template <class Impl> | ||||
| void StaggeredKernels<Impl>::DhopSiteHandExt(StencilImpl &st, LebesgueOrder &lo,  | ||||
| 					     DoubledGaugeField &U, DoubledGaugeField &UUU, | ||||
| 					     SiteSpinor *buf, int LLs, int sU,  | ||||
| 					     const FermionField &in, FermionField &out,int dag)  | ||||
| { | ||||
|     MULT(Yp,odd); | ||||
|   } | ||||
|   typedef typename Simd::scalar_type S; | ||||
|   typedef typename Simd::vector_type V; | ||||
|  | ||||
|   Simd even_0; // 12 regs on knc | ||||
|   Simd even_1; | ||||
|   Simd even_2; | ||||
|   Simd odd_0; // 12 regs on knc | ||||
|   Simd odd_1; | ||||
|   Simd odd_2; | ||||
|  | ||||
|   // Zp | ||||
|   SE=st.GetEntry(ptype,Zp+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|   Simd Chi_0;    // two spinor; 6 regs | ||||
|   Simd Chi_1; | ||||
|   Simd Chi_2; | ||||
|    | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(1); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   Simd U_00;  // two rows of U matrix | ||||
|   Simd U_10; | ||||
|   Simd U_20;   | ||||
|   Simd U_01; | ||||
|   Simd U_11; | ||||
|   Simd U_21;  // 2 reg left. | ||||
|   Simd U_02; | ||||
|   Simd U_12; | ||||
|   Simd U_22;  | ||||
|  | ||||
|   SiteSpinor result; | ||||
|   int offset,local,perm, ptype; | ||||
|  | ||||
|   StencilEntry *SE; | ||||
|   int skew; | ||||
|  | ||||
|   for(int s=0;s<LLs;s++){ | ||||
|     int sF=s+LLs*sU; | ||||
|  | ||||
|     even_0 = zero;    even_1 = zero;    even_2 = zero; | ||||
|      odd_0 = zero;     odd_1 = zero;     odd_2 = zero; | ||||
|     int nmu=0; | ||||
|     skew = 0; | ||||
|     HAND_STENCIL_LEG_EXT(U,Xp,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(U,Yp,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_EXT(U,Zp,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(U,Tp,0,skew,odd);   | ||||
|     HAND_STENCIL_LEG_EXT(U,Xm,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(U,Ym,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_EXT(U,Zm,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(U,Tm,0,skew,odd);   | ||||
|     skew = 8; | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Xp,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Yp,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Zp,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Tp,0,skew,odd);   | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Xm,3,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Ym,2,skew,odd);    | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Zm,1,skew,even);   | ||||
|     HAND_STENCIL_LEG_EXT(UUU,Tm,0,skew,odd);   | ||||
|  | ||||
|     // Add sum of all exterior connected stencil legs | ||||
|     if ( nmu ) {  | ||||
|       if ( dag ) { | ||||
| 	result()()(0) = - even_0 - odd_0; | ||||
| 	result()()(1) = - even_1 - odd_1; | ||||
| 	result()()(2) = - even_2 - odd_2; | ||||
|       } else {  | ||||
|     LOAD_CHI(buf); | ||||
| 	result()()(0) = even_0 + odd_0; | ||||
| 	result()()(1) = even_1 + odd_1; | ||||
| 	result()()(2) = even_2 + odd_2; | ||||
|       } | ||||
|       out._odata[sF] = out._odata[sF] + result; | ||||
|     } | ||||
|   } | ||||
|   { | ||||
|     MULT_ADD(Zp,even); | ||||
| } | ||||
|  | ||||
|   // Tp | ||||
|   SE=st.GetEntry(ptype,Tp+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|    | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(0); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   } else {  | ||||
|     LOAD_CHI(buf); | ||||
|   } | ||||
|   { | ||||
|     MULT_ADD(Tp,odd); | ||||
|   } | ||||
|    | ||||
|   // Xm | ||||
|   SE=st.GetEntry(ptype,Xm+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|    | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(3); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   } else {  | ||||
|     LOAD_CHI(buf); | ||||
|   } | ||||
|   { | ||||
|     MULT_ADD(Xm,even); | ||||
|   } | ||||
|    | ||||
|    | ||||
|   // Ym | ||||
|   SE=st.GetEntry(ptype,Ym+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|    | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(2); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   } else {  | ||||
|     LOAD_CHI(buf); | ||||
|   } | ||||
|   { | ||||
|     MULT_ADD(Ym,odd); | ||||
|   } | ||||
|  | ||||
|   // Zm | ||||
|   SE=st.GetEntry(ptype,Zm+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|    | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(1); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   } else {  | ||||
|     LOAD_CHI(buf); | ||||
|   } | ||||
|   { | ||||
|     MULT_ADD(Zm,even); | ||||
|   } | ||||
|  | ||||
|   // Tm | ||||
|   SE=st.GetEntry(ptype,Tm+skew,sF); | ||||
|   offset = SE->_offset; | ||||
|   local  = SE->_is_local; | ||||
|   perm   = SE->_permute; | ||||
|    | ||||
|   if ( local ) { | ||||
|     LOAD_CHI(in._odata); | ||||
|     if ( perm) { | ||||
|       PERMUTE_DIR(0); // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||
|     } | ||||
|   } else {  | ||||
|     LOAD_CHI(buf); | ||||
|   } | ||||
|   { | ||||
|     MULT_ADD(Tm,odd); | ||||
|   } | ||||
|  | ||||
|   vstream(out()()(0),even_0+odd_0); | ||||
|   vstream(out()()(1),even_1+odd_1); | ||||
|   vstream(out()()(2),even_2+odd_2); | ||||
|  | ||||
| } | ||||
|  | ||||
| #define DHOP_SITE_HAND_INSTANTIATE(IMPL)				\ | ||||
|   template void StaggeredKernels<IMPL>::DhopSiteHand(StencilImpl &st, LebesgueOrder &lo, \ | ||||
| 						     DoubledGaugeField &U,DoubledGaugeField &UUU, \ | ||||
| 						     SiteSpinor *buf, int LLs, \ | ||||
| 						     int sU, const FermionField &in, FermionField &out, int dag); | ||||
| 						     SiteSpinor *buf, int LLs, int sU, \ | ||||
| 						     const FermionField &in, FermionField &out, int dag); \ | ||||
| 									\ | ||||
|   template void StaggeredKernels<IMPL>::DhopSiteHandInt(StencilImpl &st, LebesgueOrder &lo, \ | ||||
| 						     DoubledGaugeField &U,DoubledGaugeField &UUU, \ | ||||
| 						     SiteSpinor *buf, int LLs, int sU, \ | ||||
| 						     const FermionField &in, FermionField &out, int dag); \ | ||||
| 									\ | ||||
|   template void StaggeredKernels<IMPL>::DhopSiteHandExt(StencilImpl &st, LebesgueOrder &lo, \ | ||||
| 						     DoubledGaugeField &U,DoubledGaugeField &UUU, \ | ||||
| 						     SiteSpinor *buf, int LLs, int sU, \ | ||||
| 						     const FermionField &in, FermionField &out, int dag); \ | ||||
|  | ||||
| #define DHOP_SITE_DEPTH_HAND_INSTANTIATE(IMPL)				\ | ||||
|   template void StaggeredKernels<IMPL>::DhopSiteDepthHand(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, \ | ||||
| 							  SiteSpinor *buf, int sF, \ | ||||
| 							  int sU, const FermionField &in, SiteSpinor &out,int threeLink) ; | ||||
| DHOP_SITE_HAND_INSTANTIATE(StaggeredImplD); | ||||
| DHOP_SITE_HAND_INSTANTIATE(StaggeredImplF); | ||||
| DHOP_SITE_HAND_INSTANTIATE(StaggeredVec5dImplD); | ||||
| DHOP_SITE_HAND_INSTANTIATE(StaggeredVec5dImplF); | ||||
|  | ||||
| DHOP_SITE_DEPTH_HAND_INSTANTIATE(StaggeredImplD); | ||||
| DHOP_SITE_DEPTH_HAND_INSTANTIATE(StaggeredImplF); | ||||
| DHOP_SITE_DEPTH_HAND_INSTANTIATE(StaggeredVec5dImplD); | ||||
| DHOP_SITE_DEPTH_HAND_INSTANTIATE(StaggeredVec5dImplF); | ||||
|  | ||||
| }} | ||||
| } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -274,41 +274,16 @@ public: | ||||
|     if ( timer4 ) std::cout << GridLogMessage << " timer4 " <<timer4 <<std::endl; | ||||
|   } | ||||
|  | ||||
|   std::vector<int> same_node; | ||||
|   std::vector<int> surface_list; | ||||
|  | ||||
|   WilsonStencil(GridBase *grid, | ||||
| 		int npoints, | ||||
| 		int checkerboard, | ||||
| 		const std::vector<int> &directions, | ||||
| 		const std::vector<int> &distances)   | ||||
|     : CartesianStencil<vobj,cobj> (grid,npoints,checkerboard,directions,distances) , | ||||
|     same_node(npoints) | ||||
|     : CartesianStencil<vobj,cobj> (grid,npoints,checkerboard,directions,distances)  | ||||
|   {  | ||||
|     ZeroCountersi(); | ||||
|     surface_list.resize(0); | ||||
|   }; | ||||
|  | ||||
|   void BuildSurfaceList(int Ls,int vol4){ | ||||
|  | ||||
|     // find same node for SHM | ||||
|     // Here we know the distance is 1 for WilsonStencil | ||||
|     for(int point=0;point<this->_npoints;point++){ | ||||
|       same_node[point] = this->SameNode(point); | ||||
|     } | ||||
|      | ||||
|     for(int site = 0 ;site< vol4;site++){ | ||||
|       int local = 1; | ||||
|       for(int point=0;point<this->_npoints;point++){ | ||||
| 	if( (!this->GetNodeLocal(site*Ls,point)) && (!same_node[point]) ){  | ||||
| 	  local = 0; | ||||
| 	} | ||||
|       } | ||||
|       if(local == 0) {  | ||||
| 	surface_list.push_back(site); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   template < class compressor> | ||||
|   void HaloExchangeOpt(const Lattice<vobj> &source,compressor &compress)  | ||||
| @@ -369,23 +344,23 @@ public: | ||||
|     int dag = compress.dag; | ||||
|     int face_idx=0; | ||||
|     if ( dag ) {  | ||||
|       assert(same_node[Xp]==this->HaloGatherDir(source,XpCompress,Xp,face_idx)); | ||||
|       assert(same_node[Yp]==this->HaloGatherDir(source,YpCompress,Yp,face_idx)); | ||||
|       assert(same_node[Zp]==this->HaloGatherDir(source,ZpCompress,Zp,face_idx)); | ||||
|       assert(same_node[Tp]==this->HaloGatherDir(source,TpCompress,Tp,face_idx)); | ||||
|       assert(same_node[Xm]==this->HaloGatherDir(source,XmCompress,Xm,face_idx)); | ||||
|       assert(same_node[Ym]==this->HaloGatherDir(source,YmCompress,Ym,face_idx)); | ||||
|       assert(same_node[Zm]==this->HaloGatherDir(source,ZmCompress,Zm,face_idx)); | ||||
|       assert(same_node[Tm]==this->HaloGatherDir(source,TmCompress,Tm,face_idx)); | ||||
|       assert(this->same_node[Xp]==this->HaloGatherDir(source,XpCompress,Xp,face_idx)); | ||||
|       assert(this->same_node[Yp]==this->HaloGatherDir(source,YpCompress,Yp,face_idx)); | ||||
|       assert(this->same_node[Zp]==this->HaloGatherDir(source,ZpCompress,Zp,face_idx)); | ||||
|       assert(this->same_node[Tp]==this->HaloGatherDir(source,TpCompress,Tp,face_idx)); | ||||
|       assert(this->same_node[Xm]==this->HaloGatherDir(source,XmCompress,Xm,face_idx)); | ||||
|       assert(this->same_node[Ym]==this->HaloGatherDir(source,YmCompress,Ym,face_idx)); | ||||
|       assert(this->same_node[Zm]==this->HaloGatherDir(source,ZmCompress,Zm,face_idx)); | ||||
|       assert(this->same_node[Tm]==this->HaloGatherDir(source,TmCompress,Tm,face_idx)); | ||||
|     } else { | ||||
|       assert(same_node[Xp]==this->HaloGatherDir(source,XmCompress,Xp,face_idx)); | ||||
|       assert(same_node[Yp]==this->HaloGatherDir(source,YmCompress,Yp,face_idx)); | ||||
|       assert(same_node[Zp]==this->HaloGatherDir(source,ZmCompress,Zp,face_idx)); | ||||
|       assert(same_node[Tp]==this->HaloGatherDir(source,TmCompress,Tp,face_idx)); | ||||
|       assert(same_node[Xm]==this->HaloGatherDir(source,XpCompress,Xm,face_idx)); | ||||
|       assert(same_node[Ym]==this->HaloGatherDir(source,YpCompress,Ym,face_idx)); | ||||
|       assert(same_node[Zm]==this->HaloGatherDir(source,ZpCompress,Zm,face_idx)); | ||||
|       assert(same_node[Tm]==this->HaloGatherDir(source,TpCompress,Tm,face_idx)); | ||||
|       assert(this->same_node[Xp]==this->HaloGatherDir(source,XmCompress,Xp,face_idx)); | ||||
|       assert(this->same_node[Yp]==this->HaloGatherDir(source,YmCompress,Yp,face_idx)); | ||||
|       assert(this->same_node[Zp]==this->HaloGatherDir(source,ZmCompress,Zp,face_idx)); | ||||
|       assert(this->same_node[Tp]==this->HaloGatherDir(source,TmCompress,Tp,face_idx)); | ||||
|       assert(this->same_node[Xm]==this->HaloGatherDir(source,XpCompress,Xm,face_idx)); | ||||
|       assert(this->same_node[Ym]==this->HaloGatherDir(source,YpCompress,Ym,face_idx)); | ||||
|       assert(this->same_node[Zm]==this->HaloGatherDir(source,ZpCompress,Zm,face_idx)); | ||||
|       assert(this->same_node[Tm]==this->HaloGatherDir(source,TpCompress,Tm,face_idx)); | ||||
|     } | ||||
|     this->face_table_computed=1; | ||||
|     assert(this->u_comm_offset==this->_unified_buffer_size); | ||||
|   | ||||
| @@ -348,15 +348,98 @@ void WilsonFermion<Impl>::DhopDirDisp(const FermionField &in, FermionField &out, | ||||
|   parallel_for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|     Kernels::DhopDir(Stencil, Umu, Stencil.CommBuf(), sss, sss, in, out, dirdisp, gamma); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| }  | ||||
| /*Change starts*/ | ||||
| template <class Impl> | ||||
| void WilsonFermion<Impl>::DhopInternal(StencilImpl &st, LebesgueOrder &lo, | ||||
|                                        DoubledGaugeField &U, | ||||
|                                        const FermionField &in, | ||||
|                                        FermionField &out, int dag) { | ||||
|   assert((dag == DaggerNo) || (dag == DaggerYes)); | ||||
| #ifdef GRID_OMP | ||||
|   if ( WilsonKernelsStatic::Comms == WilsonKernelsStatic::CommsAndCompute ) | ||||
|     DhopInternalOverlappedComms(st,lo,U,in,out,dag); | ||||
|   else | ||||
| #endif  | ||||
|     DhopInternalSerial(st,lo,U,in,out,dag); | ||||
|  | ||||
| } | ||||
|  | ||||
| template <class Impl> | ||||
| void WilsonFermion<Impl>::DhopInternalOverlappedComms(StencilImpl &st, LebesgueOrder &lo, | ||||
|                                        DoubledGaugeField &U, | ||||
|                                        const FermionField &in, | ||||
|                                        FermionField &out, int dag) { | ||||
|   assert((dag == DaggerNo) || (dag == DaggerYes)); | ||||
| #ifdef GRID_OMP | ||||
|   Compressor compressor; | ||||
|   int len =  U._grid->oSites(); | ||||
|   const int LLs =  1; | ||||
|  | ||||
|   st.Prepare(); | ||||
|   st.HaloGather(in,compressor); | ||||
|   st.CommsMergeSHM(compressor); | ||||
| #pragma omp parallel | ||||
|   { | ||||
|     int tid = omp_get_thread_num(); | ||||
|     int nthreads = omp_get_num_threads(); | ||||
|     int ncomms = CartesianCommunicator::nCommThreads; | ||||
|     if (ncomms == -1) ncomms = 1; | ||||
|     assert(nthreads > ncomms); | ||||
|     if (tid >= ncomms) { | ||||
|       nthreads -= ncomms; | ||||
|       int ttid  = tid - ncomms; | ||||
|       int n     = len; | ||||
|       int chunk = n / nthreads; | ||||
|       int rem   = n % nthreads; | ||||
|       int myblock, myn; | ||||
|       if (ttid < rem) { | ||||
|         myblock = ttid * chunk + ttid; | ||||
|         myn = chunk+1; | ||||
|       } else { | ||||
|         myblock = ttid*chunk + rem; | ||||
|         myn = chunk; | ||||
|       } | ||||
|       // do the compute | ||||
|      if (dag == DaggerYes) { | ||||
|  | ||||
|         for (int sss = myblock; sss < myblock+myn; ++sss) { | ||||
|          Kernels::DhopSiteDag(st, lo, U, st.CommBuf(), sss, sss, 1, 1, in, out); | ||||
|        } | ||||
|      } else { | ||||
|         for (int sss = myblock; sss < myblock+myn; ++sss) { | ||||
|          Kernels::DhopSite(st, lo, U, st.CommBuf(), sss, sss, 1, 1, in, out); | ||||
|        } | ||||
|     } //else | ||||
|  | ||||
|     } else { | ||||
|       st.CommunicateThreaded(); | ||||
|     } | ||||
|  | ||||
|   Compressor compressor(dag); | ||||
|  | ||||
|   if (dag == DaggerYes) { | ||||
|     parallel_for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|       Kernels::DhopSiteDag(st, lo, U, st.CommBuf(), sss, sss, 1, 1, in, out); | ||||
|     } | ||||
|   } else { | ||||
|     parallel_for (int sss = 0; sss < in._grid->oSites(); sss++) { | ||||
|       Kernels::DhopSite(st, lo, U, st.CommBuf(), sss, sss, 1, 1, in, out); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   }  //pragma | ||||
| #else | ||||
|   assert(0); | ||||
| #endif | ||||
| }; | ||||
|  | ||||
|  | ||||
| template <class Impl> | ||||
| void WilsonFermion<Impl>::DhopInternalSerial(StencilImpl &st, LebesgueOrder &lo, | ||||
|                                        DoubledGaugeField &U, | ||||
|                                        const FermionField &in, | ||||
|                                        FermionField &out, int dag) { | ||||
|   assert((dag == DaggerNo) || (dag == DaggerYes)); | ||||
|   Compressor compressor(dag); | ||||
|   st.HaloExchange(in, compressor); | ||||
|  | ||||
| @@ -370,6 +453,7 @@ void WilsonFermion<Impl>::DhopInternal(StencilImpl &st, LebesgueOrder &lo, | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| /*Change ends */ | ||||
|  | ||||
| /******************************************************************************* | ||||
|  * Conserved current utilities for Wilson fermions, for contracting propagators | ||||
|   | ||||
| @@ -130,6 +130,12 @@ class WilsonFermion : public WilsonKernels<Impl>, public WilsonFermionStatic { | ||||
|   void DhopInternal(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, | ||||
|                     const FermionField &in, FermionField &out, int dag); | ||||
|  | ||||
|   void DhopInternalSerial(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, | ||||
|                     const FermionField &in, FermionField &out, int dag); | ||||
|  | ||||
|   void DhopInternalOverlappedComms(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, | ||||
|                     const FermionField &in, FermionField &out, int dag); | ||||
|  | ||||
|   // Constructor | ||||
|   WilsonFermion(GaugeField &_Umu, GridCartesian &Fgrid, | ||||
|                 GridRedBlackCartesian &Hgrid, RealD _mass,  | ||||
| @@ -145,6 +151,8 @@ class WilsonFermion : public WilsonKernels<Impl>, public WilsonFermionStatic { | ||||
|  | ||||
|   //    protected: | ||||
|  public: | ||||
|   virtual RealD Mass(void) { return mass; } | ||||
|   virtual int   isTrivialEE(void) { return 1; }; | ||||
|   RealD mass; | ||||
|   RealD diag_mass; | ||||
|  | ||||
|   | ||||
| @@ -445,8 +445,7 @@ void WilsonFermion5D<Impl>::DhopInternalOverlappedComms(StencilImpl & st, Lebesg | ||||
| 	} | ||||
|       } | ||||
| 	ptime = usecond() - start; | ||||
|     } | ||||
|     { | ||||
|     } else { | ||||
|       double start = usecond(); | ||||
|       st.CommunicateThreaded(); | ||||
|       ctime = usecond() - start; | ||||
|   | ||||
| @@ -232,6 +232,7 @@ private: | ||||
|   void GenericDhopSiteDagExt(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, SiteHalfSpinor * buf, | ||||
| 			     int sF, int sU, const FermionField &in, FermionField &out); | ||||
|  | ||||
|  | ||||
|   void AsmDhopSite(StencilImpl &st, LebesgueOrder &lo, DoubledGaugeField &U, SiteHalfSpinor * buf, | ||||
| 		   int sF, int sU, int Ls, int Ns, const FermionField &in,FermionField &out); | ||||
|  | ||||
|   | ||||
| @@ -80,12 +80,8 @@ class ScidacHmcCheckpointer : public BaseHmcCheckpointer<Implementation> { | ||||
|       this->build_filenames(traj, Params, config, rng); | ||||
|       GridBase *grid = U._grid; | ||||
|       uint32_t nersc_csum,scidac_csuma,scidac_csumb; | ||||
|       BinaryIO::writeRNG(sRNG, pRNG, rng, 0,nersc_csum,scidac_csuma,scidac_csumb); | ||||
|       ScidacWriter _ScidacWriter(grid->IsBoss()); | ||||
|       _ScidacWriter.open(rng); | ||||
|       _ScidacWriter.writeScidacRNGRecord(sRNG, pRNG); | ||||
|       _ScidacWriter.close();    | ||||
|        | ||||
|       //BinaryIO::writeRNG(sRNG, pRNG, rng, 0,nersc_csum,scidac_csuma,scidac_csumb); | ||||
|       _ScidacWriter.open(config); | ||||
|       _ScidacWriter.writeScidacFieldRecord(U, MData); | ||||
|       _ScidacWriter.close(); | ||||
| @@ -106,12 +102,10 @@ class ScidacHmcCheckpointer : public BaseHmcCheckpointer<Implementation> { | ||||
|  | ||||
|  | ||||
|     uint32_t nersc_csum,scidac_csuma,scidac_csumb; | ||||
|     ScidacReader _ScidacReader; | ||||
|     //BinaryIO::readRNG(sRNG, pRNG, rng, 0,nersc_csum,scidac_csuma,scidac_csumb); | ||||
|     _ScidacReader.open(rng); | ||||
|     _ScidacReader.readScidacRNGRecord(sRNG, pRNG);   | ||||
|     _ScidacReader.close(); | ||||
|     BinaryIO::readRNG(sRNG, pRNG, rng, 0,nersc_csum,scidac_csuma,scidac_csumb); | ||||
|  | ||||
|     Metadata md_content; | ||||
|     ScidacReader _ScidacReader; | ||||
|     _ScidacReader.open(config); | ||||
|     _ScidacReader.readScidacFieldRecord(U,md_content);  // format from the header | ||||
|     _ScidacReader.close(); | ||||
|   | ||||
| @@ -150,6 +150,8 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | ||||
|   std::vector<int>                  _distances; | ||||
|   std::vector<int>                  _comm_buf_size; | ||||
|   std::vector<int>                  _permute_type; | ||||
|   std::vector<int> same_node; | ||||
|   std::vector<int> surface_list; | ||||
|  | ||||
|   Vector<StencilEntry>  _entries; | ||||
|   std::vector<Packet> Packets; | ||||
| @@ -201,7 +203,7 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | ||||
|  | ||||
|     int dimension    = _directions[point]; | ||||
|     int displacement = _distances[point]; | ||||
|     assert( (displacement==1) || (displacement==-1)); | ||||
|  | ||||
|  | ||||
|     int pd              = _grid->_processors[dimension]; | ||||
|     int fd              = _grid->_fdimensions[dimension]; | ||||
| @@ -216,9 +218,12 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | ||||
|     if ( ! comm_dim ) return 1; | ||||
|  | ||||
|     int nbr_proc; | ||||
|     if (displacement==1) nbr_proc = 1; | ||||
|     if (displacement>0) nbr_proc = 1; | ||||
|     else                nbr_proc = pd-1; | ||||
|  | ||||
|     // FIXME  this logic needs to be sorted for three link term | ||||
|     //    assert( (displacement==1) || (displacement==-1)); | ||||
|     // Present hack only works for >= 4^4 subvol per node | ||||
|     _grid->ShiftedRanks(dimension,nbr_proc,xmit_to_rank,recv_from_rank);  | ||||
|  | ||||
|     void *shm = (void *) _grid->ShmBufferTranslate(recv_from_rank,u_recv_buf_p); | ||||
| @@ -539,6 +544,29 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   // Move interior/exterior split into the generic stencil | ||||
|   // FIXME Explicit Ls in interface is a pain. Should just use a vol | ||||
|   void BuildSurfaceList(int Ls,int vol4){ | ||||
|  | ||||
|     // find same node for SHM | ||||
|     // Here we know the distance is 1 for WilsonStencil | ||||
|     for(int point=0;point<this->_npoints;point++){ | ||||
|       same_node[point] = this->SameNode(point); | ||||
|     } | ||||
|      | ||||
|     for(int site = 0 ;site< vol4;site++){ | ||||
|       int local = 1; | ||||
|       for(int point=0;point<this->_npoints;point++){ | ||||
| 	if( (!this->GetNodeLocal(site*Ls,point)) && (!same_node[point]) ){  | ||||
| 	  local = 0; | ||||
| 	} | ||||
|       } | ||||
|       if(local == 0) {  | ||||
| 	surface_list.push_back(site); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  CartesianStencil(GridBase *grid, | ||||
| 		  int npoints, | ||||
| 		  int checkerboard, | ||||
| @@ -549,7 +577,8 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | ||||
|     comm_bytes_thr(npoints),  | ||||
|     comm_enter_thr(npoints), | ||||
|     comm_leave_thr(npoints),  | ||||
|        comm_time_thr(npoints) | ||||
|     comm_time_thr(npoints), | ||||
|     same_node(npoints) | ||||
|   { | ||||
|     face_table_computed=0; | ||||
|     _npoints = npoints; | ||||
| @@ -557,6 +586,7 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | ||||
|     _directions = directions; | ||||
|     _distances  = distances; | ||||
|     _unified_buffer_size=0; | ||||
|     surface_list.resize(0); | ||||
|  | ||||
|     int osites  = _grid->oSites(); | ||||
|      | ||||
|   | ||||
| @@ -368,8 +368,10 @@ void Grid_init(int *argc,char ***argv) | ||||
|   } | ||||
|   if( GridCmdOptionExists(*argv,*argv+*argc,"--comms-overlap") ){ | ||||
|     QCD::WilsonKernelsStatic::Comms = QCD::WilsonKernelsStatic::CommsAndCompute; | ||||
|     QCD::StaggeredKernelsStatic::Comms = QCD::StaggeredKernelsStatic::CommsAndCompute; | ||||
|   } else { | ||||
|     QCD::WilsonKernelsStatic::Comms = QCD::WilsonKernelsStatic::CommsThenCompute; | ||||
|     QCD::StaggeredKernelsStatic::Comms = QCD::StaggeredKernelsStatic::CommsThenCompute; | ||||
|   } | ||||
|   if( GridCmdOptionExists(*argv,*argv+*argc,"--comms-concurrent") ){ | ||||
|     CartesianCommunicator::SetCommunicatorPolicy(CartesianCommunicator::CommunicatorPolicyConcurrent); | ||||
|   | ||||
| @@ -141,6 +141,7 @@ int main (int argc, char ** argv) | ||||
|   t1=usecond(); | ||||
|   | ||||
|   std::cout<<GridLogMessage << "Called Ds ASM"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm src "<< norm2(src)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm result "<< norm2(tmp)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)<<std::endl; | ||||
|  | ||||
| @@ -160,7 +161,8 @@ int main (int argc, char ** argv) | ||||
|   localConvert(sresult,tmp); | ||||
|   | ||||
|   std::cout<<GridLogMessage << "Called sDs unroll"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm result "<< norm2(sresult)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm ssrc "<< norm2(ssrc)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm sresult "<< norm2(sresult)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)<<std::endl; | ||||
|  | ||||
|  | ||||
| @@ -181,6 +183,7 @@ int main (int argc, char ** argv) | ||||
|   localConvert(sresult,tmp); | ||||
|   | ||||
|   std::cout<<GridLogMessage << "Called sDs asm"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm ssrc   "<< norm2(ssrc)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm result "<< norm2(sresult)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)*extra<<std::endl; | ||||
|  | ||||
|   | ||||
							
								
								
									
										196
									
								
								tests/core/Test_staggered5DvecF.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								tests/core/Test_staggered5DvecF.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | ||||
|     /************************************************************************************* | ||||
|  | ||||
|     Grid physics library, www.github.com/paboyle/Grid  | ||||
|  | ||||
|     Source file: ./benchmarks/Benchmark_wilson.cc | ||||
|  | ||||
|     Copyright (C) 2015 | ||||
|  | ||||
| Author: Peter Boyle <paboyle@ph.ed.ac.uk> | ||||
| Author: paboyle <paboyle@ph.ed.ac.uk> | ||||
|  | ||||
|     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. | ||||
|  | ||||
|     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 */ | ||||
| #include <Grid/Grid.h> | ||||
|  | ||||
| using namespace std; | ||||
| using namespace Grid; | ||||
| using namespace Grid::QCD; | ||||
|  | ||||
| int main (int argc, char ** argv) | ||||
| { | ||||
|   Grid_init(&argc,&argv); | ||||
|  | ||||
|   std::vector<int> latt_size   = GridDefaultLatt(); | ||||
|   std::vector<int> simd_layout = GridDefaultSimd(Nd,vComplex::Nsimd()); | ||||
|   std::vector<int> mpi_layout  = GridDefaultMpi(); | ||||
|  | ||||
|   const int Ls=16; | ||||
|   GridCartesian         * UGrid   = SpaceTimeGrid::makeFourDimGrid(GridDefaultLatt(), GridDefaultSimd(Nd,vComplexF::Nsimd()),GridDefaultMpi()); | ||||
|   GridRedBlackCartesian * UrbGrid = SpaceTimeGrid::makeFourDimRedBlackGrid(UGrid); | ||||
|   GridCartesian         * FGrid   = SpaceTimeGrid::makeFiveDimGrid(Ls,UGrid); | ||||
|   GridRedBlackCartesian * FrbGrid = SpaceTimeGrid::makeFiveDimRedBlackGrid(Ls,UGrid); | ||||
|  | ||||
|   std::cout << GridLogMessage << "Making s innermost grids"<<std::endl; | ||||
|   GridCartesian         * sUGrid   = SpaceTimeGrid::makeFourDimDWFGrid(GridDefaultLatt(),GridDefaultMpi()); | ||||
|   GridRedBlackCartesian * sUrbGrid = SpaceTimeGrid::makeFourDimRedBlackGrid(sUGrid); | ||||
|   GridCartesian         * sFGrid   = SpaceTimeGrid::makeFiveDimDWFGrid(Ls,UGrid); | ||||
|   GridRedBlackCartesian * sFrbGrid = SpaceTimeGrid::makeFiveDimDWFRedBlackGrid(Ls,UGrid); | ||||
|  | ||||
|   int threads = GridThread::GetThreads(); | ||||
|  | ||||
|   std::cout<<GridLogMessage << "Grid is setup to use "<<threads<<" threads"<<std::endl; | ||||
|  | ||||
|   std::vector<int> seeds({1,2,3,4}); | ||||
|  | ||||
|   GridParallelRNG          pRNG4(UGrid); | ||||
|   GridParallelRNG          pRNG5(FGrid); | ||||
|   pRNG4.SeedFixedIntegers(seeds); | ||||
|   pRNG5.SeedFixedIntegers(seeds); | ||||
|  | ||||
|   typedef typename ImprovedStaggeredFermion5DF::FermionField FermionField;  | ||||
|   typedef typename ImprovedStaggeredFermion5DF::ComplexField ComplexField;  | ||||
|   typename ImprovedStaggeredFermion5DF::ImplParams params;  | ||||
|  | ||||
|   FermionField src   (FGrid); | ||||
|   random(pRNG5,src); | ||||
|   /* | ||||
|   std::vector<int> site({0,1,2,0,0}); | ||||
|   ColourVector cv = zero; | ||||
|   cv()()(0)=1.0; | ||||
|   src = zero; | ||||
|   pokeSite(cv,src,site); | ||||
|   */ | ||||
|   FermionField result(FGrid); result=zero; | ||||
|   FermionField    tmp(FGrid);    tmp=zero; | ||||
|   FermionField    err(FGrid);    tmp=zero; | ||||
|   FermionField phi   (FGrid); random(pRNG5,phi); | ||||
|   FermionField chi   (FGrid); random(pRNG5,chi); | ||||
|  | ||||
|   LatticeGaugeFieldF Umu(UGrid); | ||||
|   SU3::HotConfiguration(pRNG4,Umu); | ||||
|  | ||||
|   /* | ||||
|   for(int mu=1;mu<4;mu++){ | ||||
|     auto tmp = PeekIndex<LorentzIndex>(Umu,mu); | ||||
|         tmp = zero; | ||||
|     PokeIndex<LorentzIndex>(Umu,tmp,mu); | ||||
|   } | ||||
|   */ | ||||
|   double volume=Ls; | ||||
|   for(int mu=0;mu<Nd;mu++){ | ||||
|     volume=volume*latt_size[mu]; | ||||
|   }   | ||||
|  | ||||
|   RealD mass=0.1; | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|  | ||||
|   ImprovedStaggeredFermion5DF     Ds(Umu,Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,c1,c2,u0,params); | ||||
|   ImprovedStaggeredFermionVec5dF sDs(Umu,Umu,*sFGrid,*sFrbGrid,*sUGrid,*sUrbGrid,mass,c1,c2,u0,params); | ||||
|  | ||||
|   std::cout<<GridLogMessage<<"=========================================================="<<std::endl; | ||||
|   std::cout<<GridLogMessage<<"= Testing Dhop against cshift implementation         "<<std::endl; | ||||
|   std::cout<<GridLogMessage<<"=========================================================="<<std::endl; | ||||
|  | ||||
|   int ncall=1000; | ||||
|   int ncall1=1000; | ||||
|   double t0(0),t1(0); | ||||
|   double flops=(16*(3*(6+8+8)) + 15*3*2)*volume*ncall; // == 66*16 +  == 1146 | ||||
|  | ||||
|   std::cout<<GridLogMessage << "Calling staggered operator"<<std::endl; | ||||
|   t0=usecond(); | ||||
|   for(int i=0;i<ncall1;i++){ | ||||
|     Ds.Dhop(src,result,0); | ||||
|   } | ||||
|   t1=usecond(); | ||||
|  | ||||
|    | ||||
|   std::cout<<GridLogMessage << "Called Ds"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm result "<< norm2(result)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)<<std::endl; | ||||
|  | ||||
|   std::cout<<GridLogMessage << "Calling vectorised staggered operator"<<std::endl; | ||||
|  | ||||
| #ifdef AVX512 | ||||
|   QCD::StaggeredKernelsStatic::Opt=QCD::StaggeredKernelsStatic::OptInlineAsm; | ||||
| #else | ||||
|   QCD::StaggeredKernelsStatic::Opt=QCD::StaggeredKernelsStatic::OptGeneric; | ||||
| #endif | ||||
|  | ||||
|   t0=usecond(); | ||||
|   for(int i=0;i<ncall1;i++){ | ||||
|     Ds.Dhop(src,tmp,0); | ||||
|   } | ||||
|   t1=usecond(); | ||||
|   | ||||
|   std::cout<<GridLogMessage << "Called Ds ASM"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm src "<< norm2(src)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm result "<< norm2(tmp)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)<<std::endl; | ||||
|  | ||||
|   err = tmp-result;  | ||||
|   std::cout<<GridLogMessage << "norm diff   "<< norm2(err)<<std::endl; | ||||
|  | ||||
|    | ||||
|   FermionField ssrc  (sFGrid);  localConvert(src,ssrc); | ||||
|   FermionField sresult(sFGrid); sresult=zero; | ||||
|  | ||||
|   QCD::StaggeredKernelsStatic::Opt=QCD::StaggeredKernelsStatic::OptHandUnroll; | ||||
|   t0=usecond(); | ||||
|   for(int i=0;i<ncall1;i++){ | ||||
|     sDs.Dhop(ssrc,sresult,0); | ||||
|   } | ||||
|   t1=usecond(); | ||||
|   localConvert(sresult,tmp); | ||||
|   | ||||
|   std::cout<<GridLogMessage << "Called sDs unroll"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm ssrc "<< norm2(ssrc)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm sresult "<< norm2(sresult)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)<<std::endl; | ||||
|  | ||||
|  | ||||
| #ifdef AVX512 | ||||
|   QCD::StaggeredKernelsStatic::Opt=QCD::StaggeredKernelsStatic::OptInlineAsm; | ||||
| #else | ||||
|   QCD::StaggeredKernelsStatic::Opt=QCD::StaggeredKernelsStatic::OptGeneric; | ||||
| #endif | ||||
|  | ||||
|   err = tmp-result;  | ||||
|   std::cout<<GridLogMessage << "norm diff   "<< norm2(err)<<std::endl; | ||||
|   int extra=1; | ||||
|   t0=usecond(); | ||||
|   for(int i=0;i<ncall1*extra;i++){ | ||||
|     sDs.Dhop(ssrc,sresult,0); | ||||
|   } | ||||
|   t1=usecond(); | ||||
|   localConvert(sresult,tmp); | ||||
|   | ||||
|   std::cout<<GridLogMessage << "Called sDs asm"<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm ssrc   "<< norm2(ssrc)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "norm result "<< norm2(sresult)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t1-t0)*extra<<std::endl; | ||||
|  | ||||
|   err = tmp-result;  | ||||
|   std::cout<<GridLogMessage << "norm diff   "<< norm2(err)<<std::endl; | ||||
|  | ||||
|  | ||||
|  | ||||
|   Grid_finalize(); | ||||
| } | ||||
							
								
								
									
										963
									
								
								tests/solver/Test_compression.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										963
									
								
								tests/solver/Test_compression.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,963 @@ | ||||
|     /************************************************************************************* | ||||
|  | ||||
|     Grid physics library, www.github.com/paboyle/Grid  | ||||
|  | ||||
|     Source file: ./tests/Test_dwf_cg_prec.cc | ||||
|  | ||||
|     Copyright (C) 2015 | ||||
|  | ||||
| Author: Peter Boyle <paboyle@ph.ed.ac.uk> | ||||
|  | ||||
|     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. | ||||
|  | ||||
|     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 */ | ||||
| #include<bitset> | ||||
| #include <Grid/Grid.h> | ||||
|  | ||||
| using namespace std; | ||||
| using namespace Grid; | ||||
| using namespace Grid::QCD; | ||||
|  | ||||
| //Preconditioning:   M psi = chi | ||||
| //               =  M P^-1 P psi = chi | ||||
| //               =  M P^-1 psi' = chi | ||||
|  | ||||
| //Solve for psi' using M P^-1 as operator, then apply  P^-1 psi' = psi  | ||||
|  | ||||
| //Inexact preconditioned CG requires slight modification because we want to avoid computing P^-1 exactly | ||||
|  | ||||
|  | ||||
| ///////////////////////////////////////////////////////////// | ||||
| // Base classes for iterative processes based on operators | ||||
| // single input vec, single output vec. | ||||
| ///////////////////////////////////////////////////////////// | ||||
|  | ||||
| //The compressor | ||||
| #if 0 | ||||
|  | ||||
|  | ||||
| //Basic copy of WilsonCompressor for demonstration | ||||
| template<class _Hspinor,class _Spinor, class projector> | ||||
| class WilsonTestCompressorTemplate | ||||
| { | ||||
|  public: | ||||
|    | ||||
|   int mu,dag;   | ||||
|  | ||||
|   void Point(int p) { mu=p; }; | ||||
|  | ||||
|   WilsonTestCompressorTemplate(int _dag=0){ | ||||
|     //printf("WilsonTestCompressorTemplate constructor\n"); | ||||
|     dag = _dag; | ||||
|   } | ||||
|  | ||||
|   typedef _Spinor         SiteSpinor; | ||||
|   typedef _Hspinor     SiteHalfSpinor; | ||||
|   typedef _Hspinor SiteHalfCommSpinor; | ||||
|   typedef typename SiteHalfSpinor::vector_type     vComplexIn; | ||||
|   constexpr static int Nw=sizeof(SiteHalfSpinor)/sizeof(vComplexIn); //number of complex numbers in SiteHalfSpinor | ||||
|  | ||||
|   inline int CommDatumSize(void) { | ||||
|     //printf("WilsonTestCompressorTemplate CommDatumSize\n"); | ||||
|     return sizeof(SiteHalfCommSpinor); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Compress includes precision change if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   inline void Compress(SiteHalfSpinor *buf,Integer o,const SiteSpinor &in) { | ||||
|     //printf("WilsonTestCompressorTemplate Compress\n"); | ||||
|     projector::Proj(buf[o],in,mu,dag); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Exchange includes precision change if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   inline void Exchange(SiteHalfSpinor *mp, | ||||
|                        SiteHalfSpinor *vp0, | ||||
|                        SiteHalfSpinor *vp1, | ||||
| 		       Integer type,Integer o){ | ||||
|     //printf("WilsonTestCompressorTemplate Exchange\n"); | ||||
|     exchange(mp[2*o],mp[2*o+1],vp0[o],vp1[o],type); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Have a decompression step if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   inline void Decompress(SiteHalfSpinor *out, | ||||
| 			 SiteHalfSpinor *in, Integer o) {     | ||||
|     //printf("WilsonTestCompressorTemplate Decompress\n"); | ||||
|     assert(0); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Compress Exchange                                 */ | ||||
|   /*****************************************************/ | ||||
|   inline void CompressExchange(SiteHalfSpinor *out0, | ||||
| 			       SiteHalfSpinor *out1, | ||||
| 			       const SiteSpinor *in, | ||||
| 			       Integer j,Integer k, Integer m,Integer type){ | ||||
|     //printf("WilsonTestCompressorTemplate CompressExchange\n"); | ||||
|     SiteHalfSpinor temp1, temp2,temp3,temp4; | ||||
|     projector::Proj(temp1,in[k],mu,dag); | ||||
|     projector::Proj(temp2,in[m],mu,dag); | ||||
|     exchange(out0[j],out1[j],temp1,temp2,type); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Pass the info to the stencil */ | ||||
|   /*****************************************************/ | ||||
|   inline bool DecompressionStep(void) { return false; } | ||||
|  | ||||
| }; | ||||
|  | ||||
| #elif 0 | ||||
|  | ||||
|  | ||||
| //Compressor that unpacks vectorized data to scalar | ||||
| template<class _Hspinor,class _Spinor, class projector> | ||||
| class WilsonTestCompressorTemplate | ||||
| { | ||||
|  public: | ||||
|    | ||||
|   int mu,dag;   | ||||
|  | ||||
|   void Point(int p) { mu=p; }; | ||||
|  | ||||
|   WilsonTestCompressorTemplate(int _dag=0){ | ||||
|     dag = _dag; | ||||
|   } | ||||
|  | ||||
|   typedef _Spinor         SiteSpinor; | ||||
|   typedef _Hspinor     SiteHalfSpinor; | ||||
|   typedef _Hspinor SiteHalfCommSpinor; | ||||
|   typedef typename SiteHalfSpinor::vector_type     vComplexIn; | ||||
|   constexpr static int Nw=sizeof(SiteHalfSpinor)/sizeof(vComplexIn); //number of complex numbers in SiteHalfSpinor | ||||
|  | ||||
|   typedef typename SiteHalfSpinor::scalar_object ScalarSiteHalfSpinor; | ||||
|  | ||||
|   constexpr static int Nsimd = vComplexIn::Nsimd(); | ||||
|  | ||||
|   inline int CommDatumSize(void) { | ||||
|     return Nsimd*sizeof(ScalarSiteHalfSpinor); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Compress includes precision change if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   inline void Compress(SiteHalfSpinor *buf,Integer o,const SiteSpinor &in) { | ||||
|     SiteHalfSpinor hsp; | ||||
|     projector::Proj(hsp,in,mu,dag); | ||||
|      | ||||
|     ScalarSiteHalfSpinor* to = (ScalarSiteHalfSpinor*)buf + o*Nsimd; | ||||
|      | ||||
|     std::vector<ScalarSiteHalfSpinor*> extract_args(Nsimd); | ||||
|     for(int i=0;i<Nsimd;i++) extract_args[i] = to+i; | ||||
|     extract1(hsp,extract_args,0); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Exchange includes precision change if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   inline void Exchange(SiteHalfSpinor *mp, | ||||
|                        SiteHalfSpinor *vp0, | ||||
|                        SiteHalfSpinor *vp1, | ||||
| 		       Integer type,Integer o){ | ||||
|     ScalarSiteHalfSpinor* vpp0 = (ScalarSiteHalfSpinor*)vp0 + o*Nsimd; | ||||
|     ScalarSiteHalfSpinor* vpp1 = (ScalarSiteHalfSpinor*)vp1 + o*Nsimd; | ||||
|      | ||||
|     std::vector<ScalarSiteHalfSpinor*> merge_args0(Nsimd), merge_args1(Nsimd); | ||||
|     for(int i=0;i<Nsimd;i++){ | ||||
|       merge_args0[i] = vpp0+i; | ||||
|       merge_args1[i] = vpp1+i; | ||||
|     } | ||||
|  | ||||
|     SiteHalfSpinor vt0,vt1; | ||||
|     merge1(vt0,merge_args0,0); | ||||
|     merge1(vt1,merge_args1,0); | ||||
|  | ||||
|     exchange(mp[2*o],mp[2*o+1],vt0,vt1,type); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Have a decompression step if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   inline void Decompress(SiteHalfSpinor *out, | ||||
| 			 SiteHalfSpinor *in, Integer o) {     | ||||
|     ScalarSiteHalfSpinor* hin = (ScalarSiteHalfSpinor*)in + o*Nsimd; | ||||
|     std::vector<ScalarSiteHalfSpinor*> merge_args(Nsimd); | ||||
|     for(int i=0;i<Nsimd;i++) merge_args[i] = hin+i; | ||||
|     merge1(out[o],merge_args,0); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Compress Exchange                                 */ | ||||
|   /*****************************************************/ | ||||
|   inline void CompressExchange(SiteHalfSpinor *out0, | ||||
| 			       SiteHalfSpinor *out1, | ||||
| 			       const SiteSpinor *in, | ||||
| 			       Integer j,Integer k, Integer m,Integer type){ | ||||
|     SiteHalfSpinor temp1, temp2,temp3,temp4; | ||||
|     projector::Proj(temp1,in[k],mu,dag); | ||||
|     projector::Proj(temp2,in[m],mu,dag); | ||||
|     exchange(temp3,temp4,temp1,temp2,type); | ||||
|  | ||||
|     ScalarSiteHalfSpinor* hout0 = (ScalarSiteHalfSpinor*)out0 + j*Nsimd; | ||||
|     ScalarSiteHalfSpinor* hout1 = (ScalarSiteHalfSpinor*)out1 + j*Nsimd; | ||||
|  | ||||
|     std::vector<ScalarSiteHalfSpinor*> extract_args0(Nsimd), extract_args1(Nsimd); | ||||
|     for(int i=0;i<Nsimd;i++){ | ||||
|       extract_args0[i] = hout0+i; | ||||
|       extract_args1[i] = hout1+i; | ||||
|     } | ||||
|     extract1(temp3,extract_args0,0); | ||||
|     extract1(temp4,extract_args1,0); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Pass the info to the stencil */ | ||||
|   /*****************************************************/ | ||||
|   inline bool DecompressionStep(void) { return true; } | ||||
|  | ||||
| }; | ||||
|  | ||||
| #else | ||||
|  | ||||
| //Access elements of std::complex | ||||
| template<typename T> | ||||
| inline T & cmplx_reim(std::complex<T> &c, const int reim){ | ||||
|   return reinterpret_cast<T(&)[2]>(c)[reim]; | ||||
| } | ||||
|  | ||||
| template<typename T> | ||||
| inline const T & cmplx_reim(const std::complex<T> &c, const int reim){ | ||||
|   return reinterpret_cast<const T(&)[2]>(c)[reim]; | ||||
| } | ||||
|  | ||||
|  | ||||
| //Pack and unpack float/double to fixed point representation of SZ bits | ||||
| template<int SZ> | ||||
| struct signedIntMap{}; | ||||
|  | ||||
| template<> | ||||
| struct signedIntMap<8>{ typedef int8_t type; }; | ||||
| template<> | ||||
| struct signedIntMap<16>{ typedef int16_t type; }; | ||||
|  | ||||
|  | ||||
| template<typename T, int SZ> | ||||
| inline typename signedIntMap<SZ>::type packN(T val){ | ||||
|   return typename signedIntMap<SZ>::type( (1<<(SZ-2) ) * val ); | ||||
| } | ||||
| template<typename T, int SZ> | ||||
| inline T unpackN(typename signedIntMap<SZ>::type val){ | ||||
|   return T(val)/(1<<(SZ-2)); | ||||
| } | ||||
|  | ||||
| template<typename T> | ||||
| struct getHalfSpinorColors{ | ||||
|   //template <typename vtype> using iImplHalfSpinor        = iScalar<iVector<iVector<vtype, Dimension>, Nhs> >; | ||||
|   enum { value = sizeof(typename T::element::element)/sizeof(typename T::element::element::element) }; | ||||
| }; | ||||
|  | ||||
| //Compressor that compresses to a single magnitude and Nhs*Dimension fixed point integers of size packSize bits | ||||
| template<class _Hspinor,class _Spinor, class projector, int packSize = 16> | ||||
| class WilsonTestCompressorTemplate | ||||
| { | ||||
|  public: | ||||
|    | ||||
|   int mu,dag;   | ||||
|  | ||||
|   void Point(int p) { mu=p; }; | ||||
|  | ||||
|   WilsonTestCompressorTemplate(int _dag=0){ | ||||
|     dag = _dag; | ||||
|   } | ||||
|  | ||||
|   typedef _Spinor         SiteSpinor; | ||||
|   typedef _Hspinor     SiteHalfSpinor; | ||||
|   typedef _Hspinor SiteHalfCommSpinor; | ||||
|   typedef typename SiteHalfSpinor::vector_type     vComplexIn; | ||||
|   constexpr static int Nw=sizeof(SiteHalfSpinor)/sizeof(vComplexIn); //number of complex numbers in SiteHalfSpinor | ||||
|  | ||||
|   typedef typename SiteHalfSpinor::scalar_object ScalarSiteHalfSpinor; | ||||
|  | ||||
|   constexpr static int Nsimd = vComplexIn::Nsimd(); | ||||
|   constexpr static int Dimension = getHalfSpinorColors<SiteHalfSpinor>::value; | ||||
|   | ||||
|   typedef typename ScalarSiteHalfSpinor::scalar_type stype; //std::complex | ||||
|   typedef typename stype::value_type srtype; //float/double | ||||
|  | ||||
|   //Pack and unpack *scalar* SiteHalfSpinor objects | ||||
|   void packSpinor(void* tov, const ScalarSiteHalfSpinor &from){ | ||||
|     uint8_t* to = (uint8_t*)tov; | ||||
|     typedef typename signedIntMap<packSize>::type packedType; | ||||
|  | ||||
|     srtype max = 0; | ||||
|     for(int s=0;s<Nhs;s++) | ||||
|       for(int c=0;c<Dimension;c++) | ||||
| 	for(int reim=0;reim<2;reim++) | ||||
| 	  if(fabs(cmplx_reim( from()(s)(c), reim )) > max ) | ||||
| 	    max =  fabs(cmplx_reim( from()(s)(c), reim )) ; | ||||
|    | ||||
|     *( (srtype*)to ) = max; //copy the normalization to the buffer | ||||
|     to += sizeof(srtype); | ||||
|    | ||||
|     packedType *top = (packedType*)to; | ||||
|     packedType p; | ||||
|     srtype q; | ||||
|     for(int s=0;s<Nhs;s++) | ||||
|       for(int c=0;c<Dimension;c++) | ||||
| 	for(int reim=0;reim<2;reim++){ | ||||
| 	  q = cmplx_reim( from()(s)(c), reim ); | ||||
| 	  if(max != 0.) q /= max; | ||||
| 	  *(top++) = packN<srtype,packSize>(q); | ||||
| 	} | ||||
|   } | ||||
|  | ||||
|   void packSpinor(void* tov, const SiteHalfSpinor &from){ | ||||
|     uint8_t* to = (uint8_t*)tov; | ||||
|     std::vector<ScalarSiteHalfSpinor> extracted(Nsimd); | ||||
|     extract(from,extracted); | ||||
|  | ||||
|     static const int incr = sizeof(srtype) + Nhs*Dimension*2*sizeof(typename signedIntMap<packSize>::type); | ||||
|  | ||||
|     for(int i=0;i<Nsimd;i++){ | ||||
|       packSpinor((void*)to, extracted[i]); | ||||
|       to += incr; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  | ||||
|   void unpackSpinor(ScalarSiteHalfSpinor &to, void* fromv){ | ||||
|     uint8_t* from = (uint8_t*)fromv; | ||||
|     typedef typename signedIntMap<packSize>::type packedType; | ||||
|  | ||||
|     srtype norm = *( (srtype*)from );  | ||||
|     from += sizeof(srtype); | ||||
|  | ||||
|     packedType *fromp = (packedType*)from; | ||||
|     srtype q; | ||||
|     for(int s=0;s<Nhs;s++) | ||||
|       for(int c=0;c<Dimension;c++) | ||||
| 	for(int reim=0;reim<2;reim++){ | ||||
| 	  q = unpackN<srtype,packSize>(*(fromp++) ); | ||||
| 	  if(norm != 0.) q *= norm; | ||||
| 	  cmplx_reim( to()(s)(c), reim ) = q; | ||||
| 	} | ||||
|   } | ||||
|  | ||||
|   void unpackSpinor(SiteHalfSpinor &to, void* fromv){ | ||||
|     uint8_t* from = (uint8_t*)fromv; | ||||
|     std::vector<ScalarSiteHalfSpinor> unpacked(Nsimd); | ||||
|  | ||||
|     static const int incr = sizeof(srtype) + Nhs*Dimension*2*sizeof(typename signedIntMap<packSize>::type); | ||||
|  | ||||
|     for(int i=0;i<Nsimd;i++){ | ||||
|       unpackSpinor(unpacked[i],(void*)from); | ||||
|       from += incr; | ||||
|     } | ||||
|  | ||||
|     merge(to,unpacked); | ||||
|   } | ||||
|  | ||||
|   inline int CommDatumSize(void) { | ||||
|     return Nsimd*(  sizeof(srtype) + Nhs*Dimension*2*sizeof(typename signedIntMap<packSize>::type) ); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Compress includes precision change if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   void Compress(SiteHalfSpinor *buf,Integer o,const SiteSpinor &in) { | ||||
|     SiteHalfSpinor hsp; | ||||
|     projector::Proj(hsp,in,mu,dag); | ||||
|      | ||||
|     uint8_t* to = (uint8_t*)buf + o*CommDatumSize(); | ||||
|     packSpinor(to, hsp); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Exchange includes precision change if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   void Exchange(SiteHalfSpinor *mp, | ||||
|                        SiteHalfSpinor *vp0, | ||||
|                        SiteHalfSpinor *vp1, | ||||
| 		       Integer type,Integer o){ | ||||
|     uint8_t* vpp0 = (uint8_t*)vp0 + o*CommDatumSize(); | ||||
|     uint8_t* vpp1 = (uint8_t*)vp1 + o*CommDatumSize(); | ||||
|  | ||||
|     SiteHalfSpinor vt0, vt1; | ||||
|     unpackSpinor(vt0, vpp0); | ||||
|     unpackSpinor(vt1, vpp1); | ||||
|  | ||||
|     exchange(mp[2*o],mp[2*o+1],vt0,vt1,type); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Have a decompression step if mpi data is not same */ | ||||
|   /*****************************************************/ | ||||
|   void Decompress(SiteHalfSpinor *out, | ||||
| 			 SiteHalfSpinor *in, Integer o) {     | ||||
|     uint8_t* hin = (uint8_t*)in + o*CommDatumSize(); | ||||
|     unpackSpinor(out[o],hin); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Compress Exchange                                 */ | ||||
|   /*****************************************************/ | ||||
|   void CompressExchange(SiteHalfSpinor *out0, | ||||
| 			       SiteHalfSpinor *out1, | ||||
| 			       const SiteSpinor *in, | ||||
| 			       Integer j,Integer k, Integer m,Integer type){ | ||||
|     SiteHalfSpinor temp1, temp2,temp3,temp4; | ||||
|     projector::Proj(temp1,in[k],mu,dag); | ||||
|     projector::Proj(temp2,in[m],mu,dag); | ||||
|     exchange(temp3,temp4,temp1,temp2,type); | ||||
|  | ||||
|     uint8_t* hout0 = (uint8_t*)out0 + j*CommDatumSize(); | ||||
|     uint8_t* hout1 = (uint8_t*)out1 + j*CommDatumSize(); | ||||
|     packSpinor(hout0, temp3); | ||||
|     packSpinor(hout1, temp4); | ||||
|   } | ||||
|  | ||||
|   /*****************************************************/ | ||||
|   /* Pass the info to the stencil */ | ||||
|   /*****************************************************/ | ||||
|   inline bool DecompressionStep(void) { return true; } | ||||
|  | ||||
| }; | ||||
|  | ||||
|  | ||||
| #endif | ||||
|  | ||||
|  | ||||
|  | ||||
| template<typename HS,typename S, int packSize> using WilsonTestCompressor = WilsonTestCompressorTemplate<HS,S,WilsonProjector,packSize>; | ||||
|  | ||||
|  | ||||
| template<class vobj,class cobj> | ||||
| class WilsonStencilBasic : public CartesianStencil<vobj,cobj> { | ||||
| public: | ||||
|   double timer0; | ||||
|   double timer1; | ||||
|   double timer2; | ||||
|   double timer3; | ||||
|   double timer4; | ||||
|   double timer5; | ||||
|   double timer6; | ||||
|   uint64_t callsi; | ||||
|   void ZeroCountersi(void) | ||||
|   { | ||||
|     timer0=0; | ||||
|     timer1=0; | ||||
|     timer2=0; | ||||
|     timer3=0; | ||||
|     timer4=0; | ||||
|     timer5=0; | ||||
|     timer6=0; | ||||
|     callsi=0; | ||||
|   } | ||||
|   void Reporti(int calls) | ||||
|   { | ||||
|     if ( timer0 ) std::cout << GridLogMessage << " timer0 (HaloGatherOpt) " <<timer0/calls <<std::endl; | ||||
|     if ( timer1 ) std::cout << GridLogMessage << " timer1 (Communicate)   " <<timer1/calls <<std::endl; | ||||
|     if ( timer2 ) std::cout << GridLogMessage << " timer2 (CommsMerge )   " <<timer2/calls <<std::endl; | ||||
|     if ( timer3 ) std::cout << GridLogMessage << " timer3 (commsMergeShm) " <<timer3/calls <<std::endl; | ||||
|     if ( timer4 ) std::cout << GridLogMessage << " timer4 " <<timer4 <<std::endl; | ||||
|   } | ||||
|  | ||||
|  | ||||
|   std::vector<int> same_node; | ||||
|   std::vector<int> surface_list; | ||||
|  | ||||
|   WilsonStencilBasic(GridBase *grid, | ||||
| 		int npoints, | ||||
| 		int checkerboard, | ||||
| 		const std::vector<int> &directions, | ||||
| 		const std::vector<int> &distances)   | ||||
|     : CartesianStencil<vobj,cobj> (grid,npoints,checkerboard,directions,distances) , | ||||
|     same_node(npoints) | ||||
|   {  | ||||
|     ZeroCountersi(); | ||||
|     surface_list.resize(0); | ||||
|   }; | ||||
|  | ||||
|   void BuildSurfaceList(int Ls,int vol4){ | ||||
|  | ||||
|     // find same node for SHM | ||||
|     // Here we know the distance is 1 for WilsonStencil | ||||
|     for(int point=0;point<this->_npoints;point++){ | ||||
|       same_node[point] = this->SameNode(point); | ||||
|     } | ||||
|      | ||||
|     for(int site = 0 ;site< vol4;site++){ | ||||
|       int local = 1; | ||||
|       for(int point=0;point<this->_npoints;point++){ | ||||
| 	if( (!this->GetNodeLocal(site*Ls,point)) && (!same_node[point]) ){  | ||||
| 	  local = 0; | ||||
| 	} | ||||
|       } | ||||
|       if(local == 0) {  | ||||
| 	surface_list.push_back(site); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  | ||||
|   template < class compressor> | ||||
|   void HaloExchangeOpt(const Lattice<vobj> &source,compressor &compress)  | ||||
|   { | ||||
|     std::vector<std::vector<CommsRequest_t> > reqs; | ||||
|     this->HaloExchangeOptGather(source,compress); | ||||
|     double t1=usecond(); | ||||
|     this->Communicate(); | ||||
|     double t2=usecond(); timer1 += t2-t1; | ||||
|     this->CommsMerge(compress); | ||||
|     double t3=usecond(); timer2 += t3-t2; | ||||
|     this->CommsMergeSHM(compress); | ||||
|     double t4=usecond(); timer3 += t4-t3; | ||||
|   } | ||||
|    | ||||
|   template <class compressor> | ||||
|   void HaloExchangeOptGather(const Lattice<vobj> &source,compressor &compress){ | ||||
|     this->Prepare(); | ||||
|     double t0=usecond(); | ||||
|     this->HaloGatherOpt(source,compress); | ||||
|     double t1=usecond(); | ||||
|     timer0 += t1-t0; | ||||
|     callsi++; | ||||
|   } | ||||
|  | ||||
|   template <class compressor> | ||||
|   void HaloGatherOpt(const Lattice<vobj> &source,compressor &compress) | ||||
|   { | ||||
|     this->halogtime-=usecond(); | ||||
|     this->HaloGather(source,compress); | ||||
|     this->halogtime+=usecond(); | ||||
|   } | ||||
| }; | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| //This is hideous | ||||
| template<class S, int packSize = 16> | ||||
| class WilsonCompressedCommsImpl: public WilsonImpl<S,FundamentalRepresentation,CoeffReal>{ | ||||
| public: | ||||
|   typedef WilsonImpl<S,FundamentalRepresentation,CoeffReal> WilsonBase; | ||||
|  | ||||
| #define INHERIT_BASE(TYPE) typedef typename WilsonBase::TYPE TYPE | ||||
|  | ||||
|   INHERIT_BASE(Gimpl); | ||||
|   INHERIT_GIMPL_TYPES(Gimpl); | ||||
|  | ||||
|   INHERIT_BASE(Coeff_t); | ||||
|    | ||||
|   INHERIT_BASE(SiteSpinor); | ||||
|   INHERIT_BASE(SitePropagator); | ||||
|   INHERIT_BASE(SiteHalfSpinor); | ||||
|   INHERIT_BASE(SiteHalfCommSpinor);     | ||||
|   INHERIT_BASE(SiteDoubledGaugeField); | ||||
|  | ||||
|   INHERIT_BASE(FermionField); | ||||
|   INHERIT_BASE(PropagatorField); | ||||
|   INHERIT_BASE(DoubledGaugeField); | ||||
|  | ||||
|   //typedef WilsonCompressor<SiteHalfCommSpinor,SiteHalfSpinor, SiteSpinor> Compressor; | ||||
|   typedef WilsonTestCompressor<SiteHalfSpinor, SiteSpinor, packSize> Compressor;   | ||||
|  | ||||
|   INHERIT_BASE(ImplParams); | ||||
|   //INHERIT_BASE(StencilImpl); | ||||
|   typedef WilsonStencilBasic<SiteSpinor, SiteHalfSpinor> StencilImpl; | ||||
|  | ||||
|   WilsonCompressedCommsImpl(const ImplParams &p = ImplParams()) : WilsonBase(p){} | ||||
|    | ||||
|   inline void multLink(SiteHalfSpinor &phi, | ||||
| 		       const SiteDoubledGaugeField &U, | ||||
| 		       const SiteHalfSpinor &chi, | ||||
| 		       int mu, | ||||
| 		       StencilEntry *SE, | ||||
| 		       StencilImpl &St) { | ||||
|     mult(&phi(), &U(mu), &chi()); | ||||
|   } | ||||
|  | ||||
| #undef INHERIT_BASE | ||||
| }; | ||||
|  | ||||
| typedef WilsonCompressedCommsImpl<vComplexF,8> WilsonCompressedComms8ImplF; | ||||
| typedef WilsonCompressedCommsImpl<vComplexD,8> WilsonCompressedComms8ImplD; | ||||
| typedef WilsonCompressedCommsImpl<vComplexF,16> WilsonCompressedComms16ImplF; | ||||
| typedef WilsonCompressedCommsImpl<vComplexD,16> WilsonCompressedComms16ImplD; | ||||
|  | ||||
|  | ||||
| #define TO_INSTANTIATE \ | ||||
|   DOIT(WilsonCompressedComms8ImplF)\ | ||||
|   DOIT(WilsonCompressedComms8ImplD)\ | ||||
|   DOIT(WilsonCompressedComms16ImplF)\ | ||||
|   DOIT(WilsonCompressedComms16ImplD) | ||||
|  | ||||
| #include "InstantiateImpl.impl" | ||||
|  | ||||
| #undef TO_INSTANTIATE | ||||
|  | ||||
| typedef DomainWallFermion<WilsonCompressedComms8ImplD> DomainWallFermionCompressedComms8D; | ||||
| typedef DomainWallFermion<WilsonCompressedComms8ImplF> DomainWallFermionCompressedComms8F; | ||||
| typedef DomainWallFermion<WilsonCompressedComms16ImplD> DomainWallFermionCompressedComms16D; | ||||
| typedef DomainWallFermion<WilsonCompressedComms16ImplF> DomainWallFermionCompressedComms16F; | ||||
|  | ||||
| template<typename T> | ||||
| T parse(const std::string &name, std::istream &in){ | ||||
|   std::string p; | ||||
|   in >> p; | ||||
|   assert(p==name); | ||||
|   char eq; | ||||
|   in >> eq; | ||||
|   assert(eq == '='); | ||||
|   T out; | ||||
|   in >> out; | ||||
|   return out; | ||||
| } | ||||
|  | ||||
| int main (int argc, char ** argv) | ||||
| { | ||||
|   Grid_init(&argc,&argv); | ||||
|  | ||||
|   int Ls=8; | ||||
|   RealD mass=0.1; | ||||
|   RealD outer_tol = 1e-8; | ||||
|   RealD inner_tol_full = 1e-5; | ||||
|   RealD inner_tol_half = 1e-5; | ||||
|   RealD inner_tol_16c = 1e-5; | ||||
|   RealD inner_tol_8c = 1e-5; | ||||
|  | ||||
|   RealD relup_delta_full = 0.1; | ||||
|   RealD relup_delta_half = 0.1; | ||||
|   RealD relup_delta_16c = 0.1; | ||||
|   RealD relup_delta_8c = 0.1; | ||||
|  | ||||
|   std::string config_file = ""; | ||||
|  | ||||
|   for(int i=1;i<argc;i++){ | ||||
|     if(std::string(argv[i]) == "--params"){ | ||||
|       std::ifstream f(argv[i+1]); | ||||
|       f.exceptions ( std::ifstream::failbit | std::ifstream::badbit ); | ||||
|       Ls = parse<int>("Ls", f); | ||||
| #define PARSEIT(NM) NM = parse<RealD>(#NM, f) | ||||
|       PARSEIT(mass); | ||||
|       PARSEIT(outer_tol); | ||||
|       PARSEIT(inner_tol_full); | ||||
|       PARSEIT(inner_tol_half); | ||||
|       PARSEIT(inner_tol_16c); | ||||
|       PARSEIT(inner_tol_8c); | ||||
|       PARSEIT(relup_delta_full); | ||||
|       PARSEIT(relup_delta_half); | ||||
|       PARSEIT(relup_delta_16c); | ||||
|       PARSEIT(relup_delta_8c); | ||||
| #undef PARSEIT | ||||
|  | ||||
|       //f >> outer_tol >> inner_tol_full >> inner_tol_half >> inner_tol_16c >> inner_tol_8c;       | ||||
|     }else if(std::string(argv[i]) == "--config"){ | ||||
|       config_file = argv[i+1]; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   GridCartesian         * UGrid   = SpaceTimeGrid::makeFourDimGrid(GridDefaultLatt(), GridDefaultSimd(Nd,vComplexD::Nsimd()),GridDefaultMpi()); | ||||
|   GridRedBlackCartesian * UrbGrid = SpaceTimeGrid::makeFourDimRedBlackGrid(UGrid); | ||||
|   GridCartesian         * FGrid   = SpaceTimeGrid::makeFiveDimGrid(Ls,UGrid); | ||||
|   GridRedBlackCartesian * FrbGrid = SpaceTimeGrid::makeFiveDimRedBlackGrid(Ls,UGrid); | ||||
|  | ||||
|   GridCartesian         * UGrid_f   = SpaceTimeGrid::makeFourDimGrid(GridDefaultLatt(), GridDefaultSimd(Nd,vComplexF::Nsimd()),GridDefaultMpi()); | ||||
|   GridRedBlackCartesian * UrbGrid_f = SpaceTimeGrid::makeFourDimRedBlackGrid(UGrid_f); | ||||
|   GridCartesian         * FGrid_f   = SpaceTimeGrid::makeFiveDimGrid(Ls,UGrid_f); | ||||
|   GridRedBlackCartesian * FrbGrid_f = SpaceTimeGrid::makeFiveDimRedBlackGrid(Ls,UGrid_f); | ||||
|    | ||||
|   std::vector<int> seeds4({1,2,3,4}); | ||||
|   std::vector<int> seeds5({5,6,7,8}); | ||||
|   GridParallelRNG          RNG5(FGrid);  RNG5.SeedFixedIntegers(seeds5); | ||||
|   GridParallelRNG          RNG4(UGrid);  RNG4.SeedFixedIntegers(seeds4); | ||||
|  | ||||
|   LatticeFermionD    src(FGrid); random(RNG5,src); | ||||
|   LatticeFermionD result(FGrid); result=zero; | ||||
|   LatticeGaugeFieldD Umu(UGrid); | ||||
|   LatticeGaugeFieldF Umu_f(UGrid_f);  | ||||
|  | ||||
|   if(config_file.size() > 0){ | ||||
|     FieldMetaData header; | ||||
|     NerscIO::readConfiguration(Umu,header,config_file); | ||||
|   }else{ | ||||
|     SU3::HotConfiguration(RNG4,Umu); | ||||
|   } | ||||
|  | ||||
|   precisionChange(Umu_f,Umu); | ||||
|    | ||||
|   RealD M5=1.8; | ||||
|  | ||||
|   LatticeFermionD    src_o(FrbGrid); | ||||
|   pickCheckerboard(Odd,src_o,src); | ||||
|  | ||||
|   if(0){ //Test preconditioned CG | ||||
|     LatticeFermionD result_o(FrbGrid); | ||||
|     LatticeFermionD result_o_2(FrbGrid); | ||||
|     result_o.checkerboard = Odd; | ||||
|     result_o = zero; | ||||
|     result_o_2.checkerboard = Odd; | ||||
|     result_o_2 = zero; | ||||
|  | ||||
|     DomainWallFermionD Ddwf(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionD,LatticeFermionD> HermOpEO(Ddwf); | ||||
|     //DoNothingLinearOperator<LatticeFermionD> Prec; | ||||
|     //FixedIterConjugateGradientPreconditioner<LatticeFermionD> Prec(HermOpEO, 20); | ||||
|     SloppyConjugateGradientPreconditioner<LatticeFermionD> Prec(HermOpEO, 1e-2, 1000); | ||||
|  | ||||
|     std::cout << "Preconditioned CG" << std::endl; | ||||
|     InexactPreconditionedConjugateGradient<LatticeFermionD> pCG(Prec,1.0e-8,10000); | ||||
|     pCG(HermOpEO,src_o,result_o); | ||||
|  | ||||
|     std::cout << "Starting regular CG" << std::endl; | ||||
|     ConjugateGradient<LatticeFermionD> CG(1.0e-8,10000); | ||||
|     CG(HermOpEO,src_o,result_o_2); | ||||
|  | ||||
|     LatticeFermionD diff_o(FrbGrid); | ||||
|     RealD diff = axpy_norm(diff_o, -1.0, result_o, result_o_2); | ||||
|  | ||||
|     std::cout << "pCG HermOp applications " << pCG.IterationsToComplete << "(outer) + " << Prec.InnerIterations << "(inner) = " << pCG.IterationsToComplete + Prec.InnerIterations << std::endl; | ||||
|     std::cout << "CG HermOp applications " << CG.IterationsToComplete << std::endl; | ||||
|     std::cout << "Diff between results: " << diff << std::endl; | ||||
|   } | ||||
|  | ||||
|   if(0){ //Test compressor | ||||
|     LatticeFermionD result_o(FrbGrid); | ||||
|     LatticeFermionD result_o_2(FrbGrid); | ||||
|     result_o.checkerboard = Odd; | ||||
|     result_o = zero; | ||||
|     result_o_2.checkerboard = Odd; | ||||
|     result_o_2 = zero; | ||||
|  | ||||
|     DomainWallFermionD Ddwf(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionD,LatticeFermionD> HermOpEO(Ddwf); | ||||
|  | ||||
|     DomainWallFermionCompressedComms16D DdwfC(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionCompressedComms16D,LatticeFermionD> HermOpEOC(DdwfC); | ||||
|  | ||||
|     std::cout << "Starting regular CG with compressed operator" << std::endl; | ||||
|     Integer iter1; | ||||
|     { | ||||
|       ConjugateGradient<LatticeFermionD> CG(1.0e-8,10000); | ||||
|       CG.ErrorOnNoConverge = false; | ||||
|       CG(HermOpEOC,src_o,result_o); | ||||
|       iter1 = CG.IterationsToComplete; | ||||
|     } | ||||
|     Integer iter2; | ||||
|     { | ||||
|       std::cout << "Starting regular CG" << std::endl; | ||||
|       ConjugateGradient<LatticeFermionD> CG(1.0e-8,10000); | ||||
|       CG(HermOpEO,src_o,result_o_2); | ||||
|       iter2 = CG.IterationsToComplete; | ||||
|     } | ||||
|  | ||||
|     LatticeFermionD diff_o(FrbGrid); | ||||
|     RealD diff = axpy_norm(diff_o, -1.0, result_o, result_o_2); | ||||
|  | ||||
|     std::cout << "CG HermOp CC applications " << iter1 << std::endl; | ||||
|     std::cout << "CG HermOp applications " << iter2 << std::endl; | ||||
|     std::cout << "Diff between results: " << diff << std::endl; | ||||
|   } | ||||
|    | ||||
|   if(1){ //Compare mixed prec restarted single/single internal with same but with single/compressed | ||||
|     LatticeFermionD result_o_full(FrbGrid); | ||||
|     LatticeFermionD result_o_half(FrbGrid); | ||||
|     LatticeFermionD result_o_16(FrbGrid); | ||||
|     LatticeFermionD result_o_8(FrbGrid); | ||||
|     result_o_full.checkerboard = Odd; | ||||
|     result_o_full = zero; | ||||
|     result_o_16 = result_o_8 = result_o_half = result_o_full; | ||||
|  | ||||
|     //Std | ||||
|     DomainWallFermionD Ddwf(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionD,LatticeFermionD> HermOpEO(Ddwf); | ||||
|  | ||||
|     DomainWallFermionF Ddwf_f(Umu_f,*FGrid_f,*FrbGrid_f,*UGrid_f,*UrbGrid_f,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionF,LatticeFermionF> HermOpEO_f(Ddwf_f); | ||||
|  | ||||
|     //1/2 prec | ||||
|     DomainWallFermionFH Ddwfhalf_f(Umu_f,*FGrid_f,*FrbGrid_f,*UGrid_f,*UrbGrid_f,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionFH,LatticeFermionF> HermOpEOhalf_f(Ddwfhalf_f); | ||||
|  | ||||
|     //16 | ||||
|     DomainWallFermionCompressedComms16F DdwfC16_f(Umu_f,*FGrid_f,*FrbGrid_f,*UGrid_f,*UrbGrid_f,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionCompressedComms16F,LatticeFermionF> HermOpEOC16_f(DdwfC16_f); | ||||
|  | ||||
|     //8 | ||||
|     DomainWallFermionCompressedComms8F DdwfC8_f(Umu_f,*FGrid_f,*FrbGrid_f,*UGrid_f,*UrbGrid_f,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionCompressedComms8F,LatticeFermionF> HermOpEOC8_f(DdwfC8_f); | ||||
|  | ||||
|     #define ALGORITHM_MIXEDCG | ||||
|     //#define ALGORITHM_RELUP | ||||
|     //#define ALGORITHM_SLOPPY_PREC_CG | ||||
|  | ||||
| #ifdef ALGORITHM_MIXEDCG | ||||
|     std::cout << "Starting mixed CG with single/compressed-16 inner\n"; | ||||
|  | ||||
|     Integer inner_16, outer_16, patchup_16; | ||||
|     { | ||||
|       MixedPrecisionConjugateGradient<LatticeFermionD,LatticeFermionF> mCG(outer_tol, 10000, 50, FrbGrid_f, HermOpEOC16_f, HermOpEO); | ||||
|       mCG.InnerTolerance = inner_tol_16c; | ||||
|       mCG(src_o,result_o_16); | ||||
|       inner_16 = mCG.TotalInnerIterations; outer_16 = mCG.TotalOuterIterations; patchup_16 = mCG.TotalFinalStepIterations; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting mixed CG with single/compressed-8 inner\n"; | ||||
|     Integer inner_8, outer_8, patchup_8; | ||||
|     { | ||||
|       MixedPrecisionConjugateGradient<LatticeFermionD,LatticeFermionF> mCG(outer_tol, 10000, 50, FrbGrid_f, HermOpEOC8_f, HermOpEO); | ||||
|       mCG.InnerTolerance = inner_tol_8c; | ||||
|       mCG(src_o,result_o_8); | ||||
|       inner_8 = mCG.TotalInnerIterations; outer_8 = mCG.TotalOuterIterations; patchup_8 = mCG.TotalFinalStepIterations; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting mixed CG with single/half inner\n"; | ||||
|     Integer inner_half, outer_half, patchup_half; | ||||
|     { | ||||
|       MixedPrecisionConjugateGradient<LatticeFermionD,LatticeFermionF> mCG(outer_tol, 10000, 50, FrbGrid_f, HermOpEOhalf_f, HermOpEO); | ||||
|       mCG.InnerTolerance = inner_tol_half; | ||||
|       mCG(src_o,result_o_half); | ||||
|       inner_half = mCG.TotalInnerIterations; outer_half = mCG.TotalOuterIterations; patchup_half = mCG.TotalFinalStepIterations; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting mixed CG with single/single inner\n"; | ||||
|     Integer inner_full, outer_full, patchup_full; | ||||
|     { | ||||
|       MixedPrecisionConjugateGradient<LatticeFermionD,LatticeFermionF> mCG(outer_tol, 10000, 50, FrbGrid_f, HermOpEO_f, HermOpEO); | ||||
|       mCG.InnerTolerance = inner_tol_full; | ||||
|       mCG(src_o,result_o_full); | ||||
|       inner_full = mCG.TotalInnerIterations; outer_full = mCG.TotalOuterIterations; patchup_full = mCG.TotalFinalStepIterations; | ||||
|     } | ||||
| #elif defined(ALGORITHM_RELUP) | ||||
|     std::cout << "Starting relup CG with single/compressed-16 inner\n";     | ||||
|     Integer inner_16, outer_16, patchup_16; | ||||
|     { | ||||
|       ConjugateGradientReliableUpdate<LatticeFermionD,LatticeFermionF> relup(outer_tol, 2000, relup_delta_16c, FrbGrid_f, HermOpEOC16_f, HermOpEO); | ||||
|       relup(src_o,result_o_16); | ||||
|       inner_16 = relup.IterationsToComplete; outer_16 = relup.ReliableUpdatesPerformed; patchup_16 = relup.IterationsToCleanup; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting relup CG with single/compressed-8 inner\n"; | ||||
|     Integer inner_8, outer_8, patchup_8; | ||||
|     { | ||||
|       ConjugateGradientReliableUpdate<LatticeFermionD,LatticeFermionF> relup(outer_tol, 2000, relup_delta_8c, FrbGrid_f, HermOpEOC8_f, HermOpEO); | ||||
|       relup.ErrorOnNoConverge = false; | ||||
|       relup(src_o,result_o_8); | ||||
|       inner_8 = relup.IterationsToComplete; outer_8 = relup.ReliableUpdatesPerformed; patchup_8 = relup.IterationsToCleanup;      | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting relup CG with single/half inner\n"; | ||||
|     Integer inner_half, outer_half, patchup_half; | ||||
|     { | ||||
|       ConjugateGradientReliableUpdate<LatticeFermionD,LatticeFermionF> relup(outer_tol, 2000, relup_delta_half, FrbGrid_f, HermOpEOhalf_f, HermOpEO); | ||||
|       relup(src_o,result_o_half); | ||||
|       inner_half = relup.IterationsToComplete; outer_half = relup.ReliableUpdatesPerformed; patchup_half = relup.IterationsToCleanup; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting relup CG with single/single inner\n"; | ||||
|     Integer inner_full, outer_full, patchup_full; | ||||
|     { | ||||
|       ConjugateGradientReliableUpdate<LatticeFermionD,LatticeFermionF> relup(outer_tol, 2000, relup_delta_full, FrbGrid_f, HermOpEO_f, HermOpEO); | ||||
|       relup(src_o,result_o_full); | ||||
|       inner_full = relup.IterationsToComplete; outer_full = relup.ReliableUpdatesPerformed; patchup_full = relup.IterationsToCleanup; | ||||
|     } | ||||
| #elif defined(ALGORITHM_SLOPPY_PREC_CG) | ||||
|  | ||||
|     std::cout << "Starting sloppy pCG with single/compressed-16 inner\n";     | ||||
|     Integer inner_16, outer_16; | ||||
|     { | ||||
|       SloppyConjugateGradientLowerPrecPreconditioner<LatticeFermionD,LatticeFermionF> prec(HermOpEOC16_f, FrbGrid_f, inner_tol_16c, 1000); | ||||
|       InexactPreconditionedConjugateGradient<LatticeFermionD> CG(prec, outer_tol, 100); | ||||
|       CG(HermOpEO,src_o,result_o_16); | ||||
|       inner_16 = prec.InnerIterations; outer_16 = CG.IterationsToComplete; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting sloppy pCG with single/compressed-8 inner\n";     | ||||
|     Integer inner_8, outer_8; | ||||
|     { | ||||
|       SloppyConjugateGradientLowerPrecPreconditioner<LatticeFermionD,LatticeFermionF> prec(HermOpEOC8_f, FrbGrid_f, inner_tol_8c, 1000); | ||||
|       InexactPreconditionedConjugateGradient<LatticeFermionD> CG(prec, outer_tol, 100); | ||||
|       CG(HermOpEO,src_o,result_o_8); | ||||
|       inner_8 = prec.InnerIterations; outer_8 = CG.IterationsToComplete; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting sloppy pCG with single/half inner\n";     | ||||
|     Integer inner_half, outer_half; | ||||
|     { | ||||
|       SloppyConjugateGradientLowerPrecPreconditioner<LatticeFermionD,LatticeFermionF> prec(HermOpEOhalf_f, FrbGrid_f, inner_tol_half, 1000); | ||||
|       InexactPreconditionedConjugateGradient<LatticeFermionD> CG(prec, outer_tol, 100); | ||||
|       CG(HermOpEO,src_o,result_o_half); | ||||
|       inner_half = prec.InnerIterations; outer_half = CG.IterationsToComplete; | ||||
|     } | ||||
|  | ||||
|     std::cout << "Starting sloppy pCG with single/single inner\n";     | ||||
|     Integer inner_full, outer_full; | ||||
|     { | ||||
|       SloppyConjugateGradientLowerPrecPreconditioner<LatticeFermionD,LatticeFermionF> prec(HermOpEO_f, FrbGrid_f, inner_tol_full, 1000); | ||||
|       InexactPreconditionedConjugateGradient<LatticeFermionD> CG(prec, outer_tol, 100); | ||||
|       CG(HermOpEO,src_o,result_o_full); | ||||
|       inner_full = prec.InnerIterations; outer_full = CG.IterationsToComplete; | ||||
|     } | ||||
|  | ||||
| #endif | ||||
|  | ||||
|     std::cout << "Ls " << Ls << std::endl; | ||||
|     std::cout << "Mass " << mass << std::endl; | ||||
|     std::cout << "Outer tolerance " << outer_tol << std::endl; | ||||
|  | ||||
| #if defined(ALGORITHM_MIXEDCG) || defined(ALGORITHM_SLOPPY_PREC_CG) | ||||
|     std::cout << "Inner tol full " << inner_tol_full << std::endl; | ||||
|     std::cout << "Inner tol 1/2 prec " << inner_tol_half << std::endl; | ||||
|     std::cout << "Inner tol compressed-16 " << inner_tol_16c << std::endl; | ||||
|     std::cout << "Inner tol compressed-8 " << inner_tol_8c << std::endl; | ||||
| #elif defined(ALGORITHM_RELUP) | ||||
|     std::cout << "Relup delta full " << relup_delta_full << std::endl; | ||||
|     std::cout << "Relup delta 1/2 prec " << relup_delta_half << std::endl; | ||||
|     std::cout << "Relup delta compressed-16 " << relup_delta_16c << std::endl; | ||||
|     std::cout << "Relup delta compressed-8 " << relup_delta_8c << std::endl; | ||||
| #endif | ||||
|     LatticeFermionD diff_o(FrbGrid); | ||||
|     RealD diff = axpy_norm(diff_o, -1.0, result_o_16, result_o_full); | ||||
|     std::cout << "Diff between results (s/c16): " << diff << std::endl; | ||||
|  | ||||
|     diff = axpy_norm(diff_o, -1.0, result_o_8, result_o_full); | ||||
|     std::cout << "Diff between results (s/c8): " << diff << std::endl; | ||||
|  | ||||
|     diff = axpy_norm(diff_o, -1.0, result_o_half, result_o_full); | ||||
|     std::cout << "Diff between results (s/h): " << diff << std::endl; | ||||
|  | ||||
| #if defined(ALGORITHM_MIXEDCG) || defined(ALGORITHM_RELUP) | ||||
|     std::cout << "Iterations (s/c16) inner: " << inner_16 << " outer: " << outer_16 << " patchup: " << patchup_16 << std::endl; | ||||
|     std::cout << "Iterations (s/c8) inner: " << inner_8 << " outer: " << outer_8 << " patchup: " << patchup_8 << std::endl; | ||||
|     std::cout << "Iterations (s/h) inner: " << inner_half << " outer: " << outer_half << " patchup: " << patchup_half << std::endl; | ||||
|     std::cout << "Iterations (s/s) inner: " << inner_full << " outer: " << outer_full << " patchup: " << patchup_full << std::endl; | ||||
| #else | ||||
|     std::cout << "Iterations (s/c16) inner: " << inner_16 << " outer: " << outer_16 << std::endl; | ||||
|     std::cout << "Iterations (s/c8) inner: " << inner_8 << " outer: " << outer_8 << std::endl; | ||||
|     std::cout << "Iterations (s/h) inner: " << inner_half << " outer: " << outer_half << std::endl; | ||||
|     std::cout << "Iterations (s/s) inner: " << inner_full << " outer: " << outer_full << std::endl; | ||||
| #endif | ||||
|  | ||||
|   } | ||||
|  | ||||
|  | ||||
|   Grid_finalize(); | ||||
| } | ||||
							
								
								
									
										247
									
								
								tests/solver/Test_pcg.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										247
									
								
								tests/solver/Test_pcg.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,247 @@ | ||||
|     /************************************************************************************* | ||||
|  | ||||
|     Grid physics library, www.github.com/paboyle/Grid  | ||||
|  | ||||
|     Source file: ./tests/Test_pcg.cc | ||||
|  | ||||
|     Copyright (C) 2015 | ||||
|  | ||||
| Author: Peter Boyle <paboyle@ph.ed.ac.uk> | ||||
|  | ||||
|     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. | ||||
|  | ||||
|     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 */ | ||||
| #include<bitset> | ||||
| #include <Grid/Grid.h> | ||||
|  | ||||
| using namespace std; | ||||
| using namespace Grid; | ||||
| using namespace Grid::QCD; | ||||
|  | ||||
| //Preconditioning:   M psi = chi | ||||
| //               =  M P^-1 P psi = chi | ||||
| //               =  M P^-1 psi' = chi | ||||
|  | ||||
| //Solve for psi' using M P^-1 as operator, then apply  P^-1 psi' = psi  | ||||
|  | ||||
| //Inexact preconditioned CG requires slight modification because we want to avoid computing P^-1 exactly | ||||
|  | ||||
|  | ||||
| ///////////////////////////////////////////////////////////// | ||||
| // Base classes for iterative processes based on operators | ||||
| // single input vec, single output vec. | ||||
| ///////////////////////////////////////////////////////////// | ||||
|  | ||||
| template<typename T> | ||||
| T parse(const std::string &name, std::istream &in){ | ||||
|   std::string p; | ||||
|   in >> p; | ||||
|   assert(p==name); | ||||
|   char eq; | ||||
|   in >> eq; | ||||
|   assert(eq == '='); | ||||
|   T out; | ||||
|   in >> out; | ||||
|   return out; | ||||
| } | ||||
|  | ||||
| int main (int argc, char ** argv) | ||||
| { | ||||
|   Grid_init(&argc,&argv); | ||||
|  | ||||
|   int Ls=12; | ||||
|   RealD mass=0.01; | ||||
|   RealD outer_tol = 1e-8; | ||||
|   RealD inner_tol_full = 1e-5; | ||||
|   RealD inner_tol_half = 1e-5; | ||||
|   RealD inner_tol_16c = 1e-5; | ||||
|   RealD inner_tol_8c = 1e-5; | ||||
|  | ||||
|   RealD relup_delta_full = 0.1; | ||||
|   RealD relup_delta_half = 0.1; | ||||
|   RealD relup_delta_16c = 0.1; | ||||
|   RealD relup_delta_8c = 0.1; | ||||
|  | ||||
|   std::string config_file = ""; | ||||
|   RealD lo = 1.0; | ||||
|   RealD hi = 64.0; | ||||
|   int   order=10; | ||||
|   for(int i=1;i<argc;i++){ | ||||
|     if(std::string(argv[i]) == "--params"){ | ||||
|       std::ifstream f(argv[i+1]); | ||||
|       f.exceptions ( std::ifstream::failbit | std::ifstream::badbit ); | ||||
|       Ls = parse<int>("Ls", f); | ||||
| #define PARSEIT(NM) NM = parse<RealD>(#NM, f) | ||||
|       PARSEIT(mass); | ||||
|       PARSEIT(outer_tol); | ||||
|       PARSEIT(inner_tol_full); | ||||
|       PARSEIT(inner_tol_half); | ||||
|       PARSEIT(inner_tol_16c); | ||||
|       PARSEIT(inner_tol_8c); | ||||
|       PARSEIT(relup_delta_full); | ||||
|       PARSEIT(relup_delta_half); | ||||
|       PARSEIT(relup_delta_16c); | ||||
|       PARSEIT(relup_delta_8c); | ||||
| #undef PARSEIT | ||||
|  | ||||
|       //f >> outer_tol >> inner_tol_full >> inner_tol_half >> inner_tol_16c >> inner_tol_8c;       | ||||
|     }else if(std::string(argv[i]) == "--config"){ | ||||
|       config_file = argv[i+1]; | ||||
|     }else if(std::string(argv[i]) == "--order"){ | ||||
|       std::string ss(argv[i+1]); | ||||
|       std::stringstream f(ss); | ||||
|       f>>order; std::cout << " Order poly set to " <<order<<std::endl; | ||||
|     }else if(std::string(argv[i]) == "--lo"){ | ||||
|       std::string ss(argv[i+1]); | ||||
|       std::stringstream f(ss); | ||||
|       f>>lo; std::cout << " Lo poly set to " <<lo<<std::endl; | ||||
|     }else if(std::string(argv[i]) == "--hi"){ | ||||
|       std::string ss(argv[i+1]); | ||||
|       std::stringstream f(ss); | ||||
|       f>>hi; std::cout << " Hi poly set to " <<hi<<std::endl; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   GridCartesian         * UGrid   = SpaceTimeGrid::makeFourDimGrid(GridDefaultLatt(), GridDefaultSimd(Nd,vComplexD::Nsimd()),GridDefaultMpi()); | ||||
|   GridRedBlackCartesian * UrbGrid = SpaceTimeGrid::makeFourDimRedBlackGrid(UGrid); | ||||
|   GridCartesian         * FGrid   = SpaceTimeGrid::makeFiveDimGrid(Ls,UGrid); | ||||
|   GridRedBlackCartesian * FrbGrid = SpaceTimeGrid::makeFiveDimRedBlackGrid(Ls,UGrid); | ||||
|  | ||||
|   GridCartesian         * UGrid_f   = SpaceTimeGrid::makeFourDimGrid(GridDefaultLatt(), GridDefaultSimd(Nd,vComplexF::Nsimd()),GridDefaultMpi()); | ||||
|   GridRedBlackCartesian * UrbGrid_f = SpaceTimeGrid::makeFourDimRedBlackGrid(UGrid_f); | ||||
|   GridCartesian         * FGrid_f   = SpaceTimeGrid::makeFiveDimGrid(Ls,UGrid_f); | ||||
|   GridRedBlackCartesian * FrbGrid_f = SpaceTimeGrid::makeFiveDimRedBlackGrid(Ls,UGrid_f); | ||||
|    | ||||
|   std::vector<int> seeds4({1,2,3,4}); | ||||
|   std::vector<int> seeds5({5,6,7,8}); | ||||
|   GridParallelRNG          RNG5(FGrid);  RNG5.SeedFixedIntegers(seeds5); | ||||
|   GridParallelRNG          RNG4(UGrid);  RNG4.SeedFixedIntegers(seeds4); | ||||
|  | ||||
|   LatticeFermionD    src(FGrid); random(RNG5,src); | ||||
|   LatticeFermionD result(FGrid); result=zero; | ||||
|   LatticeGaugeFieldD Umu(UGrid); | ||||
|   LatticeGaugeFieldF Umu_f(UGrid_f);  | ||||
|  | ||||
|   if(config_file.size() > 0){ | ||||
|     FieldMetaData header; | ||||
|     NerscIO::readConfiguration(Umu,header,config_file); | ||||
|   }else{ | ||||
|     SU3::HotConfiguration(RNG4,Umu); | ||||
|   } | ||||
|  | ||||
|   precisionChange(Umu_f,Umu); | ||||
|    | ||||
|   RealD M5=1.8; | ||||
|  | ||||
|   LatticeFermionD    src_o(FrbGrid); | ||||
|   pickCheckerboard(Odd,src_o,src); | ||||
|  | ||||
|   //if(0){ //Test preconditioned CG | ||||
|     std::cout << "Test preconditioned CG" << std::endl; | ||||
|     LatticeFermionD result_o(FrbGrid); | ||||
|     LatticeFermionD result_o_2(FrbGrid); | ||||
|     result_o.checkerboard = Odd; | ||||
|     result_o = zero; | ||||
|     result_o_2.checkerboard = Odd; | ||||
|     result_o_2 = zero; | ||||
|  | ||||
|     DomainWallFermionD Ddwf(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionD,LatticeFermionD> HermOpEO(Ddwf); | ||||
|     //DoNothingLinearOperator<LatticeFermionD> Prec; | ||||
|     //FixedIterConjugateGradientPreconditioner<LatticeFermionD> Prec(HermOpEO, 20); | ||||
|     //    SloppyConjugateGradientPreconditioner<LatticeFermionD> Prec(HermOpEO, 1e-2, 1000); | ||||
|     PolynomialPreconditioner<LatticeFermionD> Prec(HermOpEO,lo,hi,order) ; | ||||
|  | ||||
|     std::cout << "Preconditioned CG" << std::endl; | ||||
|     InexactPreconditionedConjugateGradient<LatticeFermionD> pCG(Prec,1.0e-8,10000); | ||||
|     pCG(HermOpEO,src_o,result_o); | ||||
|  | ||||
|     std::cout << "Starting regular CG" << std::endl; | ||||
|     ConjugateGradient<LatticeFermionD> CG(1.0e-8,10000); | ||||
|     CG(HermOpEO,src_o,result_o_2); | ||||
|  | ||||
|     LatticeFermionD diff_o(FrbGrid); | ||||
|     RealD diff = axpy_norm(diff_o, -1.0, result_o, result_o_2); | ||||
|  | ||||
|     std::cout << "pCG HermOp applications " << " Lo " << lo << " Hi " << hi << " Order " << order << " " <<  pCG.IterationsToComplete << "(outer) + " << Prec.InnerIterations << "(inner) = " << pCG.IterationsToComplete + Prec.InnerIterations << std::endl; | ||||
|     std::cout << "CG HermOp applications " << CG.IterationsToComplete << std::endl; | ||||
|     std::cout << "Diff between results: " << diff << std::endl; | ||||
|   //} | ||||
|  | ||||
|   if(0){ //Test compressor | ||||
|     LatticeFermionD result_o(FrbGrid); | ||||
|     LatticeFermionD result_o_2(FrbGrid); | ||||
|     result_o.checkerboard = Odd; | ||||
|     result_o = zero; | ||||
|     result_o_2.checkerboard = Odd; | ||||
|     result_o_2 = zero; | ||||
|  | ||||
|     DomainWallFermionD Ddwf(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionD,LatticeFermionD> HermOpEO(Ddwf); | ||||
|  | ||||
|     DomainWallFermionDF DdwfC(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionDF,LatticeFermionD> HermOpEOC(DdwfC); | ||||
|  | ||||
|     std::cout << "Starting regular CG with compressed operator" << std::endl; | ||||
|     Integer iter1; | ||||
|     { | ||||
|       ConjugateGradient<LatticeFermionD> CG(1.0e-8,10000); | ||||
|       CG.ErrorOnNoConverge = false; | ||||
|       CG(HermOpEOC,src_o,result_o); | ||||
|       iter1 = CG.IterationsToComplete; | ||||
|     } | ||||
|     Integer iter2; | ||||
|     { | ||||
|       std::cout << "Starting regular CG" << std::endl; | ||||
|       ConjugateGradient<LatticeFermionD> CG(1.0e-8,10000); | ||||
|       CG(HermOpEO,src_o,result_o_2); | ||||
|       iter2 = CG.IterationsToComplete; | ||||
|     } | ||||
|  | ||||
|     LatticeFermionD diff_o(FrbGrid); | ||||
|     RealD diff = axpy_norm(diff_o, -1.0, result_o, result_o_2); | ||||
|  | ||||
|     std::cout << "CG HermOp CC applications " << iter1 << std::endl; | ||||
|     std::cout << "CG HermOp applications " << iter2 << std::endl; | ||||
|     std::cout << "Diff between results: " << diff << std::endl; | ||||
|   } | ||||
|    | ||||
|   if(1){ //Compare mixed prec restarted single/single internal with same but with single/compressed | ||||
|     LatticeFermionD result_o_full(FrbGrid); | ||||
|     LatticeFermionD result_o_half(FrbGrid); | ||||
|     LatticeFermionD result_o_16(FrbGrid); | ||||
|     LatticeFermionD result_o_8(FrbGrid); | ||||
|     result_o_full.checkerboard = Odd; | ||||
|     result_o_full = zero; | ||||
|     result_o_16 = result_o_8 = result_o_half = result_o_full; | ||||
|  | ||||
|     //Std | ||||
|     DomainWallFermionD Ddwf(Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionD,LatticeFermionD> HermOpEO(Ddwf); | ||||
|  | ||||
|     DomainWallFermionF Ddwf_f(Umu_f,*FGrid_f,*FrbGrid_f,*UGrid_f,*UrbGrid_f,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionF,LatticeFermionF> HermOpEO_f(Ddwf_f); | ||||
|  | ||||
|     //1/2 prec | ||||
|     DomainWallFermionFH Ddwfhalf_f(Umu_f,*FGrid_f,*FrbGrid_f,*UGrid_f,*UrbGrid_f,mass,M5); | ||||
|     SchurDiagMooeeOperator<DomainWallFermionFH,LatticeFermionF> HermOpEOhalf_f(Ddwfhalf_f); | ||||
|  | ||||
|   } | ||||
|  | ||||
|  | ||||
|   Grid_finalize(); | ||||
| } | ||||
| @@ -74,8 +74,16 @@ int main (int argc, char ** argv) | ||||
|  | ||||
|   LatticeGaugeField Umu(UGrid); SU3::HotConfiguration(pRNG,Umu); | ||||
|  | ||||
|   double volume=1; | ||||
|   for(int mu=0;mu<Nd;mu++){ | ||||
|     volume=volume*latt_size[mu]; | ||||
|   }   | ||||
|  | ||||
|   RealD mass=0.003; | ||||
|   ImprovedStaggeredFermion5DR Ds(Umu,Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass);  | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|   ImprovedStaggeredFermion5DR Ds(Umu,Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,c1,c2,u0);  | ||||
|   SchurStaggeredOperator<ImprovedStaggeredFermion5DR,FermionField> HermOp(Ds); | ||||
|  | ||||
|   ConjugateGradient<FermionField> CG(1.0e-8,10000); | ||||
| @@ -87,14 +95,26 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "****************************************************************** "<<std::endl; | ||||
|   std::cout << GridLogMessage << " Calling 4d CG "<<std::endl; | ||||
|   std::cout << GridLogMessage << "****************************************************************** "<<std::endl; | ||||
|   ImprovedStaggeredFermionR Ds4d(Umu,Umu,*UGrid,*UrbGrid,mass); | ||||
|   ImprovedStaggeredFermionR Ds4d(Umu,Umu,*UGrid,*UrbGrid,mass,c1,c2,u0); | ||||
|   SchurStaggeredOperator<ImprovedStaggeredFermionR,FermionField> HermOp4d(Ds4d); | ||||
|   FermionField src4d(UGrid); random(pRNG,src4d); | ||||
|   FermionField src4d_o(UrbGrid);   pickCheckerboard(Odd,src4d_o,src4d); | ||||
|   FermionField result4d_o(UrbGrid);  | ||||
|  | ||||
|   double deodoe_flops=(16*(3*(6+8+8)) + 15*3*2)*volume; // == 66*16 +  == 1146 | ||||
|   result4d_o=zero; | ||||
|   { | ||||
|     double t1=usecond(); | ||||
|     CG(HermOp4d,src4d_o,result4d_o); | ||||
|     double t2=usecond(); | ||||
|     double ncall=CG.IterationsToComplete; | ||||
|     double flops = deodoe_flops * ncall; | ||||
|     std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|     std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|     std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|     HermOp4d.Report(); | ||||
|   } | ||||
|   Ds4d.Report(); | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
|  | ||||
| @@ -103,7 +123,17 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|   Ds.ZeroCounters(); | ||||
|   result_o=zero; | ||||
|   { | ||||
|     double t1=usecond(); | ||||
|     CG(HermOp,src_o,result_o); | ||||
|     double t2=usecond(); | ||||
|     double ncall=CG.IterationsToComplete*Ls; | ||||
|     double flops = deodoe_flops * ncall; | ||||
|     std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|     std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|     std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|     HermOp.Report(); | ||||
|   } | ||||
|   Ds.Report(); | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
| @@ -112,7 +142,18 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|   Ds.ZeroCounters(); | ||||
|   result_o=zero; | ||||
|   { | ||||
|     double t1=usecond(); | ||||
|     mCG(HermOp,src_o,result_o); | ||||
|     double t2=usecond(); | ||||
|     double ncall=mCG.IterationsToComplete*Ls; | ||||
|     double flops = deodoe_flops * ncall; | ||||
|     std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|     std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|     std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|     HermOp.Report(); | ||||
|   } | ||||
|  | ||||
|   Ds.Report(); | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
| @@ -121,7 +162,17 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|   Ds.ZeroCounters(); | ||||
|   result_o=zero; | ||||
|   { | ||||
|     double t1=usecond(); | ||||
|     BCGrQ(HermOp,src_o,result_o); | ||||
|     double t2=usecond(); | ||||
|     double ncall=BCGrQ.IterationsToComplete*Ls; | ||||
|     double flops = deodoe_flops * ncall; | ||||
|     std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|     std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|     std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|     HermOp.Report(); | ||||
|   } | ||||
|   Ds.Report(); | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
|   | ||||
| @@ -74,7 +74,16 @@ int main (int argc, char ** argv) | ||||
|   LatticeGaugeField Umu(UGrid); SU3::HotConfiguration(pRNG,Umu); | ||||
|  | ||||
|   RealD mass=0.003; | ||||
|   ImprovedStaggeredFermion5DR Ds(Umu,Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass);  | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|  | ||||
|   double volume=1; | ||||
|   for(int mu=0;mu<Nd;mu++){ | ||||
|     volume=volume*latt_size[mu]; | ||||
|   } | ||||
|  | ||||
|   ImprovedStaggeredFermion5DR Ds(Umu,Umu,*FGrid,*FrbGrid,*UGrid,*UrbGrid,mass,c1,c2,u0);  | ||||
|   MdagMLinearOperator<ImprovedStaggeredFermion5DR,FermionField> HermOp(Ds); | ||||
|  | ||||
|   ConjugateGradient<FermionField> CG(1.0e-8,10000); | ||||
| @@ -86,11 +95,23 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "****************************************************************** "<<std::endl; | ||||
|   std::cout << GridLogMessage << " Calling 4d CG "<<std::endl; | ||||
|   std::cout << GridLogMessage << "****************************************************************** "<<std::endl; | ||||
|   ImprovedStaggeredFermionR Ds4d(Umu,Umu,*UGrid,*UrbGrid,mass); | ||||
|   ImprovedStaggeredFermionR Ds4d(Umu,Umu,*UGrid,*UrbGrid,mass,c1,c2,u0); | ||||
|   MdagMLinearOperator<ImprovedStaggeredFermionR,FermionField> HermOp4d(Ds4d); | ||||
|   FermionField src4d(UGrid); random(pRNG,src4d); | ||||
|   FermionField result4d(UGrid); result4d=zero; | ||||
|  | ||||
|   double deodoe_flops=(16*(3*(6+8+8)) + 15*3*2)*volume; // == 66*16 +  == 1146 | ||||
|   { | ||||
|     double t1=usecond(); | ||||
|     CG(HermOp4d,src4d,result4d); | ||||
|     double t2=usecond(); | ||||
|     double ncall=CG.IterationsToComplete; | ||||
|     double flops = deodoe_flops * ncall; | ||||
|     std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|     std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|     std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|    } | ||||
|  | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
|  | ||||
| @@ -98,9 +119,18 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << " Calling 5d CG for "<<Ls <<" right hand sides" <<std::endl; | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|   result=zero; | ||||
| { | ||||
|   Ds.ZeroCounters(); | ||||
|   double t1=usecond(); | ||||
|   CG(HermOp,src,result); | ||||
|   double t2=usecond(); | ||||
|   double ncall=CG.IterationsToComplete; | ||||
|   double flops = deodoe_flops * ncall; | ||||
|   std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|     std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|     std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|   Ds.Report(); | ||||
| } | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
| @@ -108,7 +138,16 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|   result=zero; | ||||
|   Ds.ZeroCounters(); | ||||
| { | ||||
|   double t1=usecond(); | ||||
|   mCG(HermOp,src,result); | ||||
|   double t2=usecond(); | ||||
|   double ncall=CG.IterationsToComplete; | ||||
|   double flops = deodoe_flops * ncall; | ||||
|   std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
| } | ||||
|   Ds.Report(); | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
| @@ -117,7 +156,16 @@ int main (int argc, char ** argv) | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|   result=zero; | ||||
|   Ds.ZeroCounters(); | ||||
| { | ||||
|   double t1=usecond(); | ||||
|   BCGrQ(HermOp,src,result); | ||||
|   double t2=usecond(); | ||||
|   double ncall=CG.IterationsToComplete; | ||||
|   double flops = deodoe_flops * ncall; | ||||
|   std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
| } | ||||
|   Ds.Report(); | ||||
|   std::cout << GridLogMessage << "************************************************************************ "<<std::endl; | ||||
|  | ||||
|   | ||||
| @@ -71,7 +71,10 @@ int main (int argc, char ** argv) | ||||
|   }   | ||||
|    | ||||
|   RealD mass=0.003; | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass); | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass,c1,c2,u0); | ||||
|  | ||||
|   FermionField res_o(&RBGrid);  | ||||
|   FermionField src_o(&RBGrid);  | ||||
| @@ -80,7 +83,19 @@ int main (int argc, char ** argv) | ||||
|  | ||||
|   SchurStaggeredOperator<ImprovedStaggeredFermionR,FermionField> HermOpEO(Ds); | ||||
|   ConjugateGradient<FermionField> CG(1.0e-8,10000); | ||||
|   double t1=usecond(); | ||||
|   CG(HermOpEO,src_o,res_o); | ||||
|   double t2=usecond(); | ||||
|  | ||||
|   // Schur solver: uses DeoDoe => volume * 1146 | ||||
|   double ncall=CG.IterationsToComplete; | ||||
|   double flops=(16*(3*(6+8+8)) + 15*3*2)*volume*ncall; // == 66*16 +  == 1146 | ||||
|  | ||||
|   std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|  | ||||
|  | ||||
|  | ||||
|   FermionField tmp(&RBGrid); | ||||
|  | ||||
|   | ||||
| @@ -65,7 +65,10 @@ int main (int argc, char ** argv) | ||||
|   FermionField  resid(&Grid);  | ||||
|  | ||||
|   RealD mass=0.1; | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass); | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass,c1,c2,u0); | ||||
|  | ||||
|   ConjugateGradient<FermionField> CG(1.0e-8,10000); | ||||
|   SchurRedBlackStaggeredSolve<FermionField> SchurSolver(CG); | ||||
|   | ||||
| @@ -73,7 +73,10 @@ int main (int argc, char ** argv) | ||||
|   }   | ||||
|    | ||||
|   RealD mass=0.1; | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass); | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass,c1,c2,u0); | ||||
|  | ||||
|   MdagMLinearOperator<ImprovedStaggeredFermionR,FermionField> HermOp(Ds); | ||||
|   ConjugateGradient<FermionField> CG(1.0e-6,10000); | ||||
|   | ||||
							
								
								
									
										121
									
								
								tests/solver/Test_staggered_multishift.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								tests/solver/Test_staggered_multishift.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | ||||
|     /************************************************************************************* | ||||
|  | ||||
|     Grid physics library, www.github.com/paboyle/Grid  | ||||
|  | ||||
|     Source file: ./tests/Test_wilson_cg_unprec.cc | ||||
|  | ||||
|     Copyright (C) 2015 | ||||
|  | ||||
| Author: Azusa Yamaguchi <ayamaguc@staffmail.ed.ac.uk> | ||||
| Author: Peter Boyle <paboyle@ph.ed.ac.uk> | ||||
|  | ||||
|     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. | ||||
|  | ||||
|     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 */ | ||||
| #include <Grid/Grid.h> | ||||
| #include <Grid/algorithms/iterative/BlockConjugateGradient.h> | ||||
|  | ||||
| using namespace std; | ||||
| using namespace Grid; | ||||
| using namespace Grid::QCD; | ||||
|  | ||||
| template<class d> | ||||
| struct scal { | ||||
|   d internal; | ||||
| }; | ||||
|  | ||||
|   Gamma::Algebra Gmu [] = { | ||||
|     Gamma::Algebra::GammaX, | ||||
|     Gamma::Algebra::GammaY, | ||||
|     Gamma::Algebra::GammaZ, | ||||
|     Gamma::Algebra::GammaT | ||||
|   }; | ||||
|  | ||||
| int main (int argc, char ** argv) | ||||
| { | ||||
|   typedef typename ImprovedStaggeredFermionR::FermionField FermionField;  | ||||
|   typename ImprovedStaggeredFermionR::ImplParams params;  | ||||
|  | ||||
|   Grid_init(&argc,&argv); | ||||
|  | ||||
|   std::vector<int> latt_size   = GridDefaultLatt(); | ||||
|   std::vector<int> simd_layout = GridDefaultSimd(Nd,vComplex::Nsimd()); | ||||
|   std::vector<int> mpi_layout  = GridDefaultMpi(); | ||||
|  | ||||
|   GridCartesian               Grid(latt_size,simd_layout,mpi_layout); | ||||
|   GridRedBlackCartesian     RBGrid(&Grid); | ||||
|  | ||||
|   std::vector<int> seeds({1,2,3,4}); | ||||
|   GridParallelRNG          pRNG(&Grid);  pRNG.SeedFixedIntegers(seeds); | ||||
|  | ||||
|  | ||||
|   LatticeGaugeField Umu(&Grid); SU3::HotConfiguration(pRNG,Umu); | ||||
|  | ||||
|   double volume=1; | ||||
|   for(int mu=0;mu<Nd;mu++){ | ||||
|     volume=volume*latt_size[mu]; | ||||
|   }   | ||||
|  | ||||
|   //////////////////////////////////////// | ||||
|   // sqrt  | ||||
|   //////////////////////////////////////// | ||||
|   double     lo=0.001; | ||||
|   double     hi=1.0; | ||||
|   int precision=64; | ||||
|   int    degree=10; | ||||
|   AlgRemez remez(lo,hi,precision); | ||||
|   remez.generateApprox(degree,1,2); | ||||
|   MultiShiftFunction Sqrt(remez,1.0e-6,false); | ||||
|   std::cout<<GridLogMessage << "Generating degree "<<degree<<" for x^(1/2)"<<std::endl; | ||||
|  | ||||
|  | ||||
|   //////////////////////////////////////////// | ||||
|   // Setup staggered | ||||
|   //////////////////////////////////////////// | ||||
|   RealD mass=0.003; | ||||
|   RealD c1=9.0/8.0; | ||||
|   RealD c2=-1.0/24.0; | ||||
|   RealD u0=1.0; | ||||
|  | ||||
|   ImprovedStaggeredFermionR Ds(Umu,Umu,Grid,RBGrid,mass,c1,c2,u0); | ||||
|   SchurStaggeredOperator<ImprovedStaggeredFermionR,FermionField> HermOpEO(Ds); | ||||
|  | ||||
|   FermionField src(&Grid); random(pRNG,src); | ||||
|   FermionField src_o(&RBGrid);  | ||||
|   pickCheckerboard(Odd,src_o,src); | ||||
|  | ||||
|  | ||||
|   ///////////////////////////////// | ||||
|   //Multishift CG | ||||
|   ///////////////////////////////// | ||||
|   std::vector<FermionField> result(degree,&RBGrid); | ||||
|   ConjugateGradientMultiShift<FermionField> MSCG(10000,Sqrt); | ||||
|  | ||||
|   double deodoe_flops=(1205+15*degree)*volume; // == 66*16 +  == 1146 | ||||
|  | ||||
|   double t1=usecond(); | ||||
|   MSCG(HermOpEO,src_o,result); | ||||
|   double t2=usecond(); | ||||
|   double ncall=MSCG.IterationsToComplete; | ||||
|   double flops = deodoe_flops * ncall; | ||||
|   std::cout<<GridLogMessage << "usec    =   "<< (t2-t1)<<std::endl; | ||||
|   std::cout<<GridLogMessage << "flops   =   "<< flops<<std::endl; | ||||
|   std::cout<<GridLogMessage << "mflop/s =   "<< flops/(t2-t1)<<std::endl; | ||||
|   //  HermOpEO.Report(); | ||||
|  | ||||
|   Grid_finalize(); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user