mirror of
				https://github.com/paboyle/Grid.git
				synced 2025-10-26 01:29:34 +00:00 
			
		
		
		
	Merge branch 'release/0.8.1'
This commit is contained in:
		| @@ -158,8 +158,10 @@ public: | |||||||
|  |  | ||||||
| 	  dbytes=0; | 	  dbytes=0; | ||||||
| 	  ncomm=0; | 	  ncomm=0; | ||||||
|  | #ifdef GRID_OMP | ||||||
| 	  parallel_for(int dir=0;dir<8;dir++){ | #pragma omp parallel for num_threads(Grid::CartesianCommunicator::nCommThreads) | ||||||
|  | #endif | ||||||
|  | 	  for(int dir=0;dir<8;dir++){ | ||||||
|  |  | ||||||
| 	    double tbytes; | 	    double tbytes; | ||||||
| 	    int mu =dir % 4; | 	    int mu =dir % 4; | ||||||
| @@ -175,9 +177,14 @@ public: | |||||||
| 		int comm_proc = mpi_layout[mu]-1; | 		int comm_proc = mpi_layout[mu]-1; | ||||||
| 		Grid.ShiftedRanks(mu,comm_proc,xmit_to_rank,recv_from_rank); | 		Grid.ShiftedRanks(mu,comm_proc,xmit_to_rank,recv_from_rank); | ||||||
| 	      } | 	      } | ||||||
|  | #ifdef GRID_OMP | ||||||
|  | 	int tid = omp_get_thread_num();  | ||||||
|  | #else  | ||||||
|  |         int tid = dir; | ||||||
|  | #endif | ||||||
| 	      tbytes= Grid.StencilSendToRecvFrom((void *)&xbuf[dir][0], xmit_to_rank, | 	      tbytes= Grid.StencilSendToRecvFrom((void *)&xbuf[dir][0], xmit_to_rank, | ||||||
| 						 (void *)&rbuf[dir][0], recv_from_rank, | 						 (void *)&rbuf[dir][0], recv_from_rank, | ||||||
| 						 bytes,dir); | 						 bytes,tid); | ||||||
| 	   | 	   | ||||||
| #ifdef GRID_OMP | #ifdef GRID_OMP | ||||||
| #pragma omp atomic | #pragma omp atomic | ||||||
|   | |||||||
| @@ -169,7 +169,11 @@ int main (int argc, char ** argv) | |||||||
|   for(int lat=4;lat<=maxlat;lat+=4){ |   for(int lat=4;lat<=maxlat;lat+=4){ | ||||||
|     for(int Ls=8;Ls<=8;Ls*=2){ |     for(int Ls=8;Ls<=8;Ls*=2){ | ||||||
|  |  | ||||||
|       std::vector<int> latt_size  ({lat,lat,lat,lat}); |       std::vector<int> latt_size  ({lat*mpi_layout[0], | ||||||
|  |                                     lat*mpi_layout[1], | ||||||
|  |                                     lat*mpi_layout[2], | ||||||
|  |                                     lat*mpi_layout[3]}); | ||||||
|  |  | ||||||
|  |  | ||||||
|       GridCartesian     Grid(latt_size,simd_layout,mpi_layout); |       GridCartesian     Grid(latt_size,simd_layout,mpi_layout); | ||||||
|       RealD Nrank = Grid._Nprocessors; |       RealD Nrank = Grid._Nprocessors; | ||||||
| @@ -446,7 +450,7 @@ int main (int argc, char ** argv) | |||||||
|   }     |   }     | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifdef GRID_OMP | ||||||
|   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; |   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; | ||||||
|   std::cout<<GridLogMessage << "= Benchmarking threaded STENCIL halo exchange in "<<nmu<<" dimensions"<<std::endl; |   std::cout<<GridLogMessage << "= Benchmarking threaded STENCIL halo exchange in "<<nmu<<" dimensions"<<std::endl; | ||||||
|   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; |   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; | ||||||
| @@ -485,7 +489,8 @@ int main (int argc, char ** argv) | |||||||
| 	dbytes=0; | 	dbytes=0; | ||||||
| 	ncomm=0; | 	ncomm=0; | ||||||
|  |  | ||||||
| 	parallel_for(int dir=0;dir<8;dir++){ | #pragma omp parallel for num_threads(Grid::CartesianCommunicator::nCommThreads) | ||||||
|  | 	for(int dir=0;dir<8;dir++){ | ||||||
|  |  | ||||||
| 	  double tbytes; | 	  double tbytes; | ||||||
| 	  int mu =dir % 4; | 	  int mu =dir % 4; | ||||||
| @@ -502,9 +507,9 @@ int main (int argc, char ** argv) | |||||||
| 	      int comm_proc = mpi_layout[mu]-1; | 	      int comm_proc = mpi_layout[mu]-1; | ||||||
| 	      Grid.ShiftedRanks(mu,comm_proc,xmit_to_rank,recv_from_rank); | 	      Grid.ShiftedRanks(mu,comm_proc,xmit_to_rank,recv_from_rank); | ||||||
| 	    } | 	    } | ||||||
|  |             int tid = omp_get_thread_num(); | ||||||
| 	    tbytes= Grid.StencilSendToRecvFrom((void *)&xbuf[dir][0], xmit_to_rank, | 	    tbytes= Grid.StencilSendToRecvFrom((void *)&xbuf[dir][0], xmit_to_rank, | ||||||
| 					       (void *)&rbuf[dir][0], recv_from_rank, bytes,dir); | 					       (void *)&rbuf[dir][0], recv_from_rank, bytes,tid); | ||||||
|  |  | ||||||
| #pragma omp atomic | #pragma omp atomic | ||||||
| 	    dbytes+=tbytes; | 	    dbytes+=tbytes; | ||||||
| @@ -532,7 +537,7 @@ int main (int argc, char ** argv) | |||||||
|   |   | ||||||
|     } |     } | ||||||
|   }     |   }     | ||||||
|  | #endif | ||||||
|   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; |   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; | ||||||
|   std::cout<<GridLogMessage << "= All done; Bye Bye"<<std::endl; |   std::cout<<GridLogMessage << "= All done; Bye Bye"<<std::endl; | ||||||
|   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; |   std::cout<<GridLogMessage << "===================================================================================================="<<std::endl; | ||||||
|   | |||||||
| @@ -340,7 +340,7 @@ case ${ac_PRECISION} in | |||||||
| esac | esac | ||||||
|  |  | ||||||
| ######################  Shared memory allocation technique under MPI3 | ######################  Shared memory allocation technique under MPI3 | ||||||
| AC_ARG_ENABLE([shm],[AC_HELP_STRING([--enable-shm=shmopen|hugetlbfs|shmnone], | AC_ARG_ENABLE([shm],[AC_HELP_STRING([--enable-shm=shmopen|shmget|hugetlbfs|shmnone], | ||||||
|               [Select SHM allocation technique])],[ac_SHM=${enable_shm}],[ac_SHM=shmopen]) |               [Select SHM allocation technique])],[ac_SHM=${enable_shm}],[ac_SHM=shmopen]) | ||||||
|  |  | ||||||
| case ${ac_SHM} in | case ${ac_SHM} in | ||||||
| @@ -349,6 +349,10 @@ case ${ac_SHM} in | |||||||
|      AC_DEFINE([GRID_MPI3_SHMOPEN],[1],[GRID_MPI3_SHMOPEN] ) |      AC_DEFINE([GRID_MPI3_SHMOPEN],[1],[GRID_MPI3_SHMOPEN] ) | ||||||
|      ;; |      ;; | ||||||
|  |  | ||||||
|  |      shmget) | ||||||
|  |      AC_DEFINE([GRID_MPI3_SHMGET],[1],[GRID_MPI3_SHMGET] ) | ||||||
|  |      ;; | ||||||
|  |  | ||||||
|      shmnone) |      shmnone) | ||||||
|      AC_DEFINE([GRID_MPI3_SHM_NONE],[1],[GRID_MPI3_SHM_NONE] ) |      AC_DEFINE([GRID_MPI3_SHM_NONE],[1],[GRID_MPI3_SHM_NONE] ) | ||||||
|      ;; |      ;; | ||||||
| @@ -366,7 +370,7 @@ esac | |||||||
| AC_ARG_ENABLE([shmpath],[AC_HELP_STRING([--enable-shmpath=path], | AC_ARG_ENABLE([shmpath],[AC_HELP_STRING([--enable-shmpath=path], | ||||||
|               [Select SHM mmap base path for hugetlbfs])], |               [Select SHM mmap base path for hugetlbfs])], | ||||||
| 	      [ac_SHMPATH=${enable_shmpath}], | 	      [ac_SHMPATH=${enable_shmpath}], | ||||||
| 	      [ac_SHMPATH=/var/lib/hugetlbfs/pagesize-2MB/]) | 	      [ac_SHMPATH=/var/lib/hugetlbfs/global/pagesize-2MB/]) | ||||||
| AC_DEFINE_UNQUOTED([GRID_SHM_PATH],["$ac_SHMPATH"],[Path to a hugetlbfs filesystem for MMAPing]) | AC_DEFINE_UNQUOTED([GRID_SHM_PATH],["$ac_SHMPATH"],[Path to a hugetlbfs filesystem for MMAPing]) | ||||||
|  |  | ||||||
| ############### communication type selection | ############### communication type selection | ||||||
|   | |||||||
| @@ -114,19 +114,151 @@ void GlobalSharedMemory::Init(Grid_MPI_Comm comm) | |||||||
|   assert(WorldNode!=-1); |   assert(WorldNode!=-1); | ||||||
|   _ShmSetup=1; |   _ShmSetup=1; | ||||||
| } | } | ||||||
|  | // Gray encode support  | ||||||
| void GlobalSharedMemory::OptimalCommunicator(const std::vector<int> &processors,Grid_MPI_Comm & optimal_comm) | int BinaryToGray (int  binary) { | ||||||
|  |   int gray = (binary>>1)^binary; | ||||||
|  |   return gray; | ||||||
|  | } | ||||||
|  | int Log2Size(int TwoToPower,int MAXLOG2) | ||||||
| { | { | ||||||
|   //////////////////////////////////////////////////////////////// |  | ||||||
|   // Assert power of two shm_size. |  | ||||||
|   //////////////////////////////////////////////////////////////// |  | ||||||
|   int log2size = -1; |   int log2size = -1; | ||||||
|   for(int i=0;i<=MAXLOG2RANKSPERNODE;i++){   |   for(int i=0;i<=MAXLOG2;i++){ | ||||||
|     if ( (0x1<<i) == WorldShmSize ) { |     if ( (0x1<<i) == TwoToPower ) { | ||||||
|       log2size = i; |       log2size = i; | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   return log2size; | ||||||
|  | } | ||||||
|  | void GlobalSharedMemory::OptimalCommunicator(const std::vector<int> &processors,Grid_MPI_Comm & optimal_comm) | ||||||
|  | { | ||||||
|  | #undef HYPERCUBE  | ||||||
|  | #ifdef HYPERCUBE | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Assert power of two shm_size. | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   int log2size = Log2Size(WorldShmSize,MAXLOG2RANKSPERNODE); | ||||||
|  |   assert(log2size != -1); | ||||||
|  |  | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Identify the hypercube coordinate of this node using hostname | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // n runs 0...7 9...16 18...25 27...34     (8*4)  5 bits | ||||||
|  |   // i runs 0..7                                    3 bits | ||||||
|  |   // r runs 0..3                                    2 bits | ||||||
|  |   // 2^10 = 1024 nodes | ||||||
|  |   const int maxhdim = 10;  | ||||||
|  |   std::vector<int> HyperCubeCoords(maxhdim,0); | ||||||
|  |   std::vector<int> RootHyperCubeCoords(maxhdim,0); | ||||||
|  |   int R; | ||||||
|  |   int I; | ||||||
|  |   int N; | ||||||
|  |   const int namelen = _POSIX_HOST_NAME_MAX; | ||||||
|  |   char name[namelen]; | ||||||
|  |  | ||||||
|  |   // Parse ICE-XA hostname to get hypercube location | ||||||
|  |   gethostname(name,namelen); | ||||||
|  |   int nscan = sscanf(name,"r%di%dn%d",&R,&I,&N) ; | ||||||
|  |   assert(nscan==3); | ||||||
|  |  | ||||||
|  |   int nlo = N%9; | ||||||
|  |   int nhi = N/9; | ||||||
|  |   uint32_t hypercoor = (R<<8)|(I<<5)|(nhi<<3)|nlo ; | ||||||
|  |   uint32_t rootcoor  = hypercoor; | ||||||
|  |  | ||||||
|  |   ////////////////////////////////////////////////////////////////// | ||||||
|  |   // Print debug info | ||||||
|  |   ////////////////////////////////////////////////////////////////// | ||||||
|  |   for(int d=0;d<maxhdim;d++){ | ||||||
|  |     HyperCubeCoords[d] = (hypercoor>>d)&0x1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   std::string hname(name); | ||||||
|  |   std::cout << "hostname "<<hname<<std::endl; | ||||||
|  |   std::cout << "R " << R << " I " << I << " N "<< N<< | ||||||
|  |             << " hypercoor 0x"<<std::hex<<hypercoor<<std::dec<<std::endl; | ||||||
|  |  | ||||||
|  |   ////////////////////////////////////////////////////////////////// | ||||||
|  |   // broadcast node 0's base coordinate for this partition. | ||||||
|  |   ////////////////////////////////////////////////////////////////// | ||||||
|  |   MPI_Bcast(&rootcoor, sizeof(rootcoor), MPI_BYTE, 0, WorldComm);  | ||||||
|  |   hypercoor=hypercoor-rootcoor; | ||||||
|  |   assert(hypercoor<WorldSize); | ||||||
|  |   assert(hypercoor>=0); | ||||||
|  |  | ||||||
|  |   ////////////////////////////////////// | ||||||
|  |   // Printing | ||||||
|  |   ////////////////////////////////////// | ||||||
|  |   for(int d=0;d<maxhdim;d++){ | ||||||
|  |     HyperCubeCoords[d] = (hypercoor>>d)&0x1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Identify subblock of ranks on node spreading across dims | ||||||
|  |   // in a maximally symmetrical way | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   int ndimension              = processors.size(); | ||||||
|  |   std::vector<int> processor_coor(ndimension); | ||||||
|  |   std::vector<int> WorldDims = processors;   std::vector<int> ShmDims  (ndimension,1);  std::vector<int> NodeDims (ndimension); | ||||||
|  |   std::vector<int> ShmCoor  (ndimension);    std::vector<int> NodeCoor (ndimension);    std::vector<int> WorldCoor(ndimension); | ||||||
|  |   std::vector<int> HyperCoor(ndimension); | ||||||
|  |   int dim = 0; | ||||||
|  |   for(int l2=0;l2<log2size;l2++){ | ||||||
|  |     while ( (WorldDims[dim] / ShmDims[dim]) <= 1 ) dim=(dim+1)%ndimension; | ||||||
|  |     ShmDims[dim]*=2; | ||||||
|  |     dim=(dim+1)%ndimension; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Establish torus of processes and nodes with sub-blockings | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   for(int d=0;d<ndimension;d++){ | ||||||
|  |     NodeDims[d] = WorldDims[d]/ShmDims[d]; | ||||||
|  |   } | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Map Hcube according to physical lattice  | ||||||
|  |   // must partition. Loop over dims and find out who would join. | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   int hcoor = hypercoor; | ||||||
|  |   for(int d=0;d<ndimension;d++){ | ||||||
|  |      int bits = Log2Size(NodeDims[d],MAXLOG2RANKSPERNODE); | ||||||
|  |      int msk  = (0x1<<bits)-1; | ||||||
|  |      HyperCoor[d]=hcoor & msk;   | ||||||
|  |      HyperCoor[d]=BinaryToGray(HyperCoor[d]); // Space filling curve magic | ||||||
|  |      hcoor = hcoor >> bits; | ||||||
|  |   }  | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Check processor counts match | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   int Nprocessors=1; | ||||||
|  |   for(int i=0;i<ndimension;i++){ | ||||||
|  |     Nprocessors*=processors[i]; | ||||||
|  |   } | ||||||
|  |   assert(WorldSize==Nprocessors); | ||||||
|  |  | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Establish mapping between lexico physics coord and WorldRank | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   int rank; | ||||||
|  |  | ||||||
|  |   Lexicographic::CoorFromIndexReversed(NodeCoor,WorldNode   ,NodeDims); | ||||||
|  |  | ||||||
|  |   for(int d=0;d<ndimension;d++) NodeCoor[d]=HyperCoor[d]; | ||||||
|  |  | ||||||
|  |   Lexicographic::CoorFromIndexReversed(ShmCoor ,WorldShmRank,ShmDims); | ||||||
|  |   for(int d=0;d<ndimension;d++) WorldCoor[d] = NodeCoor[d]*ShmDims[d]+ShmCoor[d]; | ||||||
|  |   Lexicographic::IndexFromCoorReversed(WorldCoor,rank,WorldDims); | ||||||
|  |  | ||||||
|  |   ///////////////////////////////////////////////////////////////// | ||||||
|  |   // Build the new communicator | ||||||
|  |   ///////////////////////////////////////////////////////////////// | ||||||
|  |   int ierr= MPI_Comm_split(WorldComm,0,rank,&optimal_comm); | ||||||
|  |   assert(ierr==0); | ||||||
|  | #else  | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   // Assert power of two shm_size. | ||||||
|  |   //////////////////////////////////////////////////////////////// | ||||||
|  |   int log2size = Log2Size(WorldShmSize,MAXLOG2RANKSPERNODE); | ||||||
|   assert(log2size != -1); |   assert(log2size != -1); | ||||||
|  |  | ||||||
|   //////////////////////////////////////////////////////////////// |   //////////////////////////////////////////////////////////////// | ||||||
| @@ -175,7 +307,69 @@ void GlobalSharedMemory::OptimalCommunicator(const std::vector<int> &processors, | |||||||
|   ///////////////////////////////////////////////////////////////// |   ///////////////////////////////////////////////////////////////// | ||||||
|   int ierr= MPI_Comm_split(WorldComm,0,rank,&optimal_comm); |   int ierr= MPI_Comm_split(WorldComm,0,rank,&optimal_comm); | ||||||
|   assert(ierr==0); |   assert(ierr==0); | ||||||
|  | #endif | ||||||
| } | } | ||||||
|  | //////////////////////////////////////////////////////////////////////////////////////////// | ||||||
|  | // SHMGET | ||||||
|  | //////////////////////////////////////////////////////////////////////////////////////////// | ||||||
|  | #ifdef GRID_MPI3_SHMGET | ||||||
|  | void GlobalSharedMemory::SharedMemoryAllocate(uint64_t bytes, int flags) | ||||||
|  | { | ||||||
|  |   std::cout << "SharedMemoryAllocate "<< bytes<< " shmget implementation "<<std::endl; | ||||||
|  |   assert(_ShmSetup==1); | ||||||
|  |   assert(_ShmAlloc==0); | ||||||
|  |  | ||||||
|  |   ////////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||||
|  |   // allocate the shared windows for our group | ||||||
|  |   ////////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||||
|  |   MPI_Barrier(WorldShmComm); | ||||||
|  |   WorldShmCommBufs.resize(WorldShmSize); | ||||||
|  |   std::vector<int> shmids(WorldShmSize); | ||||||
|  |  | ||||||
|  |   if ( WorldShmRank == 0 ) { | ||||||
|  |     for(int r=0;r<WorldShmSize;r++){ | ||||||
|  |       size_t size = bytes; | ||||||
|  |       key_t key   = IPC_PRIVATE; | ||||||
|  |       int flags = IPC_CREAT | SHM_R | SHM_W; | ||||||
|  | #ifdef SHM_HUGETLB | ||||||
|  |       if (Hugepages) flags|=SHM_HUGETLB; | ||||||
|  | #endif | ||||||
|  |       if ((shmids[r]= shmget(key,size, flags)) ==-1) { | ||||||
|  |         int errsv = errno; | ||||||
|  |         printf("Errno %d\n",errsv); | ||||||
|  |         printf("key   %d\n",key); | ||||||
|  |         printf("size  %lld\n",size); | ||||||
|  |         printf("flags %d\n",flags); | ||||||
|  |         perror("shmget"); | ||||||
|  |         exit(1); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   MPI_Barrier(WorldShmComm); | ||||||
|  |   MPI_Bcast(&shmids[0],WorldShmSize*sizeof(int),MPI_BYTE,0,WorldShmComm); | ||||||
|  |   MPI_Barrier(WorldShmComm); | ||||||
|  |  | ||||||
|  |   for(int r=0;r<WorldShmSize;r++){ | ||||||
|  |     WorldShmCommBufs[r] = (uint64_t *)shmat(shmids[r], NULL,0); | ||||||
|  |     if (WorldShmCommBufs[r] == (uint64_t *)-1) { | ||||||
|  |       perror("Shared memory attach failure"); | ||||||
|  |       shmctl(shmids[r], IPC_RMID, NULL); | ||||||
|  |       exit(2); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   MPI_Barrier(WorldShmComm); | ||||||
|  |   /////////////////////////////////// | ||||||
|  |   // Mark for clean up | ||||||
|  |   /////////////////////////////////// | ||||||
|  |   for(int r=0;r<WorldShmSize;r++){ | ||||||
|  |     shmctl(shmids[r], IPC_RMID,(struct shmid_ds *)NULL); | ||||||
|  |   } | ||||||
|  |   MPI_Barrier(WorldShmComm); | ||||||
|  |  | ||||||
|  |   _ShmAlloc=1; | ||||||
|  |   _ShmAllocBytes  = bytes; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|   |   | ||||||
| //////////////////////////////////////////////////////////////////////////////////////////// | //////////////////////////////////////////////////////////////////////////////////////////// | ||||||
| // Hugetlbfs mapping intended | // Hugetlbfs mapping intended | ||||||
| @@ -183,7 +377,7 @@ void GlobalSharedMemory::OptimalCommunicator(const std::vector<int> &processors, | |||||||
| #ifdef GRID_MPI3_SHMMMAP | #ifdef GRID_MPI3_SHMMMAP | ||||||
| void GlobalSharedMemory::SharedMemoryAllocate(uint64_t bytes, int flags) | void GlobalSharedMemory::SharedMemoryAllocate(uint64_t bytes, int flags) | ||||||
| { | { | ||||||
|   std::cout << "SharedMemoryAllocate "<< bytes<< " MMAP implementation "<<std::endl; |   std::cout << "SharedMemoryAllocate "<< bytes<< " MMAP implementation "<< GRID_SHM_PATH <<std::endl; | ||||||
|   assert(_ShmSetup==1); |   assert(_ShmSetup==1); | ||||||
|   assert(_ShmAlloc==0); |   assert(_ShmAlloc==0); | ||||||
|   ////////////////////////////////////////////////////////////////////////////////////////////////////////// |   ////////////////////////////////////////////////////////////////////////////////////////////////////////// | ||||||
| @@ -193,7 +387,7 @@ void GlobalSharedMemory::SharedMemoryAllocate(uint64_t bytes, int flags) | |||||||
|   WorldShmCommBufs.resize(WorldShmSize); |   WorldShmCommBufs.resize(WorldShmSize); | ||||||
|    |    | ||||||
|   //////////////////////////////////////////////////////////////////////////////////////////// |   //////////////////////////////////////////////////////////////////////////////////////////// | ||||||
|   // Hugetlbf and others map filesystems as mappable huge pages |   // Hugetlbfs and others map filesystems as mappable huge pages | ||||||
|   //////////////////////////////////////////////////////////////////////////////////////////// |   //////////////////////////////////////////////////////////////////////////////////////////// | ||||||
|   char shm_name [NAME_MAX]; |   char shm_name [NAME_MAX]; | ||||||
|   for(int r=0;r<WorldShmSize;r++){ |   for(int r=0;r<WorldShmSize;r++){ | ||||||
| @@ -344,6 +538,9 @@ void GlobalSharedMemory::SharedMemoryAllocate(uint64_t bytes, int flags) | |||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   //////////////////////////////////////////////////////// |   //////////////////////////////////////////////////////// | ||||||
|   // Global shared functionality finished |   // Global shared functionality finished | ||||||
|   // Now move to per communicator functionality |   // Now move to per communicator functionality | ||||||
|   | |||||||
| @@ -263,7 +263,7 @@ PARALLEL_CRITICAL | |||||||
| 			      GridBase *grid, | 			      GridBase *grid, | ||||||
| 			      std::vector<fobj> &iodata, | 			      std::vector<fobj> &iodata, | ||||||
| 			      std::string file, | 			      std::string file, | ||||||
| 			      uint64_t offset, | 			      uint64_t& offset, | ||||||
| 			      const std::string &format, int control, | 			      const std::string &format, int control, | ||||||
| 			      uint32_t &nersc_csum, | 			      uint32_t &nersc_csum, | ||||||
| 			      uint32_t &scidac_csuma, | 			      uint32_t &scidac_csuma, | ||||||
| @@ -495,6 +495,7 @@ PARALLEL_CRITICAL | |||||||
| 	  exit(1); | 	  exit(1); | ||||||
| #endif | #endif | ||||||
| 	} | 	} | ||||||
|  |   offset  = fout.tellp(); | ||||||
| 	fout.close(); | 	fout.close(); | ||||||
|       } |       } | ||||||
|       timer.Stop(); |       timer.Stop(); | ||||||
| @@ -699,7 +700,6 @@ PARALLEL_CRITICAL | |||||||
|  |  | ||||||
|     IOobject(w,grid,iodata,file,offset,format,BINARYIO_WRITE|BINARYIO_LEXICOGRAPHIC, |     IOobject(w,grid,iodata,file,offset,format,BINARYIO_WRITE|BINARYIO_LEXICOGRAPHIC, | ||||||
| 	     nersc_csum,scidac_csuma,scidac_csumb); | 	     nersc_csum,scidac_csuma,scidac_csumb); | ||||||
|  |  | ||||||
|     iodata.resize(1); |     iodata.resize(1); | ||||||
|     { |     { | ||||||
|       std::vector<RngStateType> tmp(RngStateCount); |       std::vector<RngStateType> tmp(RngStateCount); | ||||||
|   | |||||||
| @@ -69,39 +69,47 @@ class WilsonCompressorTemplate< _HCspinor, _Hspinor, _Spinor, projector, | |||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   /* Compress includes precision change if mpi data is not same */ |   /* Compress includes precision change if mpi data is not same */ | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   inline void Compress(SiteHalfSpinor *buf,Integer o,const SiteSpinor &in) { |   inline void Compress(SiteHalfSpinor * __restrict__ buf,Integer o,const SiteSpinor &in) { | ||||||
|     projector::Proj(buf[o],in,mu,dag); |     SiteHalfSpinor tmp; | ||||||
|  |     projector::Proj(tmp,in,mu,dag); | ||||||
|  |     vstream(buf[o],tmp); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   /* Exchange includes precision change if mpi data is not same */ |   /* Exchange includes precision change if mpi data is not same */ | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   inline void Exchange(SiteHalfSpinor *mp, |   inline void Exchange(SiteHalfSpinor * __restrict__ mp, | ||||||
|                        SiteHalfSpinor *vp0, |                        const SiteHalfSpinor * __restrict__ vp0, | ||||||
|                        SiteHalfSpinor *vp1, |                        const SiteHalfSpinor * __restrict__ vp1, | ||||||
| 		       Integer type,Integer o){ | 		       Integer type,Integer o){ | ||||||
|     exchange(mp[2*o],mp[2*o+1],vp0[o],vp1[o],type); |     SiteHalfSpinor tmp1; | ||||||
|  |     SiteHalfSpinor tmp2; | ||||||
|  |     exchange(tmp1,tmp2,vp0[o],vp1[o],type); | ||||||
|  |     vstream(mp[2*o  ],tmp1); | ||||||
|  |     vstream(mp[2*o+1],tmp2); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   /* Have a decompression step if mpi data is not same */ |   /* Have a decompression step if mpi data is not same */ | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   inline void Decompress(SiteHalfSpinor *out, |   inline void Decompress(SiteHalfSpinor * __restrict__ out, | ||||||
| 			 SiteHalfSpinor *in, Integer o) {     | 			 SiteHalfSpinor * __restrict__ in, Integer o) {     | ||||||
|     assert(0); |     assert(0); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   /* Compress Exchange                                 */ |   /* Compress Exchange                                 */ | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   inline void CompressExchange(SiteHalfSpinor *out0, |   inline void CompressExchange(SiteHalfSpinor * __restrict__ out0, | ||||||
| 			       SiteHalfSpinor *out1, | 			       SiteHalfSpinor * __restrict__ out1, | ||||||
| 			       const SiteSpinor *in, | 			       const SiteSpinor * __restrict__ in, | ||||||
| 			       Integer j,Integer k, Integer m,Integer type){ | 			       Integer j,Integer k, Integer m,Integer type){ | ||||||
|     SiteHalfSpinor temp1, temp2,temp3,temp4; |     SiteHalfSpinor temp1, temp2,temp3,temp4; | ||||||
|     projector::Proj(temp1,in[k],mu,dag); |     projector::Proj(temp1,in[k],mu,dag); | ||||||
|     projector::Proj(temp2,in[m],mu,dag); |     projector::Proj(temp2,in[m],mu,dag); | ||||||
|     exchange(out0[j],out1[j],temp1,temp2,type); |     exchange(temp3,temp4,temp1,temp2,type); | ||||||
|  |     vstream(out0[j],temp3); | ||||||
|  |     vstream(out1[j],temp4); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /*****************************************************/ |   /*****************************************************/ | ||||||
|   | |||||||
| @@ -30,181 +30,60 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | |||||||
|  |  | ||||||
| #define REGISTER | #define REGISTER | ||||||
|  |  | ||||||
| #define LOAD_CHIMU_BODY(F)			\ | #define LOAD_CHIMU \ | ||||||
|   Chimu_00=ref(F)(0)(0);			\ |   {const SiteSpinor & ref (in._odata[offset]);	\ | ||||||
|   Chimu_01=ref(F)(0)(1);			\ |     Chimu_00=ref()(0)(0);\ | ||||||
|   Chimu_02=ref(F)(0)(2);			\ |     Chimu_01=ref()(0)(1);\ | ||||||
|   Chimu_10=ref(F)(1)(0);			\ |     Chimu_02=ref()(0)(2);\ | ||||||
|   Chimu_11=ref(F)(1)(1);			\ |     Chimu_10=ref()(1)(0);\ | ||||||
|   Chimu_12=ref(F)(1)(2);			\ |     Chimu_11=ref()(1)(1);\ | ||||||
|   Chimu_20=ref(F)(2)(0);			\ |     Chimu_12=ref()(1)(2);\ | ||||||
|   Chimu_21=ref(F)(2)(1);			\ |     Chimu_20=ref()(2)(0);\ | ||||||
|   Chimu_22=ref(F)(2)(2);			\ |     Chimu_21=ref()(2)(1);\ | ||||||
|   Chimu_30=ref(F)(3)(0);			\ |     Chimu_22=ref()(2)(2);\ | ||||||
|   Chimu_31=ref(F)(3)(1);			\ |     Chimu_30=ref()(3)(0);\ | ||||||
|   Chimu_32=ref(F)(3)(2) |     Chimu_31=ref()(3)(1);\ | ||||||
|  |     Chimu_32=ref()(3)(2);} | ||||||
|  |  | ||||||
| #define LOAD_CHIMU(DIR,F,PERM)						\ | #define LOAD_CHI\ | ||||||
|   { const SiteSpinor & ref (in._odata[offset]); LOAD_CHIMU_BODY(F); } |   {const SiteHalfSpinor &ref(buf[offset]);	\ | ||||||
|  |     Chi_00 = ref()(0)(0);\ | ||||||
| #define LOAD_CHI_BODY(F)				\ |     Chi_01 = ref()(0)(1);\ | ||||||
|     Chi_00 = ref(F)(0)(0);\ |     Chi_02 = ref()(0)(2);\ | ||||||
|     Chi_01 = ref(F)(0)(1);\ |     Chi_10 = ref()(1)(0);\ | ||||||
|     Chi_02 = ref(F)(0)(2);\ |     Chi_11 = ref()(1)(1);\ | ||||||
|     Chi_10 = ref(F)(1)(0);\ |     Chi_12 = ref()(1)(2);} | ||||||
|     Chi_11 = ref(F)(1)(1);\ |  | ||||||
|     Chi_12 = ref(F)(1)(2) |  | ||||||
|  |  | ||||||
| #define LOAD_CHI(DIR,F,PERM)					\ |  | ||||||
|   {const SiteHalfSpinor &ref(buf[offset]); LOAD_CHI_BODY(F); } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| //G-parity implementations using in-place intrinsic ops |  | ||||||
|  |  | ||||||
| //1l 1h -> 1h 1l |  | ||||||
| //0l 0h , 1h 1l -> 0l 1h 0h,1l |  | ||||||
| //0h,1l -> 1l,0h |  | ||||||
| //if( (distance == 1 && !perm_will_occur) || (distance == -1 && perm_will_occur) ) |  | ||||||
| //Pulled fermion through forwards face, GPBC on upper component |  | ||||||
| //Need 0= 0l 1h   1= 1l 0h |  | ||||||
| //else if( (distance == -1 && !perm) || (distance == 1 && perm) ) |  | ||||||
| //Pulled fermion through backwards face, GPBC on lower component |  | ||||||
| //Need 0= 1l 0h   1= 0l 1h |  | ||||||
|  |  | ||||||
| //1l 1h -> 1h 1l |  | ||||||
| //0l 0h , 1h 1l -> 0l 1h 0h,1l |  | ||||||
| #define DO_TWIST_0L_1H(INTO,S,C,F, PERM, tmp1, tmp2, tmp3)			\ |  | ||||||
|   permute##PERM(tmp1, ref(1)(S)(C));				\ |  | ||||||
|   exchange##PERM(tmp2,tmp3, ref(0)(S)(C), tmp1);		\ |  | ||||||
|   INTO = tmp2; |  | ||||||
|  |  | ||||||
| //0l 0h -> 0h 0l |  | ||||||
| //1l 1h, 0h 0l -> 1l 0h, 1h 0l |  | ||||||
| #define DO_TWIST_1L_0H(INTO,S,C,F, PERM, tmp1, tmp2, tmp3)			\ |  | ||||||
|   permute##PERM(tmp1, ref(0)(S)(C));				\ |  | ||||||
|   exchange##PERM(tmp2,tmp3, ref(1)(S)(C), tmp1);		\ |  | ||||||
|   INTO = tmp2; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #define LOAD_CHI_SETUP(DIR,F)						\ |  | ||||||
|   g = F;								\ |  | ||||||
|   direction = st._directions[DIR];				\ |  | ||||||
|   distance = st._distances[DIR];				\ |  | ||||||
|   sl = st._grid->_simd_layout[direction];			\ |  | ||||||
|   inplace_twist = 0;						\ |  | ||||||
|   if(SE->_around_the_world && this->Params.twists[DIR % 4]){		\ |  | ||||||
|     if(sl == 1){							\ |  | ||||||
|       g = (F+1) % 2;							\ |  | ||||||
|     }else{								\ |  | ||||||
|       inplace_twist = 1;						\ |  | ||||||
|     }									\ |  | ||||||
|   }   |  | ||||||
|  |  | ||||||
| #define LOAD_CHIMU_GPARITY_INPLACE_TWIST(DIR,F,PERM)			\ |  | ||||||
|   { const SiteSpinor &ref(in._odata[offset]);				\ |  | ||||||
|     LOAD_CHI_SETUP(DIR,F);						\ |  | ||||||
|     if(!inplace_twist){							\ |  | ||||||
|       LOAD_CHIMU_BODY(g);						\ |  | ||||||
|     }else{								\ |  | ||||||
|       if(  ( F==0 && ((distance == 1 && !perm) || (distance == -1 && perm)) ) || \ |  | ||||||
| 	   ( F==1 && ((distance == -1 && !perm) || (distance == 1 && perm)) ) ){ \ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_00,0,0,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_01,0,1,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_02,0,2,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_10,1,0,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_11,1,1,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_12,1,2,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_20,2,0,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_21,2,1,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_22,2,2,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_30,3,0,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_31,3,1,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chimu_32,3,2,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
|       }else{								\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_00,0,0,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_01,0,1,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_02,0,2,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_10,1,0,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_11,1,1,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_12,1,2,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_20,2,0,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_21,2,1,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_22,2,2,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_30,3,0,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_31,3,1,F,PERM,  U_00,U_01,U_10);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chimu_32,3,2,F,PERM,  U_11,U_20,U_21);		\ |  | ||||||
|       } \ |  | ||||||
|     } \ |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #define LOAD_CHI_GPARITY_INPLACE_TWIST(DIR,F,PERM)				\ |  | ||||||
|   { const SiteHalfSpinor &ref(buf[offset]);				\ |  | ||||||
|     LOAD_CHI_SETUP(DIR,F);						\ |  | ||||||
|     if(!inplace_twist){							\ |  | ||||||
|       LOAD_CHI_BODY(g);							\ |  | ||||||
|     }else{								\ |  | ||||||
|       if(  ( F==0 && ((distance == 1 && !perm) || (distance == -1 && perm)) ) || \ |  | ||||||
| 	   ( F==1 && ((distance == -1 && !perm) || (distance == 1 && perm)) ) ){ \ |  | ||||||
| 	DO_TWIST_0L_1H(Chi_00,0,0,F,PERM,  U_00,U_01,U_10);			\ |  | ||||||
| 	DO_TWIST_0L_1H(Chi_01,0,1,F,PERM,  U_11,U_20,U_21);			\ |  | ||||||
| 	DO_TWIST_0L_1H(Chi_02,0,2,F,PERM,  UChi_00,UChi_01,UChi_02);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chi_10,1,0,F,PERM,  UChi_10,UChi_11,UChi_12);		\ |  | ||||||
| 	DO_TWIST_0L_1H(Chi_11,1,1,F,PERM,  U_00,U_01,U_10);			\ |  | ||||||
| 	DO_TWIST_0L_1H(Chi_12,1,2,F,PERM,  U_11,U_20,U_21);			\ |  | ||||||
|       }else{								\ |  | ||||||
| 	DO_TWIST_1L_0H(Chi_00,0,0,F,PERM,  U_00,U_01,U_10);			\ |  | ||||||
| 	DO_TWIST_1L_0H(Chi_01,0,1,F,PERM,  U_11,U_20,U_21);			\ |  | ||||||
| 	DO_TWIST_1L_0H(Chi_02,0,2,F,PERM,  UChi_00,UChi_01,UChi_02);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chi_10,1,0,F,PERM,  UChi_10,UChi_11,UChi_12);		\ |  | ||||||
| 	DO_TWIST_1L_0H(Chi_11,1,1,F,PERM,  U_00,U_01,U_10);			\ |  | ||||||
| 	DO_TWIST_1L_0H(Chi_12,1,2,F,PERM,  U_11,U_20,U_21);			\ |  | ||||||
|       }									\ |  | ||||||
|     }									\ |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #define LOAD_CHI_GPARITY(DIR,F,PERM) LOAD_CHI_GPARITY_INPLACE_TWIST(DIR,F,PERM) |  | ||||||
| #define LOAD_CHIMU_GPARITY(DIR,F,PERM) LOAD_CHIMU_GPARITY_INPLACE_TWIST(DIR,F,PERM) |  | ||||||
|  |  | ||||||
| // To splat or not to splat depends on the implementation | // To splat or not to splat depends on the implementation | ||||||
| #define MULT_2SPIN_BODY \ | #define MULT_2SPIN(A)\ | ||||||
|  |   {auto & ref(U._odata[sU](A));			\ | ||||||
|    Impl::loadLinkElement(U_00,ref()(0,0));	\ |    Impl::loadLinkElement(U_00,ref()(0,0));	\ | ||||||
|    Impl::loadLinkElement(U_10,ref()(1,0));	\ |    Impl::loadLinkElement(U_10,ref()(1,0));	\ | ||||||
|    Impl::loadLinkElement(U_20,ref()(2,0));	\ |    Impl::loadLinkElement(U_20,ref()(2,0));	\ | ||||||
|    Impl::loadLinkElement(U_01,ref()(0,1));	\ |    Impl::loadLinkElement(U_01,ref()(0,1));	\ | ||||||
|    Impl::loadLinkElement(U_11,ref()(1,1));	\ |    Impl::loadLinkElement(U_11,ref()(1,1));	\ | ||||||
|    Impl::loadLinkElement(U_21,ref()(2,1));	\ |    Impl::loadLinkElement(U_21,ref()(2,1));	\ | ||||||
|   UChi_00 = U_00*Chi_00;			\ |     UChi_00 = U_00*Chi_00;\ | ||||||
|   UChi_10 = U_00*Chi_10;			\ |     UChi_10 = U_00*Chi_10;\ | ||||||
|   UChi_01 = U_10*Chi_00;			\ |     UChi_01 = U_10*Chi_00;\ | ||||||
|   UChi_11 = U_10*Chi_10;			\ |     UChi_11 = U_10*Chi_10;\ | ||||||
|   UChi_02 = U_20*Chi_00;			\ |     UChi_02 = U_20*Chi_00;\ | ||||||
|   UChi_12 = U_20*Chi_10;			\ |     UChi_12 = U_20*Chi_10;\ | ||||||
|   UChi_00+= U_01*Chi_01;			\ |     UChi_00+= U_01*Chi_01;\ | ||||||
|   UChi_10+= U_01*Chi_11;			\ |     UChi_10+= U_01*Chi_11;\ | ||||||
|   UChi_01+= U_11*Chi_01;			\ |     UChi_01+= U_11*Chi_01;\ | ||||||
|   UChi_11+= U_11*Chi_11;			\ |     UChi_11+= U_11*Chi_11;\ | ||||||
|   UChi_02+= U_21*Chi_01;			\ |     UChi_02+= U_21*Chi_01;\ | ||||||
|   UChi_12+= U_21*Chi_11;			\ |     UChi_12+= U_21*Chi_11;\ | ||||||
|     Impl::loadLinkElement(U_00,ref()(0,2));	\ |     Impl::loadLinkElement(U_00,ref()(0,2));	\ | ||||||
|     Impl::loadLinkElement(U_10,ref()(1,2));	\ |     Impl::loadLinkElement(U_10,ref()(1,2));	\ | ||||||
|     Impl::loadLinkElement(U_20,ref()(2,2));	\ |     Impl::loadLinkElement(U_20,ref()(2,2));	\ | ||||||
|   UChi_00+= U_00*Chi_02;			\ |     UChi_00+= U_00*Chi_02;\ | ||||||
|   UChi_10+= U_00*Chi_12;			\ |     UChi_10+= U_00*Chi_12;\ | ||||||
|   UChi_01+= U_10*Chi_02;			\ |     UChi_01+= U_10*Chi_02;\ | ||||||
|   UChi_11+= U_10*Chi_12;			\ |     UChi_11+= U_10*Chi_12;\ | ||||||
|   UChi_02+= U_20*Chi_02;			\ |     UChi_02+= U_20*Chi_02;\ | ||||||
|   UChi_12+= U_20*Chi_12 |     UChi_12+= U_20*Chi_12;} | ||||||
|  |  | ||||||
|  |  | ||||||
| #define MULT_2SPIN(A,F)					\ |  | ||||||
|   {auto & ref(U._odata[sU](A)); MULT_2SPIN_BODY; } |  | ||||||
|  |  | ||||||
| #define MULT_2SPIN_GPARITY(A,F)				\ |  | ||||||
|   {auto & ref(U._odata[sU](F)(A)); MULT_2SPIN_BODY; } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #define PERMUTE_DIR(dir)			\ | #define PERMUTE_DIR(dir)			\ | ||||||
| @@ -428,87 +307,84 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | |||||||
|   result_31-= UChi_11;	\ |   result_31-= UChi_11;	\ | ||||||
|   result_32-= UChi_12; |   result_32-= UChi_12; | ||||||
|  |  | ||||||
| #define HAND_STENCIL_LEG(PROJ,PERM,DIR,RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | #define HAND_STENCIL_LEG(PROJ,PERM,DIR,RECON)	\ | ||||||
|   SE=st.GetEntry(ptype,DIR,ss);			\ |   SE=st.GetEntry(ptype,DIR,ss);			\ | ||||||
|   offset = SE->_offset;				\ |   offset = SE->_offset;				\ | ||||||
|   local  = SE->_is_local;			\ |   local  = SE->_is_local;			\ | ||||||
|   perm   = SE->_permute;			\ |   perm   = SE->_permute;			\ | ||||||
|   if ( local ) {				\ |   if ( local ) {				\ | ||||||
|     LOAD_CHIMU_IMPL(DIR,F,PERM);			\ |     LOAD_CHIMU;					\ | ||||||
|     PROJ;					\ |     PROJ;					\ | ||||||
|     if ( perm) {				\ |     if ( perm) {				\ | ||||||
|       PERMUTE_DIR(PERM);			\ |       PERMUTE_DIR(PERM);			\ | ||||||
|     }						\ |     }						\ | ||||||
|   } else {					\ |   } else {					\ | ||||||
|     LOAD_CHI_IMPL(DIR,F,PERM);			\ |     LOAD_CHI;					\ | ||||||
|   }						\ |   }						\ | ||||||
|   MULT_2SPIN_IMPL(DIR,F);			\ |   MULT_2SPIN(DIR);				\ | ||||||
|   RECON;					 |   RECON;					 | ||||||
|  |  | ||||||
|  | #define HAND_STENCIL_LEG_INT(PROJ,PERM,DIR,RECON)	\ | ||||||
| #define HAND_STENCIL_LEG_INT(PROJ,PERM,DIR,RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL)	\ |  | ||||||
|   SE=st.GetEntry(ptype,DIR,ss);			\ |   SE=st.GetEntry(ptype,DIR,ss);			\ | ||||||
|   offset = SE->_offset;				\ |   offset = SE->_offset;				\ | ||||||
|   local  = SE->_is_local;			\ |   local  = SE->_is_local;			\ | ||||||
|   perm   = SE->_permute;			\ |   perm   = SE->_permute;			\ | ||||||
|   if ( local ) {				\ |   if ( local ) {				\ | ||||||
|     LOAD_CHIMU_IMPL(DIR,F,PERM);			\ |     LOAD_CHIMU;					\ | ||||||
|     PROJ;					\ |     PROJ;					\ | ||||||
|     if ( perm) {				\ |     if ( perm) {				\ | ||||||
|       PERMUTE_DIR(PERM);			\ |       PERMUTE_DIR(PERM);			\ | ||||||
|     }						\ |     }						\ | ||||||
|   } else if ( st.same_node[DIR] ) {		\ |   } else if ( st.same_node[DIR] ) {		\ | ||||||
|     LOAD_CHI_IMPL(DIR,F,PERM);			\ |     LOAD_CHI;					\ | ||||||
|   }						\ |   }						\ | ||||||
|   if (local || st.same_node[DIR] ) {		\ |   if (local || st.same_node[DIR] ) {		\ | ||||||
|     MULT_2SPIN_IMPL(DIR,F);			\ |     MULT_2SPIN(DIR);				\ | ||||||
|     RECON;					\ |     RECON;					\ | ||||||
|   } |   } | ||||||
|  |  | ||||||
| #define HAND_STENCIL_LEG_EXT(PROJ,PERM,DIR,RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL)	\ | #define HAND_STENCIL_LEG_EXT(PROJ,PERM,DIR,RECON)	\ | ||||||
|   SE=st.GetEntry(ptype,DIR,ss);			\ |   SE=st.GetEntry(ptype,DIR,ss);			\ | ||||||
|   offset = SE->_offset;				\ |   offset = SE->_offset;				\ | ||||||
|   local  = SE->_is_local;			\ |  | ||||||
|   perm   = SE->_permute;			\ |  | ||||||
|   if((!SE->_is_local)&&(!st.same_node[DIR]) ) {	\ |   if((!SE->_is_local)&&(!st.same_node[DIR]) ) {	\ | ||||||
|     LOAD_CHI_IMPL(DIR,F,PERM);			\ |     LOAD_CHI;					\ | ||||||
|     MULT_2SPIN_IMPL(DIR,F);			\ |     MULT_2SPIN(DIR);				\ | ||||||
|     RECON;					\ |     RECON;					\ | ||||||
|     nmu++;					\ |     nmu++;					\ | ||||||
|   } |   } | ||||||
|  |  | ||||||
| #define HAND_RESULT(ss,F)			\ | #define HAND_RESULT(ss)				\ | ||||||
|   {						\ |   {						\ | ||||||
|     SiteSpinor & ref (out._odata[ss]);		\ |     SiteSpinor & ref (out._odata[ss]);		\ | ||||||
|     vstream(ref(F)(0)(0),result_00);		\ |     vstream(ref()(0)(0),result_00);		\ | ||||||
|     vstream(ref(F)(0)(1),result_01);		\ |     vstream(ref()(0)(1),result_01);		\ | ||||||
|     vstream(ref(F)(0)(2),result_02);		\ |     vstream(ref()(0)(2),result_02);		\ | ||||||
|     vstream(ref(F)(1)(0),result_10);		\ |     vstream(ref()(1)(0),result_10);		\ | ||||||
|     vstream(ref(F)(1)(1),result_11);		\ |     vstream(ref()(1)(1),result_11);		\ | ||||||
|     vstream(ref(F)(1)(2),result_12);		\ |     vstream(ref()(1)(2),result_12);		\ | ||||||
|     vstream(ref(F)(2)(0),result_20);		\ |     vstream(ref()(2)(0),result_20);		\ | ||||||
|     vstream(ref(F)(2)(1),result_21);		\ |     vstream(ref()(2)(1),result_21);		\ | ||||||
|     vstream(ref(F)(2)(2),result_22);		\ |     vstream(ref()(2)(2),result_22);		\ | ||||||
|     vstream(ref(F)(3)(0),result_30);		\ |     vstream(ref()(3)(0),result_30);		\ | ||||||
|     vstream(ref(F)(3)(1),result_31);		\ |     vstream(ref()(3)(1),result_31);		\ | ||||||
|     vstream(ref(F)(3)(2),result_32);		\ |     vstream(ref()(3)(2),result_32);		\ | ||||||
|   } |   } | ||||||
|  |  | ||||||
| #define HAND_RESULT_EXT(ss,F)			\ | #define HAND_RESULT_EXT(ss)			\ | ||||||
|   if (nmu){					\ |   if (nmu){					\ | ||||||
|     SiteSpinor & ref (out._odata[ss]);		\ |     SiteSpinor & ref (out._odata[ss]);		\ | ||||||
|     ref(F)(0)(0)+=result_00;		\ |     ref()(0)(0)+=result_00;		\ | ||||||
|     ref(F)(0)(1)+=result_01;		\ |     ref()(0)(1)+=result_01;		\ | ||||||
|     ref(F)(0)(2)+=result_02;		\ |     ref()(0)(2)+=result_02;		\ | ||||||
|     ref(F)(1)(0)+=result_10;		\ |     ref()(1)(0)+=result_10;		\ | ||||||
|     ref(F)(1)(1)+=result_11;		\ |     ref()(1)(1)+=result_11;		\ | ||||||
|     ref(F)(1)(2)+=result_12;		\ |     ref()(1)(2)+=result_12;		\ | ||||||
|     ref(F)(2)(0)+=result_20;		\ |     ref()(2)(0)+=result_20;		\ | ||||||
|     ref(F)(2)(1)+=result_21;		\ |     ref()(2)(1)+=result_21;		\ | ||||||
|     ref(F)(2)(2)+=result_22;		\ |     ref()(2)(2)+=result_22;		\ | ||||||
|     ref(F)(3)(0)+=result_30;		\ |     ref()(3)(0)+=result_30;		\ | ||||||
|     ref(F)(3)(1)+=result_31;		\ |     ref()(3)(1)+=result_31;		\ | ||||||
|     ref(F)(3)(2)+=result_32;		\ |     ref()(3)(2)+=result_32;		\ | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -587,18 +463,15 @@ WilsonKernels<Impl>::HandDhopSite(StencilImpl &st,LebesgueOrder &lo,DoubledGauge | |||||||
|   int offset,local,perm, ptype; |   int offset,local,perm, ptype; | ||||||
|   StencilEntry *SE; |   StencilEntry *SE; | ||||||
|  |  | ||||||
| #define HAND_DOP_SITE(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ |   HAND_STENCIL_LEG(XM_PROJ,3,Xp,XM_RECON); | ||||||
|   HAND_STENCIL_LEG(XM_PROJ,3,Xp,XM_RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(YM_PROJ,2,Yp,YM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(YM_PROJ,2,Yp,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);	\ |   HAND_STENCIL_LEG(ZM_PROJ,1,Zp,ZM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(ZM_PROJ,1,Zp,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(TM_PROJ,0,Tp,TM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(TM_PROJ,0,Tp,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(XP_PROJ,3,Xm,XP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(XP_PROJ,3,Xm,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(YP_PROJ,2,Ym,YP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(YP_PROJ,2,Ym,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(ZP_PROJ,1,Zm,ZP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(ZP_PROJ,1,Zm,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(TP_PROJ,0,Tm,TP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(TP_PROJ,0,Tm,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_RESULT(ss); | ||||||
|   HAND_RESULT(ss,F) |  | ||||||
|  |  | ||||||
|   HAND_DOP_SITE(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| template<class Impl> | template<class Impl> | ||||||
| @@ -613,18 +486,15 @@ void WilsonKernels<Impl>::HandDhopSiteDag(StencilImpl &st,LebesgueOrder &lo,Doub | |||||||
|   StencilEntry *SE; |   StencilEntry *SE; | ||||||
|   int offset,local,perm, ptype; |   int offset,local,perm, ptype; | ||||||
|    |    | ||||||
| #define HAND_DOP_SITE_DAG(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ |   HAND_STENCIL_LEG(XP_PROJ,3,Xp,XP_RECON); | ||||||
|   HAND_STENCIL_LEG(XP_PROJ,3,Xp,XP_RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(YP_PROJ,2,Yp,YP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(YP_PROJ,2,Yp,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(ZP_PROJ,1,Zp,ZP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(ZP_PROJ,1,Zp,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(TP_PROJ,0,Tp,TP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(TP_PROJ,0,Tp,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(XM_PROJ,3,Xm,XM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(XM_PROJ,3,Xm,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(YM_PROJ,2,Ym,YM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(YM_PROJ,2,Ym,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(ZM_PROJ,1,Zm,ZM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(ZM_PROJ,1,Zm,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG(TM_PROJ,0,Tm,TM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG(TM_PROJ,0,Tm,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_RESULT(ss); | ||||||
|   HAND_RESULT(ss,F) |  | ||||||
|  |  | ||||||
|   HAND_DOP_SITE_DAG(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| template<class Impl> void  | template<class Impl> void  | ||||||
| @@ -639,20 +509,16 @@ WilsonKernels<Impl>::HandDhopSiteInt(StencilImpl &st,LebesgueOrder &lo,DoubledGa | |||||||
|  |  | ||||||
|   int offset,local,perm, ptype; |   int offset,local,perm, ptype; | ||||||
|   StencilEntry *SE; |   StencilEntry *SE; | ||||||
|  |   ZERO_RESULT; | ||||||
| #define HAND_DOP_SITE_INT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ |   HAND_STENCIL_LEG_INT(XM_PROJ,3,Xp,XM_RECON_ACCUM); | ||||||
|   ZERO_RESULT; \ |   HAND_STENCIL_LEG_INT(YM_PROJ,2,Yp,YM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(XM_PROJ,3,Xp,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_INT(ZM_PROJ,1,Zp,ZM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(YM_PROJ,2,Yp,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_INT(TM_PROJ,0,Tp,TM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(ZM_PROJ,1,Zp,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_INT(XP_PROJ,3,Xm,XP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(TM_PROJ,0,Tp,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_INT(YP_PROJ,2,Ym,YP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(XP_PROJ,3,Xm,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_INT(ZP_PROJ,1,Zm,ZP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(YP_PROJ,2,Ym,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_INT(TP_PROJ,0,Tm,TP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(ZP_PROJ,1,Zm,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_RESULT(ss); | ||||||
|   HAND_STENCIL_LEG_INT(TP_PROJ,0,Tm,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |  | ||||||
|   HAND_RESULT(ss,F) |  | ||||||
|  |  | ||||||
|   HAND_DOP_SITE_INT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| template<class Impl> | template<class Impl> | ||||||
| @@ -666,20 +532,16 @@ void WilsonKernels<Impl>::HandDhopSiteDagInt(StencilImpl &st,LebesgueOrder &lo,D | |||||||
|  |  | ||||||
|   StencilEntry *SE; |   StencilEntry *SE; | ||||||
|   int offset,local,perm, ptype; |   int offset,local,perm, ptype; | ||||||
|  |   ZERO_RESULT; | ||||||
| #define HAND_DOP_SITE_DAG_INT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL)				\ |   HAND_STENCIL_LEG_INT(XP_PROJ,3,Xp,XP_RECON_ACCUM); | ||||||
|   ZERO_RESULT;							\ |   HAND_STENCIL_LEG_INT(YP_PROJ,2,Yp,YP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(XP_PROJ,3,Xp,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_STENCIL_LEG_INT(ZP_PROJ,1,Zp,ZP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(YP_PROJ,2,Yp,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_STENCIL_LEG_INT(TP_PROJ,0,Tp,TP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(ZP_PROJ,1,Zp,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_STENCIL_LEG_INT(XM_PROJ,3,Xm,XM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(TP_PROJ,0,Tp,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_STENCIL_LEG_INT(YM_PROJ,2,Ym,YM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(XM_PROJ,3,Xm,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_STENCIL_LEG_INT(ZM_PROJ,1,Zm,ZM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(YM_PROJ,2,Ym,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_STENCIL_LEG_INT(TM_PROJ,0,Tm,TM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_INT(ZM_PROJ,1,Zm,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |   HAND_RESULT(ss); | ||||||
|   HAND_STENCIL_LEG_INT(TM_PROJ,0,Tm,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ |  | ||||||
|   HAND_RESULT(ss,F) |  | ||||||
|    |  | ||||||
|   HAND_DOP_SITE_DAG_INT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| template<class Impl> void  | template<class Impl> void  | ||||||
| @@ -695,20 +557,16 @@ WilsonKernels<Impl>::HandDhopSiteExt(StencilImpl &st,LebesgueOrder &lo,DoubledGa | |||||||
|   int offset,local,perm, ptype; |   int offset,local,perm, ptype; | ||||||
|   StencilEntry *SE; |   StencilEntry *SE; | ||||||
|   int nmu=0; |   int nmu=0; | ||||||
|  |   ZERO_RESULT; | ||||||
| #define HAND_DOP_SITE_EXT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ |   HAND_STENCIL_LEG_EXT(XM_PROJ,3,Xp,XM_RECON_ACCUM); | ||||||
|   ZERO_RESULT; \ |   HAND_STENCIL_LEG_EXT(YM_PROJ,2,Yp,YM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(XM_PROJ,3,Xp,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(ZM_PROJ,1,Zp,ZM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(YM_PROJ,2,Yp,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(TM_PROJ,0,Tp,TM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(ZM_PROJ,1,Zp,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(XP_PROJ,3,Xm,XP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(TM_PROJ,0,Tp,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(YP_PROJ,2,Ym,YP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(XP_PROJ,3,Xm,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(ZP_PROJ,1,Zm,ZP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(YP_PROJ,2,Ym,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(TP_PROJ,0,Tm,TP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(ZP_PROJ,1,Zm,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_RESULT_EXT(ss); | ||||||
|   HAND_STENCIL_LEG_EXT(TP_PROJ,0,Tm,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |  | ||||||
|   HAND_RESULT_EXT(ss,F) |  | ||||||
|  |  | ||||||
|   HAND_DOP_SITE_EXT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| template<class Impl> | template<class Impl> | ||||||
| @@ -723,193 +581,18 @@ void WilsonKernels<Impl>::HandDhopSiteDagExt(StencilImpl &st,LebesgueOrder &lo,D | |||||||
|   StencilEntry *SE; |   StencilEntry *SE; | ||||||
|   int offset,local,perm, ptype; |   int offset,local,perm, ptype; | ||||||
|   int nmu=0; |   int nmu=0; | ||||||
|  |   ZERO_RESULT; | ||||||
| #define HAND_DOP_SITE_DAG_EXT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ |   HAND_STENCIL_LEG_EXT(XP_PROJ,3,Xp,XP_RECON_ACCUM); | ||||||
|   ZERO_RESULT; \ |   HAND_STENCIL_LEG_EXT(YP_PROJ,2,Yp,YP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(XP_PROJ,3,Xp,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(ZP_PROJ,1,Zp,ZP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(YP_PROJ,2,Yp,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(TP_PROJ,0,Tp,TP_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(ZP_PROJ,1,Zp,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(XM_PROJ,3,Xm,XM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(TP_PROJ,0,Tp,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(YM_PROJ,2,Ym,YM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(XM_PROJ,3,Xm,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(ZM_PROJ,1,Zm,ZM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(YM_PROJ,2,Ym,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_STENCIL_LEG_EXT(TM_PROJ,0,Tm,TM_RECON_ACCUM); | ||||||
|   HAND_STENCIL_LEG_EXT(ZM_PROJ,1,Zm,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |   HAND_RESULT_EXT(ss); | ||||||
|   HAND_STENCIL_LEG_EXT(TM_PROJ,0,Tm,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ |  | ||||||
|   HAND_RESULT_EXT(ss,F) |  | ||||||
|  |  | ||||||
|   HAND_DOP_SITE_DAG_EXT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|   //////////////////////////////////////////////// |  | ||||||
|   // Specialise Gparity to simple implementation |  | ||||||
|   //////////////////////////////////////////////// |  | ||||||
| #define HAND_SPECIALISE_EMPTY(IMPL)					\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSite(StencilImpl &st,			\ |  | ||||||
| 				    LebesgueOrder &lo,			\ |  | ||||||
| 				    DoubledGaugeField &U,		\ |  | ||||||
| 				    SiteHalfSpinor *buf,		\ |  | ||||||
| 				    int sF,int sU,			\ |  | ||||||
| 				    const FermionField &in,		\ |  | ||||||
| 				    FermionField &out){ assert(0); }	\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteDag(StencilImpl &st,			\ |  | ||||||
| 				    LebesgueOrder &lo,			\ |  | ||||||
| 				    DoubledGaugeField &U,		\ |  | ||||||
| 				    SiteHalfSpinor *buf,		\ |  | ||||||
| 				    int sF,int sU,			\ |  | ||||||
| 				    const FermionField &in,		\ |  | ||||||
| 				    FermionField &out){ assert(0); }	\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteInt(StencilImpl &st,			\ |  | ||||||
| 				    LebesgueOrder &lo,			\ |  | ||||||
| 				    DoubledGaugeField &U,		\ |  | ||||||
| 				    SiteHalfSpinor *buf,		\ |  | ||||||
| 				    int sF,int sU,			\ |  | ||||||
| 				    const FermionField &in,		\ |  | ||||||
| 				    FermionField &out){ assert(0); }	\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteExt(StencilImpl &st,			\ |  | ||||||
| 				    LebesgueOrder &lo,			\ |  | ||||||
| 				    DoubledGaugeField &U,		\ |  | ||||||
| 				    SiteHalfSpinor *buf,		\ |  | ||||||
| 				    int sF,int sU,			\ |  | ||||||
| 				    const FermionField &in,		\ |  | ||||||
| 				    FermionField &out){ assert(0); }	\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteDagInt(StencilImpl &st,	       	\ |  | ||||||
| 				    LebesgueOrder &lo,			\ |  | ||||||
| 				    DoubledGaugeField &U,		\ |  | ||||||
| 				    SiteHalfSpinor *buf,		\ |  | ||||||
| 				    int sF,int sU,			\ |  | ||||||
| 				    const FermionField &in,		\ |  | ||||||
| 				    FermionField &out){ assert(0); }	\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteDagExt(StencilImpl &st,	       	\ |  | ||||||
| 				    LebesgueOrder &lo,			\ |  | ||||||
| 				    DoubledGaugeField &U,		\ |  | ||||||
| 				    SiteHalfSpinor *buf,		\ |  | ||||||
| 				    int sF,int sU,			\ |  | ||||||
| 				    const FermionField &in,		\ |  | ||||||
| 				    FermionField &out){ assert(0); }	\ |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #define HAND_SPECIALISE_GPARITY(IMPL)					\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSite(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, \ |  | ||||||
| 				    int ss,int sU,const FermionField &in, FermionField &out) \ |  | ||||||
|   {									\ |  | ||||||
|     typedef IMPL Impl;							\ |  | ||||||
|     typedef typename Simd::scalar_type S;				\ |  | ||||||
|     typedef typename Simd::vector_type V;				\ |  | ||||||
| 									\ |  | ||||||
|     HAND_DECLARATIONS(ignore);						\ |  | ||||||
| 									\ |  | ||||||
|     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ |  | ||||||
|     StencilEntry *SE;							\ |  | ||||||
|     HAND_DOP_SITE(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|     HAND_DOP_SITE(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|   }									\ |  | ||||||
| 									\ |  | ||||||
|   template<>								\ |  | ||||||
|   void WilsonKernels<IMPL>::HandDhopSiteDag(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ |  | ||||||
| 					    int ss,int sU,const FermionField &in, FermionField &out) \ |  | ||||||
|   {									\ |  | ||||||
|     typedef IMPL Impl;							\ |  | ||||||
|     typedef typename Simd::scalar_type S;				\ |  | ||||||
|     typedef typename Simd::vector_type V;				\ |  | ||||||
| 									\ |  | ||||||
|     HAND_DECLARATIONS(ignore);						\ |  | ||||||
| 									\ |  | ||||||
|     StencilEntry *SE;							\ |  | ||||||
|     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist;					\ |  | ||||||
|     HAND_DOP_SITE_DAG(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|     HAND_DOP_SITE_DAG(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|   }									\ |  | ||||||
| 									\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, \ |  | ||||||
| 						     int ss,int sU,const FermionField &in, FermionField &out) \ |  | ||||||
|   {									\ |  | ||||||
|     typedef IMPL Impl;							\ |  | ||||||
|     typedef typename Simd::scalar_type S;				\ |  | ||||||
|     typedef typename Simd::vector_type V;				\ |  | ||||||
| 									\ |  | ||||||
|     HAND_DECLARATIONS(ignore);						\ |  | ||||||
| 									\ |  | ||||||
|     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist;					\ |  | ||||||
|     StencilEntry *SE;							\ |  | ||||||
|     HAND_DOP_SITE_INT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|     HAND_DOP_SITE_INT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|   }									\ |  | ||||||
| 									\ |  | ||||||
|   template<>								\ |  | ||||||
|   void WilsonKernels<IMPL>::HandDhopSiteDagInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ |  | ||||||
| 							     int ss,int sU,const FermionField &in, FermionField &out) \ |  | ||||||
|   {									\ |  | ||||||
|     typedef IMPL Impl;							\ |  | ||||||
|     typedef typename Simd::scalar_type S;				\ |  | ||||||
|     typedef typename Simd::vector_type V;				\ |  | ||||||
| 									\ |  | ||||||
|     HAND_DECLARATIONS(ignore);						\ |  | ||||||
| 									\ |  | ||||||
|     StencilEntry *SE;							\ |  | ||||||
|     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ |  | ||||||
|     HAND_DOP_SITE_DAG_INT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|     HAND_DOP_SITE_DAG_INT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|   }									\ |  | ||||||
| 									\ |  | ||||||
|   template<> void							\ |  | ||||||
|   WilsonKernels<IMPL>::HandDhopSiteExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, \ |  | ||||||
| 						     int ss,int sU,const FermionField &in, FermionField &out) \ |  | ||||||
|   {									\ |  | ||||||
|     typedef IMPL Impl;							\ |  | ||||||
|     typedef typename Simd::scalar_type S;				\ |  | ||||||
|     typedef typename Simd::vector_type V;				\ |  | ||||||
| 									\ |  | ||||||
|     HAND_DECLARATIONS(ignore);						\ |  | ||||||
| 									\ |  | ||||||
|     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ |  | ||||||
|     StencilEntry *SE;							\ |  | ||||||
|     int nmu=0;								\ |  | ||||||
|     HAND_DOP_SITE_EXT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|     nmu = 0;								\ |  | ||||||
|     HAND_DOP_SITE_EXT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|   }									\ |  | ||||||
|   template<>								\ |  | ||||||
|   void WilsonKernels<IMPL>::HandDhopSiteDagExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ |  | ||||||
| 							     int ss,int sU,const FermionField &in, FermionField &out) \ |  | ||||||
|   {									\ |  | ||||||
|     typedef IMPL Impl;							\ |  | ||||||
|     typedef typename Simd::scalar_type S;				\ |  | ||||||
|     typedef typename Simd::vector_type V;				\ |  | ||||||
| 									\ |  | ||||||
|     HAND_DECLARATIONS(ignore);						\ |  | ||||||
| 									\ |  | ||||||
|     StencilEntry *SE;							\ |  | ||||||
|     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ |  | ||||||
|     int nmu=0;								\ |  | ||||||
|     HAND_DOP_SITE_DAG_EXT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|     nmu = 0;								\ |  | ||||||
|     HAND_DOP_SITE_DAG_EXT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| HAND_SPECIALISE_GPARITY(GparityWilsonImplF); |  | ||||||
| HAND_SPECIALISE_GPARITY(GparityWilsonImplD); |  | ||||||
| HAND_SPECIALISE_GPARITY(GparityWilsonImplFH); |  | ||||||
| HAND_SPECIALISE_GPARITY(GparityWilsonImplDF); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|    |  | ||||||
| ////////////// Wilson ; uses this implementation ///////////////////// | ////////////// Wilson ; uses this implementation ///////////////////// | ||||||
|  |  | ||||||
| #define INSTANTIATE_THEM(A) \ | #define INSTANTIATE_THEM(A) \ | ||||||
| @@ -930,8 +613,6 @@ INSTANTIATE_THEM(WilsonImplF); | |||||||
| INSTANTIATE_THEM(WilsonImplD); | INSTANTIATE_THEM(WilsonImplD); | ||||||
| INSTANTIATE_THEM(ZWilsonImplF); | INSTANTIATE_THEM(ZWilsonImplF); | ||||||
| INSTANTIATE_THEM(ZWilsonImplD); | INSTANTIATE_THEM(ZWilsonImplD); | ||||||
| INSTANTIATE_THEM(GparityWilsonImplF); |  | ||||||
| INSTANTIATE_THEM(GparityWilsonImplD); |  | ||||||
| INSTANTIATE_THEM(DomainWallVec5dImplF); | INSTANTIATE_THEM(DomainWallVec5dImplF); | ||||||
| INSTANTIATE_THEM(DomainWallVec5dImplD); | INSTANTIATE_THEM(DomainWallVec5dImplD); | ||||||
| INSTANTIATE_THEM(ZDomainWallVec5dImplF); | INSTANTIATE_THEM(ZDomainWallVec5dImplF); | ||||||
| @@ -940,12 +621,11 @@ INSTANTIATE_THEM(WilsonImplFH); | |||||||
| INSTANTIATE_THEM(WilsonImplDF); | INSTANTIATE_THEM(WilsonImplDF); | ||||||
| INSTANTIATE_THEM(ZWilsonImplFH); | INSTANTIATE_THEM(ZWilsonImplFH); | ||||||
| INSTANTIATE_THEM(ZWilsonImplDF); | INSTANTIATE_THEM(ZWilsonImplDF); | ||||||
| INSTANTIATE_THEM(GparityWilsonImplFH); |  | ||||||
| INSTANTIATE_THEM(GparityWilsonImplDF); |  | ||||||
| INSTANTIATE_THEM(DomainWallVec5dImplFH); | INSTANTIATE_THEM(DomainWallVec5dImplFH); | ||||||
| INSTANTIATE_THEM(DomainWallVec5dImplDF); | INSTANTIATE_THEM(DomainWallVec5dImplDF); | ||||||
| INSTANTIATE_THEM(ZDomainWallVec5dImplFH); | INSTANTIATE_THEM(ZDomainWallVec5dImplFH); | ||||||
| INSTANTIATE_THEM(ZDomainWallVec5dImplDF); | INSTANTIATE_THEM(ZDomainWallVec5dImplDF); | ||||||
| INSTANTIATE_THEM(WilsonTwoIndexAntiSymmetricImplF); | INSTANTIATE_THEM(WilsonTwoIndexAntiSymmetricImplF); | ||||||
| INSTANTIATE_THEM(WilsonTwoIndexAntiSymmetricImplD); | INSTANTIATE_THEM(WilsonTwoIndexAntiSymmetricImplD); | ||||||
|  |  | ||||||
| }} | }} | ||||||
|   | |||||||
							
								
								
									
										878
									
								
								lib/qcd/action/fermion/WilsonKernelsHandGparity.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										878
									
								
								lib/qcd/action/fermion/WilsonKernelsHandGparity.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,878 @@ | |||||||
|  |     /************************************************************************************* | ||||||
|  |  | ||||||
|  |     Grid physics library, www.github.com/paboyle/Grid  | ||||||
|  |  | ||||||
|  |     Source file: ./lib/qcd/action/fermion/WilsonKernelsHand.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/qcd/action/fermion/FermionCore.h> | ||||||
|  |  | ||||||
|  | #define REGISTER | ||||||
|  |  | ||||||
|  | #define LOAD_CHIMU_BODY(F)			\ | ||||||
|  |   Chimu_00=ref(F)(0)(0);			\ | ||||||
|  |   Chimu_01=ref(F)(0)(1);			\ | ||||||
|  |   Chimu_02=ref(F)(0)(2);			\ | ||||||
|  |   Chimu_10=ref(F)(1)(0);			\ | ||||||
|  |   Chimu_11=ref(F)(1)(1);			\ | ||||||
|  |   Chimu_12=ref(F)(1)(2);			\ | ||||||
|  |   Chimu_20=ref(F)(2)(0);			\ | ||||||
|  |   Chimu_21=ref(F)(2)(1);			\ | ||||||
|  |   Chimu_22=ref(F)(2)(2);			\ | ||||||
|  |   Chimu_30=ref(F)(3)(0);			\ | ||||||
|  |   Chimu_31=ref(F)(3)(1);			\ | ||||||
|  |   Chimu_32=ref(F)(3)(2) | ||||||
|  |  | ||||||
|  | #define LOAD_CHIMU(DIR,F,PERM)						\ | ||||||
|  |   { const SiteSpinor & ref (in._odata[offset]); LOAD_CHIMU_BODY(F); } | ||||||
|  |  | ||||||
|  | #define LOAD_CHI_BODY(F)				\ | ||||||
|  |     Chi_00 = ref(F)(0)(0);\ | ||||||
|  |     Chi_01 = ref(F)(0)(1);\ | ||||||
|  |     Chi_02 = ref(F)(0)(2);\ | ||||||
|  |     Chi_10 = ref(F)(1)(0);\ | ||||||
|  |     Chi_11 = ref(F)(1)(1);\ | ||||||
|  |     Chi_12 = ref(F)(1)(2) | ||||||
|  |  | ||||||
|  | #define LOAD_CHI(DIR,F,PERM)					\ | ||||||
|  |   {const SiteHalfSpinor &ref(buf[offset]); LOAD_CHI_BODY(F); } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | //G-parity implementations using in-place intrinsic ops | ||||||
|  |  | ||||||
|  | //1l 1h -> 1h 1l | ||||||
|  | //0l 0h , 1h 1l -> 0l 1h 0h,1l | ||||||
|  | //0h,1l -> 1l,0h | ||||||
|  | //if( (distance == 1 && !perm_will_occur) || (distance == -1 && perm_will_occur) ) | ||||||
|  | //Pulled fermion through forwards face, GPBC on upper component | ||||||
|  | //Need 0= 0l 1h   1= 1l 0h | ||||||
|  | //else if( (distance == -1 && !perm) || (distance == 1 && perm) ) | ||||||
|  | //Pulled fermion through backwards face, GPBC on lower component | ||||||
|  | //Need 0= 1l 0h   1= 0l 1h | ||||||
|  |  | ||||||
|  | //1l 1h -> 1h 1l | ||||||
|  | //0l 0h , 1h 1l -> 0l 1h 0h,1l | ||||||
|  | #define DO_TWIST_0L_1H(INTO,S,C,F, PERM, tmp1, tmp2, tmp3)			\ | ||||||
|  |   permute##PERM(tmp1, ref(1)(S)(C));				\ | ||||||
|  |   exchange##PERM(tmp2,tmp3, ref(0)(S)(C), tmp1);		\ | ||||||
|  |   INTO = tmp2; | ||||||
|  |  | ||||||
|  | //0l 0h -> 0h 0l | ||||||
|  | //1l 1h, 0h 0l -> 1l 0h, 1h 0l | ||||||
|  | #define DO_TWIST_1L_0H(INTO,S,C,F, PERM, tmp1, tmp2, tmp3)			\ | ||||||
|  |   permute##PERM(tmp1, ref(0)(S)(C));				\ | ||||||
|  |   exchange##PERM(tmp2,tmp3, ref(1)(S)(C), tmp1);		\ | ||||||
|  |   INTO = tmp2; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define LOAD_CHI_SETUP(DIR,F)						\ | ||||||
|  |   g = F;								\ | ||||||
|  |   direction = st._directions[DIR];				\ | ||||||
|  |   distance = st._distances[DIR];				\ | ||||||
|  |   sl = st._grid->_simd_layout[direction];			\ | ||||||
|  |   inplace_twist = 0;						\ | ||||||
|  |   if(SE->_around_the_world && this->Params.twists[DIR % 4]){		\ | ||||||
|  |     if(sl == 1){							\ | ||||||
|  |       g = (F+1) % 2;							\ | ||||||
|  |     }else{								\ | ||||||
|  |       inplace_twist = 1;						\ | ||||||
|  |     }									\ | ||||||
|  |   }   | ||||||
|  |  | ||||||
|  | #define LOAD_CHIMU_GPARITY_INPLACE_TWIST(DIR,F,PERM)			\ | ||||||
|  |   { const SiteSpinor &ref(in._odata[offset]);				\ | ||||||
|  |     LOAD_CHI_SETUP(DIR,F);						\ | ||||||
|  |     if(!inplace_twist){							\ | ||||||
|  |       LOAD_CHIMU_BODY(g);						\ | ||||||
|  |     }else{								\ | ||||||
|  |       if(  ( F==0 && ((distance == 1 && !perm) || (distance == -1 && perm)) ) || \ | ||||||
|  | 	   ( F==1 && ((distance == -1 && !perm) || (distance == 1 && perm)) ) ){ \ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_00,0,0,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_01,0,1,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_02,0,2,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_10,1,0,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_11,1,1,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_12,1,2,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_20,2,0,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_21,2,1,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_22,2,2,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_30,3,0,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_31,3,1,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chimu_32,3,2,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  |       }else{								\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_00,0,0,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_01,0,1,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_02,0,2,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_10,1,0,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_11,1,1,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_12,1,2,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_20,2,0,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_21,2,1,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_22,2,2,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_30,3,0,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_31,3,1,F,PERM,  U_00,U_01,U_10);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chimu_32,3,2,F,PERM,  U_11,U_20,U_21);		\ | ||||||
|  |       } \ | ||||||
|  |     } \ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define LOAD_CHI_GPARITY_INPLACE_TWIST(DIR,F,PERM)				\ | ||||||
|  |   { const SiteHalfSpinor &ref(buf[offset]);				\ | ||||||
|  |     LOAD_CHI_SETUP(DIR,F);						\ | ||||||
|  |     if(!inplace_twist){							\ | ||||||
|  |       LOAD_CHI_BODY(g);							\ | ||||||
|  |     }else{								\ | ||||||
|  |       if(  ( F==0 && ((distance == 1 && !perm) || (distance == -1 && perm)) ) || \ | ||||||
|  | 	   ( F==1 && ((distance == -1 && !perm) || (distance == 1 && perm)) ) ){ \ | ||||||
|  | 	DO_TWIST_0L_1H(Chi_00,0,0,F,PERM,  U_00,U_01,U_10);			\ | ||||||
|  | 	DO_TWIST_0L_1H(Chi_01,0,1,F,PERM,  U_11,U_20,U_21);			\ | ||||||
|  | 	DO_TWIST_0L_1H(Chi_02,0,2,F,PERM,  UChi_00,UChi_01,UChi_02);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chi_10,1,0,F,PERM,  UChi_10,UChi_11,UChi_12);		\ | ||||||
|  | 	DO_TWIST_0L_1H(Chi_11,1,1,F,PERM,  U_00,U_01,U_10);			\ | ||||||
|  | 	DO_TWIST_0L_1H(Chi_12,1,2,F,PERM,  U_11,U_20,U_21);			\ | ||||||
|  |       }else{								\ | ||||||
|  | 	DO_TWIST_1L_0H(Chi_00,0,0,F,PERM,  U_00,U_01,U_10);			\ | ||||||
|  | 	DO_TWIST_1L_0H(Chi_01,0,1,F,PERM,  U_11,U_20,U_21);			\ | ||||||
|  | 	DO_TWIST_1L_0H(Chi_02,0,2,F,PERM,  UChi_00,UChi_01,UChi_02);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chi_10,1,0,F,PERM,  UChi_10,UChi_11,UChi_12);		\ | ||||||
|  | 	DO_TWIST_1L_0H(Chi_11,1,1,F,PERM,  U_00,U_01,U_10);			\ | ||||||
|  | 	DO_TWIST_1L_0H(Chi_12,1,2,F,PERM,  U_11,U_20,U_21);			\ | ||||||
|  |       }									\ | ||||||
|  |     }									\ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define LOAD_CHI_GPARITY(DIR,F,PERM) LOAD_CHI_GPARITY_INPLACE_TWIST(DIR,F,PERM) | ||||||
|  | #define LOAD_CHIMU_GPARITY(DIR,F,PERM) LOAD_CHIMU_GPARITY_INPLACE_TWIST(DIR,F,PERM) | ||||||
|  |  | ||||||
|  | // To splat or not to splat depends on the implementation | ||||||
|  | #define MULT_2SPIN_BODY \ | ||||||
|  |   Impl::loadLinkElement(U_00,ref()(0,0));	\ | ||||||
|  |   Impl::loadLinkElement(U_10,ref()(1,0));	\ | ||||||
|  |   Impl::loadLinkElement(U_20,ref()(2,0));	\ | ||||||
|  |   Impl::loadLinkElement(U_01,ref()(0,1));	\ | ||||||
|  |   Impl::loadLinkElement(U_11,ref()(1,1));	\ | ||||||
|  |   Impl::loadLinkElement(U_21,ref()(2,1));	\ | ||||||
|  |   UChi_00 = U_00*Chi_00;			\ | ||||||
|  |   UChi_10 = U_00*Chi_10;			\ | ||||||
|  |   UChi_01 = U_10*Chi_00;			\ | ||||||
|  |   UChi_11 = U_10*Chi_10;			\ | ||||||
|  |   UChi_02 = U_20*Chi_00;			\ | ||||||
|  |   UChi_12 = U_20*Chi_10;			\ | ||||||
|  |   UChi_00+= U_01*Chi_01;			\ | ||||||
|  |   UChi_10+= U_01*Chi_11;			\ | ||||||
|  |   UChi_01+= U_11*Chi_01;			\ | ||||||
|  |   UChi_11+= U_11*Chi_11;			\ | ||||||
|  |   UChi_02+= U_21*Chi_01;			\ | ||||||
|  |   UChi_12+= U_21*Chi_11;			\ | ||||||
|  |   Impl::loadLinkElement(U_00,ref()(0,2));	\ | ||||||
|  |   Impl::loadLinkElement(U_10,ref()(1,2));	\ | ||||||
|  |   Impl::loadLinkElement(U_20,ref()(2,2));	\ | ||||||
|  |   UChi_00+= U_00*Chi_02;			\ | ||||||
|  |   UChi_10+= U_00*Chi_12;			\ | ||||||
|  |   UChi_01+= U_10*Chi_02;			\ | ||||||
|  |   UChi_11+= U_10*Chi_12;			\ | ||||||
|  |   UChi_02+= U_20*Chi_02;			\ | ||||||
|  |   UChi_12+= U_20*Chi_12 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define MULT_2SPIN(A,F)					\ | ||||||
|  |   {auto & ref(U._odata[sU](A)); MULT_2SPIN_BODY; } | ||||||
|  |  | ||||||
|  | #define MULT_2SPIN_GPARITY(A,F)				\ | ||||||
|  |   {auto & ref(U._odata[sU](F)(A)); MULT_2SPIN_BODY; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define PERMUTE_DIR(dir)			\ | ||||||
|  |       permute##dir(Chi_00,Chi_00);\ | ||||||
|  |       permute##dir(Chi_01,Chi_01);\ | ||||||
|  |       permute##dir(Chi_02,Chi_02);\ | ||||||
|  |       permute##dir(Chi_10,Chi_10);\ | ||||||
|  |       permute##dir(Chi_11,Chi_11);\ | ||||||
|  |       permute##dir(Chi_12,Chi_12); | ||||||
|  |  | ||||||
|  | //      hspin(0)=fspin(0)+timesI(fspin(3)); | ||||||
|  | //      hspin(1)=fspin(1)+timesI(fspin(2)); | ||||||
|  | #define XP_PROJ \ | ||||||
|  |     Chi_00 = Chimu_00+timesI(Chimu_30);\ | ||||||
|  |     Chi_01 = Chimu_01+timesI(Chimu_31);\ | ||||||
|  |     Chi_02 = Chimu_02+timesI(Chimu_32);\ | ||||||
|  |     Chi_10 = Chimu_10+timesI(Chimu_20);\ | ||||||
|  |     Chi_11 = Chimu_11+timesI(Chimu_21);\ | ||||||
|  |     Chi_12 = Chimu_12+timesI(Chimu_22); | ||||||
|  |  | ||||||
|  | #define YP_PROJ \ | ||||||
|  |     Chi_00 = Chimu_00-Chimu_30;\ | ||||||
|  |     Chi_01 = Chimu_01-Chimu_31;\ | ||||||
|  |     Chi_02 = Chimu_02-Chimu_32;\ | ||||||
|  |     Chi_10 = Chimu_10+Chimu_20;\ | ||||||
|  |     Chi_11 = Chimu_11+Chimu_21;\ | ||||||
|  |     Chi_12 = Chimu_12+Chimu_22; | ||||||
|  |  | ||||||
|  | #define ZP_PROJ \ | ||||||
|  |   Chi_00 = Chimu_00+timesI(Chimu_20);		\ | ||||||
|  |   Chi_01 = Chimu_01+timesI(Chimu_21);		\ | ||||||
|  |   Chi_02 = Chimu_02+timesI(Chimu_22);		\ | ||||||
|  |   Chi_10 = Chimu_10-timesI(Chimu_30);		\ | ||||||
|  |   Chi_11 = Chimu_11-timesI(Chimu_31);		\ | ||||||
|  |   Chi_12 = Chimu_12-timesI(Chimu_32); | ||||||
|  |  | ||||||
|  | #define TP_PROJ \ | ||||||
|  |   Chi_00 = Chimu_00+Chimu_20;		\ | ||||||
|  |   Chi_01 = Chimu_01+Chimu_21;		\ | ||||||
|  |   Chi_02 = Chimu_02+Chimu_22;		\ | ||||||
|  |   Chi_10 = Chimu_10+Chimu_30;		\ | ||||||
|  |   Chi_11 = Chimu_11+Chimu_31;		\ | ||||||
|  |   Chi_12 = Chimu_12+Chimu_32; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | //      hspin(0)=fspin(0)-timesI(fspin(3)); | ||||||
|  | //      hspin(1)=fspin(1)-timesI(fspin(2)); | ||||||
|  | #define XM_PROJ \ | ||||||
|  |     Chi_00 = Chimu_00-timesI(Chimu_30);\ | ||||||
|  |     Chi_01 = Chimu_01-timesI(Chimu_31);\ | ||||||
|  |     Chi_02 = Chimu_02-timesI(Chimu_32);\ | ||||||
|  |     Chi_10 = Chimu_10-timesI(Chimu_20);\ | ||||||
|  |     Chi_11 = Chimu_11-timesI(Chimu_21);\ | ||||||
|  |     Chi_12 = Chimu_12-timesI(Chimu_22); | ||||||
|  |  | ||||||
|  | #define YM_PROJ \ | ||||||
|  |     Chi_00 = Chimu_00+Chimu_30;\ | ||||||
|  |     Chi_01 = Chimu_01+Chimu_31;\ | ||||||
|  |     Chi_02 = Chimu_02+Chimu_32;\ | ||||||
|  |     Chi_10 = Chimu_10-Chimu_20;\ | ||||||
|  |     Chi_11 = Chimu_11-Chimu_21;\ | ||||||
|  |     Chi_12 = Chimu_12-Chimu_22; | ||||||
|  |  | ||||||
|  | #define ZM_PROJ \ | ||||||
|  |   Chi_00 = Chimu_00-timesI(Chimu_20);		\ | ||||||
|  |   Chi_01 = Chimu_01-timesI(Chimu_21);		\ | ||||||
|  |   Chi_02 = Chimu_02-timesI(Chimu_22);		\ | ||||||
|  |   Chi_10 = Chimu_10+timesI(Chimu_30);		\ | ||||||
|  |   Chi_11 = Chimu_11+timesI(Chimu_31);		\ | ||||||
|  |   Chi_12 = Chimu_12+timesI(Chimu_32); | ||||||
|  |  | ||||||
|  | #define TM_PROJ \ | ||||||
|  |   Chi_00 = Chimu_00-Chimu_20;		\ | ||||||
|  |   Chi_01 = Chimu_01-Chimu_21;		\ | ||||||
|  |   Chi_02 = Chimu_02-Chimu_22;		\ | ||||||
|  |   Chi_10 = Chimu_10-Chimu_30;		\ | ||||||
|  |   Chi_11 = Chimu_11-Chimu_31;		\ | ||||||
|  |   Chi_12 = Chimu_12-Chimu_32; | ||||||
|  |  | ||||||
|  | //      fspin(0)=hspin(0); | ||||||
|  | //      fspin(1)=hspin(1); | ||||||
|  | //      fspin(2)=timesMinusI(hspin(1)); | ||||||
|  | //      fspin(3)=timesMinusI(hspin(0)); | ||||||
|  | #define XP_RECON\ | ||||||
|  |   result_00 = UChi_00;\ | ||||||
|  |   result_01 = UChi_01;\ | ||||||
|  |   result_02 = UChi_02;\ | ||||||
|  |   result_10 = UChi_10;\ | ||||||
|  |   result_11 = UChi_11;\ | ||||||
|  |   result_12 = UChi_12;\ | ||||||
|  |   result_20 = timesMinusI(UChi_10);\ | ||||||
|  |   result_21 = timesMinusI(UChi_11);\ | ||||||
|  |   result_22 = timesMinusI(UChi_12);\ | ||||||
|  |   result_30 = timesMinusI(UChi_00);\ | ||||||
|  |   result_31 = timesMinusI(UChi_01);\ | ||||||
|  |   result_32 = timesMinusI(UChi_02); | ||||||
|  |  | ||||||
|  | #define XP_RECON_ACCUM\ | ||||||
|  |   result_00+=UChi_00;\ | ||||||
|  |   result_01+=UChi_01;\ | ||||||
|  |   result_02+=UChi_02;\ | ||||||
|  |   result_10+=UChi_10;\ | ||||||
|  |   result_11+=UChi_11;\ | ||||||
|  |   result_12+=UChi_12;\ | ||||||
|  |   result_20-=timesI(UChi_10);\ | ||||||
|  |   result_21-=timesI(UChi_11);\ | ||||||
|  |   result_22-=timesI(UChi_12);\ | ||||||
|  |   result_30-=timesI(UChi_00);\ | ||||||
|  |   result_31-=timesI(UChi_01);\ | ||||||
|  |   result_32-=timesI(UChi_02); | ||||||
|  |  | ||||||
|  | #define XM_RECON\ | ||||||
|  |   result_00 = UChi_00;\ | ||||||
|  |   result_01 = UChi_01;\ | ||||||
|  |   result_02 = UChi_02;\ | ||||||
|  |   result_10 = UChi_10;\ | ||||||
|  |   result_11 = UChi_11;\ | ||||||
|  |   result_12 = UChi_12;\ | ||||||
|  |   result_20 = timesI(UChi_10);\ | ||||||
|  |   result_21 = timesI(UChi_11);\ | ||||||
|  |   result_22 = timesI(UChi_12);\ | ||||||
|  |   result_30 = timesI(UChi_00);\ | ||||||
|  |   result_31 = timesI(UChi_01);\ | ||||||
|  |   result_32 = timesI(UChi_02); | ||||||
|  |  | ||||||
|  | #define XM_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20+= timesI(UChi_10);\ | ||||||
|  |   result_21+= timesI(UChi_11);\ | ||||||
|  |   result_22+= timesI(UChi_12);\ | ||||||
|  |   result_30+= timesI(UChi_00);\ | ||||||
|  |   result_31+= timesI(UChi_01);\ | ||||||
|  |   result_32+= timesI(UChi_02); | ||||||
|  |  | ||||||
|  | #define YP_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20+= UChi_10;\ | ||||||
|  |   result_21+= UChi_11;\ | ||||||
|  |   result_22+= UChi_12;\ | ||||||
|  |   result_30-= UChi_00;\ | ||||||
|  |   result_31-= UChi_01;\ | ||||||
|  |   result_32-= UChi_02; | ||||||
|  |  | ||||||
|  | #define YM_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20-= UChi_10;\ | ||||||
|  |   result_21-= UChi_11;\ | ||||||
|  |   result_22-= UChi_12;\ | ||||||
|  |   result_30+= UChi_00;\ | ||||||
|  |   result_31+= UChi_01;\ | ||||||
|  |   result_32+= UChi_02; | ||||||
|  |  | ||||||
|  | #define ZP_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20-= timesI(UChi_00);			\ | ||||||
|  |   result_21-= timesI(UChi_01);			\ | ||||||
|  |   result_22-= timesI(UChi_02);			\ | ||||||
|  |   result_30+= timesI(UChi_10);			\ | ||||||
|  |   result_31+= timesI(UChi_11);			\ | ||||||
|  |   result_32+= timesI(UChi_12); | ||||||
|  |  | ||||||
|  | #define ZM_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20+= timesI(UChi_00);			\ | ||||||
|  |   result_21+= timesI(UChi_01);			\ | ||||||
|  |   result_22+= timesI(UChi_02);			\ | ||||||
|  |   result_30-= timesI(UChi_10);			\ | ||||||
|  |   result_31-= timesI(UChi_11);			\ | ||||||
|  |   result_32-= timesI(UChi_12); | ||||||
|  |  | ||||||
|  | #define TP_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20+= UChi_00;			\ | ||||||
|  |   result_21+= UChi_01;			\ | ||||||
|  |   result_22+= UChi_02;			\ | ||||||
|  |   result_30+= UChi_10;			\ | ||||||
|  |   result_31+= UChi_11;			\ | ||||||
|  |   result_32+= UChi_12; | ||||||
|  |  | ||||||
|  | #define TM_RECON_ACCUM\ | ||||||
|  |   result_00+= UChi_00;\ | ||||||
|  |   result_01+= UChi_01;\ | ||||||
|  |   result_02+= UChi_02;\ | ||||||
|  |   result_10+= UChi_10;\ | ||||||
|  |   result_11+= UChi_11;\ | ||||||
|  |   result_12+= UChi_12;\ | ||||||
|  |   result_20-= UChi_00;	\ | ||||||
|  |   result_21-= UChi_01;	\ | ||||||
|  |   result_22-= UChi_02;	\ | ||||||
|  |   result_30-= UChi_10;	\ | ||||||
|  |   result_31-= UChi_11;	\ | ||||||
|  |   result_32-= UChi_12; | ||||||
|  |  | ||||||
|  | #define HAND_STENCIL_LEG(PROJ,PERM,DIR,RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | ||||||
|  |   SE=st.GetEntry(ptype,DIR,ss);			\ | ||||||
|  |   offset = SE->_offset;				\ | ||||||
|  |   local  = SE->_is_local;			\ | ||||||
|  |   perm   = SE->_permute;			\ | ||||||
|  |   if ( local ) {				\ | ||||||
|  |     LOAD_CHIMU_IMPL(DIR,F,PERM);			\ | ||||||
|  |     PROJ;					\ | ||||||
|  |     if ( perm) {				\ | ||||||
|  |       PERMUTE_DIR(PERM);			\ | ||||||
|  |     }						\ | ||||||
|  |   } else {					\ | ||||||
|  |     LOAD_CHI_IMPL(DIR,F,PERM);			\ | ||||||
|  |   }						\ | ||||||
|  |   MULT_2SPIN_IMPL(DIR,F);			\ | ||||||
|  |   RECON;					 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define HAND_STENCIL_LEG_INT(PROJ,PERM,DIR,RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL)	\ | ||||||
|  |   SE=st.GetEntry(ptype,DIR,ss);			\ | ||||||
|  |   offset = SE->_offset;				\ | ||||||
|  |   local  = SE->_is_local;			\ | ||||||
|  |   perm   = SE->_permute;			\ | ||||||
|  |   if ( local ) {				\ | ||||||
|  |     LOAD_CHIMU_IMPL(DIR,F,PERM);			\ | ||||||
|  |     PROJ;					\ | ||||||
|  |     if ( perm) {				\ | ||||||
|  |       PERMUTE_DIR(PERM);			\ | ||||||
|  |     }						\ | ||||||
|  |   } else if ( st.same_node[DIR] ) {		\ | ||||||
|  |     LOAD_CHI_IMPL(DIR,F,PERM);			\ | ||||||
|  |   }						\ | ||||||
|  |   if (local || st.same_node[DIR] ) {		\ | ||||||
|  |     MULT_2SPIN_IMPL(DIR,F);			\ | ||||||
|  |     RECON;					\ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | #define HAND_STENCIL_LEG_EXT(PROJ,PERM,DIR,RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL)	\ | ||||||
|  |   SE=st.GetEntry(ptype,DIR,ss);			\ | ||||||
|  |   offset = SE->_offset;				\ | ||||||
|  |   local  = SE->_is_local;			\ | ||||||
|  |   perm   = SE->_permute;			\ | ||||||
|  |   if((!SE->_is_local)&&(!st.same_node[DIR]) ) {	\ | ||||||
|  |     LOAD_CHI_IMPL(DIR,F,PERM);			\ | ||||||
|  |     MULT_2SPIN_IMPL(DIR,F);			\ | ||||||
|  |     RECON;					\ | ||||||
|  |     nmu++;					\ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | #define HAND_RESULT(ss,F)			\ | ||||||
|  |   {						\ | ||||||
|  |     SiteSpinor & ref (out._odata[ss]);		\ | ||||||
|  |     vstream(ref(F)(0)(0),result_00);		\ | ||||||
|  |     vstream(ref(F)(0)(1),result_01);		\ | ||||||
|  |     vstream(ref(F)(0)(2),result_02);		\ | ||||||
|  |     vstream(ref(F)(1)(0),result_10);		\ | ||||||
|  |     vstream(ref(F)(1)(1),result_11);		\ | ||||||
|  |     vstream(ref(F)(1)(2),result_12);		\ | ||||||
|  |     vstream(ref(F)(2)(0),result_20);		\ | ||||||
|  |     vstream(ref(F)(2)(1),result_21);		\ | ||||||
|  |     vstream(ref(F)(2)(2),result_22);		\ | ||||||
|  |     vstream(ref(F)(3)(0),result_30);		\ | ||||||
|  |     vstream(ref(F)(3)(1),result_31);		\ | ||||||
|  |     vstream(ref(F)(3)(2),result_32);		\ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | #define HAND_RESULT_EXT(ss,F)			\ | ||||||
|  |   if (nmu){					\ | ||||||
|  |     SiteSpinor & ref (out._odata[ss]);		\ | ||||||
|  |     ref(F)(0)(0)+=result_00;		\ | ||||||
|  |     ref(F)(0)(1)+=result_01;		\ | ||||||
|  |     ref(F)(0)(2)+=result_02;		\ | ||||||
|  |     ref(F)(1)(0)+=result_10;		\ | ||||||
|  |     ref(F)(1)(1)+=result_11;		\ | ||||||
|  |     ref(F)(1)(2)+=result_12;		\ | ||||||
|  |     ref(F)(2)(0)+=result_20;		\ | ||||||
|  |     ref(F)(2)(1)+=result_21;		\ | ||||||
|  |     ref(F)(2)(2)+=result_22;		\ | ||||||
|  |     ref(F)(3)(0)+=result_30;		\ | ||||||
|  |     ref(F)(3)(1)+=result_31;		\ | ||||||
|  |     ref(F)(3)(2)+=result_32;		\ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define HAND_DECLARATIONS(a)			\ | ||||||
|  |   Simd result_00;				\ | ||||||
|  |   Simd result_01;				\ | ||||||
|  |   Simd result_02;				\ | ||||||
|  |   Simd result_10;				\ | ||||||
|  |   Simd result_11;				\ | ||||||
|  |   Simd result_12;				\ | ||||||
|  |   Simd result_20;				\ | ||||||
|  |   Simd result_21;				\ | ||||||
|  |   Simd result_22;				\ | ||||||
|  |   Simd result_30;				\ | ||||||
|  |   Simd result_31;				\ | ||||||
|  |   Simd result_32;				\ | ||||||
|  |   Simd Chi_00;					\ | ||||||
|  |   Simd Chi_01;					\ | ||||||
|  |   Simd Chi_02;					\ | ||||||
|  |   Simd Chi_10;					\ | ||||||
|  |   Simd Chi_11;					\ | ||||||
|  |   Simd Chi_12;					\ | ||||||
|  |   Simd UChi_00;					\ | ||||||
|  |   Simd UChi_01;					\ | ||||||
|  |   Simd UChi_02;					\ | ||||||
|  |   Simd UChi_10;					\ | ||||||
|  |   Simd UChi_11;					\ | ||||||
|  |   Simd UChi_12;					\ | ||||||
|  |   Simd U_00;					\ | ||||||
|  |   Simd U_10;					\ | ||||||
|  |   Simd U_20;					\ | ||||||
|  |   Simd U_01;					\ | ||||||
|  |   Simd U_11;					\ | ||||||
|  |   Simd U_21; | ||||||
|  |  | ||||||
|  | #define ZERO_RESULT				\ | ||||||
|  |   result_00=zero;				\ | ||||||
|  |   result_01=zero;				\ | ||||||
|  |   result_02=zero;				\ | ||||||
|  |   result_10=zero;				\ | ||||||
|  |   result_11=zero;				\ | ||||||
|  |   result_12=zero;				\ | ||||||
|  |   result_20=zero;				\ | ||||||
|  |   result_21=zero;				\ | ||||||
|  |   result_22=zero;				\ | ||||||
|  |   result_30=zero;				\ | ||||||
|  |   result_31=zero;				\ | ||||||
|  |   result_32=zero;			 | ||||||
|  |  | ||||||
|  | #define Chimu_00 Chi_00 | ||||||
|  | #define Chimu_01 Chi_01 | ||||||
|  | #define Chimu_02 Chi_02 | ||||||
|  | #define Chimu_10 Chi_10 | ||||||
|  | #define Chimu_11 Chi_11 | ||||||
|  | #define Chimu_12 Chi_12 | ||||||
|  | #define Chimu_20 UChi_00 | ||||||
|  | #define Chimu_21 UChi_01 | ||||||
|  | #define Chimu_22 UChi_02 | ||||||
|  | #define Chimu_30 UChi_10 | ||||||
|  | #define Chimu_31 UChi_11 | ||||||
|  | #define Chimu_32 UChi_12 | ||||||
|  |  | ||||||
|  | namespace Grid { | ||||||
|  | namespace QCD { | ||||||
|  |  | ||||||
|  | template<class Impl> void  | ||||||
|  | WilsonKernels<Impl>::HandDhopSite(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, | ||||||
|  | 					  int ss,int sU,const FermionField &in, FermionField &out) | ||||||
|  | { | ||||||
|  | // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||||
|  |   typedef typename Simd::scalar_type S; | ||||||
|  |   typedef typename Simd::vector_type V; | ||||||
|  |  | ||||||
|  |   HAND_DECLARATIONS(ignore); | ||||||
|  |  | ||||||
|  |   int offset,local,perm, ptype; | ||||||
|  |   StencilEntry *SE; | ||||||
|  |  | ||||||
|  | #define HAND_DOP_SITE(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | ||||||
|  |   HAND_STENCIL_LEG(XM_PROJ,3,Xp,XM_RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(YM_PROJ,2,Yp,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);	\ | ||||||
|  |   HAND_STENCIL_LEG(ZM_PROJ,1,Zp,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(TM_PROJ,0,Tp,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(XP_PROJ,3,Xm,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(YP_PROJ,2,Ym,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(ZP_PROJ,1,Zm,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(TP_PROJ,0,Tm,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_RESULT(ss,F) | ||||||
|  |  | ||||||
|  |   HAND_DOP_SITE(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<class Impl> | ||||||
|  | void WilsonKernels<Impl>::HandDhopSiteDag(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, | ||||||
|  | 						  int ss,int sU,const FermionField &in, FermionField &out) | ||||||
|  | { | ||||||
|  |   typedef typename Simd::scalar_type S; | ||||||
|  |   typedef typename Simd::vector_type V; | ||||||
|  |  | ||||||
|  |   HAND_DECLARATIONS(ignore); | ||||||
|  |  | ||||||
|  |   StencilEntry *SE; | ||||||
|  |   int offset,local,perm, ptype; | ||||||
|  |  | ||||||
|  | #define HAND_DOP_SITE_DAG(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | ||||||
|  |   HAND_STENCIL_LEG(XP_PROJ,3,Xp,XP_RECON,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(YP_PROJ,2,Yp,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(ZP_PROJ,1,Zp,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(TP_PROJ,0,Tp,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(XM_PROJ,3,Xm,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(YM_PROJ,2,Ym,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(ZM_PROJ,1,Zm,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG(TM_PROJ,0,Tm,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_RESULT(ss,F) | ||||||
|  |  | ||||||
|  |   HAND_DOP_SITE_DAG(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<class Impl> void  | ||||||
|  | WilsonKernels<Impl>::HandDhopSiteInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, | ||||||
|  | 					  int ss,int sU,const FermionField &in, FermionField &out) | ||||||
|  | { | ||||||
|  | // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||||
|  |   typedef typename Simd::scalar_type S; | ||||||
|  |   typedef typename Simd::vector_type V; | ||||||
|  |  | ||||||
|  |   HAND_DECLARATIONS(ignore); | ||||||
|  |  | ||||||
|  |   int offset,local,perm, ptype; | ||||||
|  |   StencilEntry *SE; | ||||||
|  |  | ||||||
|  | #define HAND_DOP_SITE_INT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | ||||||
|  |   ZERO_RESULT; \ | ||||||
|  |   HAND_STENCIL_LEG_INT(XM_PROJ,3,Xp,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(YM_PROJ,2,Yp,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(ZM_PROJ,1,Zp,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(TM_PROJ,0,Tp,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(XP_PROJ,3,Xm,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(YP_PROJ,2,Ym,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(ZP_PROJ,1,Zm,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_INT(TP_PROJ,0,Tm,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_RESULT(ss,F) | ||||||
|  |  | ||||||
|  |   HAND_DOP_SITE_INT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<class Impl> | ||||||
|  | void WilsonKernels<Impl>::HandDhopSiteDagInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, | ||||||
|  | 						  int ss,int sU,const FermionField &in, FermionField &out) | ||||||
|  | { | ||||||
|  |   typedef typename Simd::scalar_type S; | ||||||
|  |   typedef typename Simd::vector_type V; | ||||||
|  |  | ||||||
|  |   HAND_DECLARATIONS(ignore); | ||||||
|  |  | ||||||
|  |   StencilEntry *SE; | ||||||
|  |   int offset,local,perm, ptype; | ||||||
|  |  | ||||||
|  | #define HAND_DOP_SITE_DAG_INT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL)				\ | ||||||
|  |   ZERO_RESULT;							\ | ||||||
|  |   HAND_STENCIL_LEG_INT(XP_PROJ,3,Xp,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(YP_PROJ,2,Yp,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(ZP_PROJ,1,Zp,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(TP_PROJ,0,Tp,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(XM_PROJ,3,Xm,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(YM_PROJ,2,Ym,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(ZM_PROJ,1,Zm,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_STENCIL_LEG_INT(TM_PROJ,0,Tm,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL);		\ | ||||||
|  |   HAND_RESULT(ss,F) | ||||||
|  |    | ||||||
|  |   HAND_DOP_SITE_DAG_INT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<class Impl> void  | ||||||
|  | WilsonKernels<Impl>::HandDhopSiteExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, | ||||||
|  | 					  int ss,int sU,const FermionField &in, FermionField &out) | ||||||
|  | { | ||||||
|  | // T==0, Z==1, Y==2, Z==3 expect 1,2,2,2 simd layout etc... | ||||||
|  |   typedef typename Simd::scalar_type S; | ||||||
|  |   typedef typename Simd::vector_type V; | ||||||
|  |  | ||||||
|  |   HAND_DECLARATIONS(ignore); | ||||||
|  |  | ||||||
|  |   int offset,local,perm, ptype; | ||||||
|  |   StencilEntry *SE; | ||||||
|  |   int nmu=0; | ||||||
|  |  | ||||||
|  | #define HAND_DOP_SITE_EXT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | ||||||
|  |   ZERO_RESULT; \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(XM_PROJ,3,Xp,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(YM_PROJ,2,Yp,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(ZM_PROJ,1,Zp,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(TM_PROJ,0,Tp,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(XP_PROJ,3,Xm,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(YP_PROJ,2,Ym,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(ZP_PROJ,1,Zm,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(TP_PROJ,0,Tm,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_RESULT_EXT(ss,F) | ||||||
|  |  | ||||||
|  |   HAND_DOP_SITE_EXT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template<class Impl> | ||||||
|  | void WilsonKernels<Impl>::HandDhopSiteDagExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, | ||||||
|  | 						  int ss,int sU,const FermionField &in, FermionField &out) | ||||||
|  | { | ||||||
|  |   typedef typename Simd::scalar_type S; | ||||||
|  |   typedef typename Simd::vector_type V; | ||||||
|  |  | ||||||
|  |   HAND_DECLARATIONS(ignore); | ||||||
|  |  | ||||||
|  |   StencilEntry *SE; | ||||||
|  |   int offset,local,perm, ptype; | ||||||
|  |   int nmu=0; | ||||||
|  |  | ||||||
|  | #define HAND_DOP_SITE_DAG_EXT(F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL) \ | ||||||
|  |   ZERO_RESULT; \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(XP_PROJ,3,Xp,XP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(YP_PROJ,2,Yp,YP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(ZP_PROJ,1,Zp,ZP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(TP_PROJ,0,Tp,TP_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(XM_PROJ,3,Xm,XM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(YM_PROJ,2,Ym,YM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(ZM_PROJ,1,Zm,ZM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_STENCIL_LEG_EXT(TM_PROJ,0,Tm,TM_RECON_ACCUM,F,LOAD_CHI_IMPL,LOAD_CHIMU_IMPL,MULT_2SPIN_IMPL); \ | ||||||
|  |   HAND_RESULT_EXT(ss,F) | ||||||
|  |  | ||||||
|  |   HAND_DOP_SITE_DAG_EXT(, LOAD_CHI,LOAD_CHIMU,MULT_2SPIN); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #define HAND_SPECIALISE_GPARITY(IMPL)					\ | ||||||
|  |   template<> void							\ | ||||||
|  |   WilsonKernels<IMPL>::HandDhopSite(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, \ | ||||||
|  | 				    int ss,int sU,const FermionField &in, FermionField &out) \ | ||||||
|  |   {									\ | ||||||
|  |     typedef IMPL Impl;							\ | ||||||
|  |     typedef typename Simd::scalar_type S;				\ | ||||||
|  |     typedef typename Simd::vector_type V;				\ | ||||||
|  | 									\ | ||||||
|  |     HAND_DECLARATIONS(ignore);						\ | ||||||
|  | 									\ | ||||||
|  |     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ | ||||||
|  |     StencilEntry *SE;							\ | ||||||
|  |     HAND_DOP_SITE(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |     HAND_DOP_SITE(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |   }									\ | ||||||
|  | 									\ | ||||||
|  |   template<>								\ | ||||||
|  |   void WilsonKernels<IMPL>::HandDhopSiteDag(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ | ||||||
|  | 					    int ss,int sU,const FermionField &in, FermionField &out) \ | ||||||
|  |   {									\ | ||||||
|  |     typedef IMPL Impl;							\ | ||||||
|  |     typedef typename Simd::scalar_type S;				\ | ||||||
|  |     typedef typename Simd::vector_type V;				\ | ||||||
|  | 									\ | ||||||
|  |     HAND_DECLARATIONS(ignore);						\ | ||||||
|  | 									\ | ||||||
|  |     StencilEntry *SE;							\ | ||||||
|  |     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist;					\ | ||||||
|  |     HAND_DOP_SITE_DAG(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |     HAND_DOP_SITE_DAG(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |   }									\ | ||||||
|  | 									\ | ||||||
|  |   template<> void							\ | ||||||
|  |   WilsonKernels<IMPL>::HandDhopSiteInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, \ | ||||||
|  | 						     int ss,int sU,const FermionField &in, FermionField &out) \ | ||||||
|  |   {									\ | ||||||
|  |     typedef IMPL Impl;							\ | ||||||
|  |     typedef typename Simd::scalar_type S;				\ | ||||||
|  |     typedef typename Simd::vector_type V;				\ | ||||||
|  | 									\ | ||||||
|  |     HAND_DECLARATIONS(ignore);						\ | ||||||
|  | 									\ | ||||||
|  |     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist;					\ | ||||||
|  |     StencilEntry *SE;							\ | ||||||
|  |     HAND_DOP_SITE_INT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |     HAND_DOP_SITE_INT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |   }									\ | ||||||
|  | 									\ | ||||||
|  |   template<>								\ | ||||||
|  |   void WilsonKernels<IMPL>::HandDhopSiteDagInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ | ||||||
|  | 							     int ss,int sU,const FermionField &in, FermionField &out) \ | ||||||
|  |   {									\ | ||||||
|  |     typedef IMPL Impl;							\ | ||||||
|  |     typedef typename Simd::scalar_type S;				\ | ||||||
|  |     typedef typename Simd::vector_type V;				\ | ||||||
|  | 									\ | ||||||
|  |     HAND_DECLARATIONS(ignore);						\ | ||||||
|  | 									\ | ||||||
|  |     StencilEntry *SE;							\ | ||||||
|  |     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ | ||||||
|  |     HAND_DOP_SITE_DAG_INT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |     HAND_DOP_SITE_DAG_INT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |   }									\ | ||||||
|  | 									\ | ||||||
|  |   template<> void							\ | ||||||
|  |   WilsonKernels<IMPL>::HandDhopSiteExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor  *buf, \ | ||||||
|  | 						     int ss,int sU,const FermionField &in, FermionField &out) \ | ||||||
|  |   {									\ | ||||||
|  |     typedef IMPL Impl;							\ | ||||||
|  |     typedef typename Simd::scalar_type S;				\ | ||||||
|  |     typedef typename Simd::vector_type V;				\ | ||||||
|  | 									\ | ||||||
|  |     HAND_DECLARATIONS(ignore);						\ | ||||||
|  | 									\ | ||||||
|  |     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ | ||||||
|  |     StencilEntry *SE;							\ | ||||||
|  |     int nmu=0;								\ | ||||||
|  |     HAND_DOP_SITE_EXT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |     nmu = 0;								\ | ||||||
|  |     HAND_DOP_SITE_EXT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |   }									\ | ||||||
|  |   template<>								\ | ||||||
|  |   void WilsonKernels<IMPL>::HandDhopSiteDagExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ | ||||||
|  | 							     int ss,int sU,const FermionField &in, FermionField &out) \ | ||||||
|  |   {									\ | ||||||
|  |     typedef IMPL Impl;							\ | ||||||
|  |     typedef typename Simd::scalar_type S;				\ | ||||||
|  |     typedef typename Simd::vector_type V;				\ | ||||||
|  | 									\ | ||||||
|  |     HAND_DECLARATIONS(ignore);						\ | ||||||
|  | 									\ | ||||||
|  |     StencilEntry *SE;							\ | ||||||
|  |     int offset,local,perm, ptype, g, direction, distance, sl, inplace_twist; \ | ||||||
|  |     int nmu=0;								\ | ||||||
|  |     HAND_DOP_SITE_DAG_EXT(0, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |     nmu = 0;								\ | ||||||
|  |     HAND_DOP_SITE_DAG_EXT(1, LOAD_CHI_GPARITY,LOAD_CHIMU_GPARITY,MULT_2SPIN_GPARITY); \ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | HAND_SPECIALISE_GPARITY(GparityWilsonImplF); | ||||||
|  | HAND_SPECIALISE_GPARITY(GparityWilsonImplD); | ||||||
|  | HAND_SPECIALISE_GPARITY(GparityWilsonImplFH); | ||||||
|  | HAND_SPECIALISE_GPARITY(GparityWilsonImplDF); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    | ||||||
|  | ////////////// Wilson ; uses this implementation ///////////////////// | ||||||
|  |  | ||||||
|  | #define INSTANTIATE_THEM(A) \ | ||||||
|  | template void WilsonKernels<A>::HandDhopSite(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf,\ | ||||||
|  | 					     int ss,int sU,const FermionField &in, FermionField &out); \ | ||||||
|  | template void WilsonKernels<A>::HandDhopSiteDag(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ | ||||||
|  | 						int ss,int sU,const FermionField &in, FermionField &out);\ | ||||||
|  | template void WilsonKernels<A>::HandDhopSiteInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf,\ | ||||||
|  | 						int ss,int sU,const FermionField &in, FermionField &out); \ | ||||||
|  | template void WilsonKernels<A>::HandDhopSiteDagInt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ | ||||||
|  | 						   int ss,int sU,const FermionField &in, FermionField &out); \ | ||||||
|  | template void WilsonKernels<A>::HandDhopSiteExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf,\ | ||||||
|  | 						int ss,int sU,const FermionField &in, FermionField &out); \ | ||||||
|  | template void WilsonKernels<A>::HandDhopSiteDagExt(StencilImpl &st,LebesgueOrder &lo,DoubledGaugeField &U,SiteHalfSpinor *buf, \ | ||||||
|  | 						   int ss,int sU,const FermionField &in, FermionField &out);  | ||||||
|  |  | ||||||
|  | INSTANTIATE_THEM(GparityWilsonImplF); | ||||||
|  | INSTANTIATE_THEM(GparityWilsonImplD); | ||||||
|  | INSTANTIATE_THEM(GparityWilsonImplFH); | ||||||
|  | INSTANTIATE_THEM(GparityWilsonImplDF); | ||||||
|  | }} | ||||||
| @@ -48,6 +48,22 @@ with this program; if not, write to the Free Software Foundation, Inc., | |||||||
|     }                                                                    \ |     }                                                                    \ | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | #define RegisterLoadCheckPointerMetadataFunction(NAME)                   \ | ||||||
|  |   template < class Metadata >                                            \ | ||||||
|  |   void Load##NAME##Checkpointer(const CheckpointerParameters& Params_, const Metadata& M_) { \ | ||||||
|  |     if (!have_CheckPointer) {                                            \ | ||||||
|  |       std::cout << GridLogDebug << "Loading Metadata Checkpointer " << #NAME      \ | ||||||
|  |                 << std::endl;                                            \ | ||||||
|  |       CP = std::unique_ptr<CheckpointerBaseModule>(                      \ | ||||||
|  |         new NAME##CPModule<ImplementationPolicy, Metadata >(Params_, M_));   \ | ||||||
|  |       have_CheckPointer = true;                                          \ | ||||||
|  |     } else {                                                             \ | ||||||
|  |       std::cout << GridLogError << "Checkpointer already loaded "        \ | ||||||
|  |                 << std::endl;                                            \ | ||||||
|  |       exit(1);                                                           \ | ||||||
|  |     }                                                                    \ | ||||||
|  |   } | ||||||
|  |  | ||||||
| namespace Grid { | namespace Grid { | ||||||
| namespace QCD { | namespace QCD { | ||||||
|  |  | ||||||
| @@ -77,7 +93,7 @@ class HMCResourceManager { | |||||||
|   bool have_CheckPointer; |   bool have_CheckPointer; | ||||||
|  |  | ||||||
|   // NOTE: operator << is not overloaded for std::vector<string>  |   // NOTE: operator << is not overloaded for std::vector<string>  | ||||||
|   // so thsi function is necessary |   // so this function is necessary | ||||||
|   void output_vector_string(const std::vector<std::string> &vs){ |   void output_vector_string(const std::vector<std::string> &vs){ | ||||||
|     for (auto &i: vs) |     for (auto &i: vs) | ||||||
|       std::cout << i << " "; |       std::cout << i << " "; | ||||||
| @@ -254,6 +270,7 @@ class HMCResourceManager { | |||||||
|   RegisterLoadCheckPointerFunction(Nersc); |   RegisterLoadCheckPointerFunction(Nersc); | ||||||
|   #ifdef HAVE_LIME |   #ifdef HAVE_LIME | ||||||
|   RegisterLoadCheckPointerFunction(ILDG); |   RegisterLoadCheckPointerFunction(ILDG); | ||||||
|  |   RegisterLoadCheckPointerMetadataFunction(Scidac); | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   //////////////////////////////////////////////////////// |   //////////////////////////////////////////////////////// | ||||||
|   | |||||||
| @@ -76,6 +76,14 @@ class BaseHmcCheckpointer : public HmcObservable<typename Impl::Field> { | |||||||
|     } |     } | ||||||
|  	}  |  	}  | ||||||
|  |  | ||||||
|  |   void check_filename(const std::string &filename){ | ||||||
|  |     std::ifstream f(filename.c_str()); | ||||||
|  |     if(!f.good()){ | ||||||
|  |       std::cout << GridLogError << "Filename " << filename << " not found. Aborting. " << std::endl; | ||||||
|  |       abort(); | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   virtual void initialize(const CheckpointerParameters &Params) = 0; |   virtual void initialize(const CheckpointerParameters &Params) = 0; | ||||||
|  |  | ||||||
|   virtual void CheckpointRestore(int traj, typename Impl::Field &U, |   virtual void CheckpointRestore(int traj, typename Impl::Field &U, | ||||||
|   | |||||||
| @@ -93,6 +93,9 @@ class BinaryHmcCheckpointer : public BaseHmcCheckpointer<Impl> { | |||||||
|   void CheckpointRestore(int traj, Field &U, GridSerialRNG &sRNG, GridParallelRNG &pRNG) { |   void CheckpointRestore(int traj, Field &U, GridSerialRNG &sRNG, GridParallelRNG &pRNG) { | ||||||
|     std::string config, rng; |     std::string config, rng; | ||||||
|     this->build_filenames(traj, Params, config, rng); |     this->build_filenames(traj, Params, config, rng); | ||||||
|  |     this->check_filename(rng); | ||||||
|  |     this->check_filename(config); | ||||||
|  |  | ||||||
|  |  | ||||||
|     BinarySimpleMunger<sobj_double, sobj> munge; |     BinarySimpleMunger<sobj_double, sobj> munge; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -136,6 +136,20 @@ class ILDGCPModule: public CheckPointerModule< ImplementationPolicy> { | |||||||
|  |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | template<class ImplementationPolicy, class Metadata> | ||||||
|  | class ScidacCPModule: public CheckPointerModule< ImplementationPolicy> { | ||||||
|  |   typedef CheckPointerModule< ImplementationPolicy> CPBase; | ||||||
|  |   Metadata M; | ||||||
|  |  | ||||||
|  |   // acquire resource | ||||||
|  |   virtual void initialize(){ | ||||||
|  |      this->CheckPointPtr.reset(new ScidacHmcCheckpointer<ImplementationPolicy, Metadata>(this->Par_, M)); | ||||||
|  |   } | ||||||
|  | public: | ||||||
|  |   ScidacCPModule(typename CPBase::APar Par, Metadata M_):M(M_), CPBase(Par) {} | ||||||
|  |   template <class ReaderClass> | ||||||
|  |   ScidacCPModule(Reader<ReaderClass>& Reader) : Parametrized<typename CPBase::APar>(Reader), M(Reader){}; | ||||||
|  | }; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -34,6 +34,7 @@ directory | |||||||
| #include <Grid/qcd/hmc/checkpointers/NerscCheckpointer.h> | #include <Grid/qcd/hmc/checkpointers/NerscCheckpointer.h> | ||||||
| #include <Grid/qcd/hmc/checkpointers/BinaryCheckpointer.h> | #include <Grid/qcd/hmc/checkpointers/BinaryCheckpointer.h> | ||||||
| #include <Grid/qcd/hmc/checkpointers/ILDGCheckpointer.h> | #include <Grid/qcd/hmc/checkpointers/ILDGCheckpointer.h> | ||||||
|  | #include <Grid/qcd/hmc/checkpointers/ScidacCheckpointer.h> | ||||||
| //#include <Grid/qcd/hmc/checkpointers/CheckPointerModules.h> | //#include <Grid/qcd/hmc/checkpointers/CheckPointerModules.h> | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -95,6 +95,10 @@ class ILDGHmcCheckpointer : public BaseHmcCheckpointer<Implementation> { | |||||||
|                          GridParallelRNG &pRNG) { |                          GridParallelRNG &pRNG) { | ||||||
|     std::string config, rng; |     std::string config, rng; | ||||||
|     this->build_filenames(traj, Params, config, rng); |     this->build_filenames(traj, Params, config, rng); | ||||||
|  |     this->check_filename(rng); | ||||||
|  |     this->check_filename(config); | ||||||
|  |  | ||||||
|  |      | ||||||
|  |  | ||||||
|     uint32_t nersc_csum,scidac_csuma,scidac_csumb; |     uint32_t nersc_csum,scidac_csuma,scidac_csumb; | ||||||
|     BinaryIO::readRNG(sRNG, pRNG, rng, 0,nersc_csum,scidac_csuma,scidac_csumb); |     BinaryIO::readRNG(sRNG, pRNG, rng, 0,nersc_csum,scidac_csuma,scidac_csumb); | ||||||
|   | |||||||
| @@ -69,6 +69,9 @@ class NerscHmcCheckpointer : public BaseHmcCheckpointer<Gimpl> { | |||||||
|                          GridParallelRNG &pRNG) { |                          GridParallelRNG &pRNG) { | ||||||
|     std::string config, rng; |     std::string config, rng; | ||||||
|     this->build_filenames(traj, Params, config, rng); |     this->build_filenames(traj, Params, config, rng); | ||||||
|  |     this->check_filename(rng); | ||||||
|  |     this->check_filename(config); | ||||||
|  |  | ||||||
|  |  | ||||||
|     FieldMetaData header; |     FieldMetaData header; | ||||||
|     NerscIO::readRNGState(sRNG, pRNG, header, rng); |     NerscIO::readRNGState(sRNG, pRNG, header, rng); | ||||||
|   | |||||||
							
								
								
									
										125
									
								
								lib/qcd/hmc/checkpointers/ScidacCheckpointer.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								lib/qcd/hmc/checkpointers/ScidacCheckpointer.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | |||||||
|  | /************************************************************************************* | ||||||
|  |  | ||||||
|  | Grid physics library, www.github.com/paboyle/Grid | ||||||
|  |  | ||||||
|  | Source file: ./lib/qcd/hmc/ScidacCheckpointer.h | ||||||
|  |  | ||||||
|  | Copyright (C) 2018 | ||||||
|  |  | ||||||
|  | Author: Guido Cossu <guido.cossu@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 */ | ||||||
|  | #ifndef SCIDAC_CHECKPOINTER | ||||||
|  | #define SCIDAC_CHECKPOINTER | ||||||
|  |  | ||||||
|  | #ifdef HAVE_LIME | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  | #include <sstream> | ||||||
|  | #include <string> | ||||||
|  |  | ||||||
|  | namespace Grid { | ||||||
|  | namespace QCD { | ||||||
|  |  | ||||||
|  | // For generic fields | ||||||
|  | template <class Implementation, class Metadata> | ||||||
|  | class ScidacHmcCheckpointer : public BaseHmcCheckpointer<Implementation> { | ||||||
|  |  private: | ||||||
|  |   CheckpointerParameters Params; | ||||||
|  |   Metadata MData; | ||||||
|  |  | ||||||
|  |   typedef typename Implementation::Field Field; | ||||||
|  |  | ||||||
|  |  public: | ||||||
|  |   //INHERIT_GIMPL_TYPES(Implementation); | ||||||
|  |  | ||||||
|  |   ScidacHmcCheckpointer(const CheckpointerParameters &Params_) { initialize(Params_); } | ||||||
|  |   ScidacHmcCheckpointer(const CheckpointerParameters &Params_, const Metadata& M_):MData(M_) { initialize(Params_); } | ||||||
|  |  | ||||||
|  |   void initialize(const CheckpointerParameters &Params_) { | ||||||
|  |     Params = Params_; | ||||||
|  |  | ||||||
|  |     // check here that the format is valid | ||||||
|  |     int ieee32big = (Params.format == std::string("IEEE32BIG")); | ||||||
|  |     int ieee32    = (Params.format == std::string("IEEE32")); | ||||||
|  |     int ieee64big = (Params.format == std::string("IEEE64BIG")); | ||||||
|  |     int ieee64    = (Params.format == std::string("IEEE64")); | ||||||
|  |  | ||||||
|  |     if (!(ieee64big || ieee32 || ieee32big || ieee64)) { | ||||||
|  |       std::cout << GridLogError << "Unrecognized file format " << Params.format | ||||||
|  |                 << std::endl; | ||||||
|  |       std::cout << GridLogError | ||||||
|  |                 << "Allowed: IEEE32BIG | IEEE32 | IEEE64BIG | IEEE64" | ||||||
|  |                 << std::endl; | ||||||
|  |  | ||||||
|  |       exit(1); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void TrajectoryComplete(int traj, Field &U, GridSerialRNG &sRNG, | ||||||
|  |                           GridParallelRNG &pRNG) { | ||||||
|  |     if ((traj % Params.saveInterval) == 0) { | ||||||
|  |       std::string config, rng; | ||||||
|  |       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(config); | ||||||
|  |       _ScidacWriter.writeScidacFieldRecord(U, MData); | ||||||
|  |       _ScidacWriter.close(); | ||||||
|  |  | ||||||
|  |       std::cout << GridLogMessage << "Written Scidac Configuration on " << config | ||||||
|  |                 << " checksum " << std::hex << nersc_csum<<"/" | ||||||
|  | 		            << scidac_csuma<<"/" << scidac_csumb | ||||||
|  | 		            << std::dec << std::endl; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   void CheckpointRestore(int traj, Field &U, GridSerialRNG &sRNG, | ||||||
|  |                          GridParallelRNG &pRNG) { | ||||||
|  |     std::string config, rng; | ||||||
|  |     this->build_filenames(traj, Params, config, rng); | ||||||
|  |     this->check_filename(rng); | ||||||
|  |     this->check_filename(config); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     uint32_t nersc_csum,scidac_csuma,scidac_csumb; | ||||||
|  |     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(); | ||||||
|  |  | ||||||
|  |     std::cout << GridLogMessage << "Read Scidac Configuration from " << config | ||||||
|  |               << " checksum " << std::hex  | ||||||
|  | 	      << nersc_csum<<"/" | ||||||
|  | 	      << scidac_csuma<<"/" | ||||||
|  | 	      << scidac_csumb | ||||||
|  | 	      << std::dec << std::endl; | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif  // HAVE_LIME | ||||||
|  | #endif  // ILDG_CHECKPOINTER | ||||||
| @@ -66,6 +66,8 @@ void Gather_plane_simple_table (std::vector<std::pair<int,int> >& table,const La | |||||||
|   parallel_for(int i=0;i<num;i++){ |   parallel_for(int i=0;i<num;i++){ | ||||||
|     compress.Compress(&buffer[off],table[i].first,rhs._odata[so+table[i].second]); |     compress.Compress(&buffer[off],table[i].first,rhs._odata[so+table[i].second]); | ||||||
|   } |   } | ||||||
|  | // Further optimisatoin: i) streaming store the result | ||||||
|  | //                       ii) software prefetch the first element of the next table entry | ||||||
| } | } | ||||||
|  |  | ||||||
| /////////////////////////////////////////////////////////////////// | /////////////////////////////////////////////////////////////////// | ||||||
| @@ -505,25 +507,24 @@ class CartesianStencil { // Stencil runs along coordinate axes only; NO diagonal | |||||||
|   template<class decompressor> |   template<class decompressor> | ||||||
|   void CommsMerge(decompressor decompress,std::vector<Merge> &mm,std::vector<Decompress> &dd) {  |   void CommsMerge(decompressor decompress,std::vector<Merge> &mm,std::vector<Decompress> &dd) {  | ||||||
|  |  | ||||||
|     for(int i=0;i<mm.size();i++){	 |  | ||||||
|     mergetime-=usecond(); |     mergetime-=usecond(); | ||||||
|  |     for(int i=0;i<mm.size();i++){	 | ||||||
|       parallel_for(int o=0;o<mm[i].buffer_size/2;o++){ |       parallel_for(int o=0;o<mm[i].buffer_size/2;o++){ | ||||||
| 	decompress.Exchange(mm[i].mpointer, | 	decompress.Exchange(mm[i].mpointer, | ||||||
| 			    mm[i].vpointers[0], | 			    mm[i].vpointers[0], | ||||||
| 			    mm[i].vpointers[1], | 			    mm[i].vpointers[1], | ||||||
| 			    mm[i].type,o); | 			    mm[i].type,o); | ||||||
|       } |       } | ||||||
|       mergetime+=usecond(); |  | ||||||
|     } |     } | ||||||
|  |     mergetime+=usecond(); | ||||||
|  |  | ||||||
|     for(int i=0;i<dd.size();i++){	 |  | ||||||
|     decompresstime-=usecond(); |     decompresstime-=usecond(); | ||||||
|  |     for(int i=0;i<dd.size();i++){	 | ||||||
|       parallel_for(int o=0;o<dd[i].buffer_size;o++){ |       parallel_for(int o=0;o<dd[i].buffer_size;o++){ | ||||||
| 	decompress.Decompress(dd[i].kernel_p,dd[i].mpi_p,o); | 	decompress.Decompress(dd[i].kernel_p,dd[i].mpi_p,o); | ||||||
|       }       |       }       | ||||||
|       decompresstime+=usecond(); |  | ||||||
|     } |     } | ||||||
|  |     decompresstime+=usecond(); | ||||||
|   } |   } | ||||||
|   //////////////////////////////////////// |   //////////////////////////////////////// | ||||||
|   // Set up routines |   // Set up routines | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ Author: paboyle <paboyle@ph.ed.ac.uk> | |||||||
|  |  | ||||||
| #define PARALLEL_FOR_LOOP        _Pragma("omp parallel for schedule(static)") | #define PARALLEL_FOR_LOOP        _Pragma("omp parallel for schedule(static)") | ||||||
| #define PARALLEL_FOR_LOOP_INTERN _Pragma("omp for schedule(static)") | #define PARALLEL_FOR_LOOP_INTERN _Pragma("omp for schedule(static)") | ||||||
| #define PARALLEL_NESTED_LOOP2 _Pragma("omp parallel for schedule(static) collapse(2)") | #define PARALLEL_NESTED_LOOP2 _Pragma("omp parallel for collapse(2)") | ||||||
| #define PARALLEL_REGION       _Pragma("omp parallel") | #define PARALLEL_REGION       _Pragma("omp parallel") | ||||||
| #define PARALLEL_CRITICAL     _Pragma("omp critical") | #define PARALLEL_CRITICAL     _Pragma("omp critical") | ||||||
| #else | #else | ||||||
|   | |||||||
| @@ -167,7 +167,7 @@ int main (int argc, char ** argv) { | |||||||
|   RealD mass = Params.mass; |   RealD mass = Params.mass; | ||||||
|   RealD M5   = Params.M5; |   RealD M5   = Params.M5; | ||||||
|   std::vector<int> blockSize = Params.blockSize; |   std::vector<int> blockSize = Params.blockSize; | ||||||
|   std::vector<int> latt({16,16,16,16}); |   std::vector<int> latt({32,32,32,32}); | ||||||
|   uint64_t     vol = Ls*latt[0]*latt[1]*latt[2]*latt[3]; |   uint64_t     vol = Ls*latt[0]*latt[1]*latt[2]*latt[3]; | ||||||
|   double   mat_flop= 2.0*1320.0*vol;     |   double   mat_flop= 2.0*1320.0*vol;     | ||||||
|   // Grids |   // Grids | ||||||
|   | |||||||
| @@ -34,6 +34,8 @@ class ScalarActionParameters : Serializable { | |||||||
|     double, lambda, |     double, lambda, | ||||||
|     double, g); |     double, g); | ||||||
|  |  | ||||||
|  |   ScalarActionParameters() = default; | ||||||
|  |  | ||||||
|     template <class ReaderClass > |     template <class ReaderClass > | ||||||
|   ScalarActionParameters(Reader<ReaderClass>& Reader){ |   ScalarActionParameters(Reader<ReaderClass>& Reader){ | ||||||
|     read(Reader, "ScalarAction", *this); |     read(Reader, "ScalarAction", *this); | ||||||
| @@ -125,9 +127,12 @@ int main(int argc, char **argv) { | |||||||
|   TheHMC.Resources.AddGrid("scalar", ScalarGrid); |   TheHMC.Resources.AddGrid("scalar", ScalarGrid); | ||||||
|   std::cout << "Lattice size : " << GridDefaultLatt() << std::endl; |   std::cout << "Lattice size : " << GridDefaultLatt() << std::endl; | ||||||
|    |    | ||||||
|  |   ScalarActionParameters SPar(Reader); | ||||||
|  |  | ||||||
|   // Checkpointer definition |   // Checkpointer definition | ||||||
|   CheckpointerParameters CPparams(Reader); |   CheckpointerParameters CPparams(Reader); | ||||||
|   TheHMC.Resources.LoadBinaryCheckpointer(CPparams); |   //TheHMC.Resources.LoadBinaryCheckpointer(CPparams); | ||||||
|  |   TheHMC.Resources.LoadScidacCheckpointer(CPparams, SPar); | ||||||
|  |  | ||||||
|   RNGModuleParameters RNGpar(Reader); |   RNGModuleParameters RNGpar(Reader); | ||||||
|   TheHMC.Resources.SetRNGSeeds(RNGpar); |   TheHMC.Resources.SetRNGSeeds(RNGpar); | ||||||
| @@ -140,7 +145,6 @@ int main(int argc, char **argv) { | |||||||
|   // Collect actions, here use more encapsulation |   // Collect actions, here use more encapsulation | ||||||
|  |  | ||||||
|   // Scalar action in adjoint representation |   // Scalar action in adjoint representation | ||||||
|   ScalarActionParameters SPar(Reader); |  | ||||||
|   ScalarAction Saction(SPar.mass_squared, SPar.lambda, SPar.g); |   ScalarAction Saction(SPar.mass_squared, SPar.lambda, SPar.g); | ||||||
|  |  | ||||||
|   // Collect actions |   // Collect actions | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ namespace Grid{ | |||||||
|     GRID_SERIALIZABLE_CLASS_MEMBERS(ActionParameters, |     GRID_SERIALIZABLE_CLASS_MEMBERS(ActionParameters, | ||||||
| 				    double, beta) | 				    double, beta) | ||||||
|  |  | ||||||
|  |     ActionParameters() = default; | ||||||
|  |  | ||||||
|     template <class ReaderClass > |     template <class ReaderClass > | ||||||
|     ActionParameters(Reader<ReaderClass>& Reader){ |     ActionParameters(Reader<ReaderClass>& Reader){ | ||||||
| @@ -68,11 +69,15 @@ int main(int argc, char **argv) { | |||||||
|   } |   } | ||||||
|   Serialiser Reader(TheHMC.ParameterFile); |   Serialiser Reader(TheHMC.ParameterFile); | ||||||
|  |  | ||||||
|  |   // Read parameters from input file | ||||||
|  |   ActionParameters WilsonPar(Reader); | ||||||
|  |  | ||||||
|   // Checkpointer definition |   // Checkpointer definition | ||||||
|   CheckpointerParameters CPparams(Reader); |   CheckpointerParameters CPparams(Reader); | ||||||
|   TheHMC.Resources.LoadNerscCheckpointer(CPparams); |   //TheHMC.Resources.LoadNerscCheckpointer(CPparams); | ||||||
|  |  | ||||||
|  |   // Store metadata in the Scidac checkpointer | ||||||
|  |   TheHMC.Resources.LoadScidacCheckpointer(CPparams, WilsonPar); | ||||||
|  |  | ||||||
|   RNGModuleParameters RNGpar(Reader); |   RNGModuleParameters RNGpar(Reader); | ||||||
|   TheHMC.Resources.SetRNGSeeds(RNGpar); |   TheHMC.Resources.SetRNGSeeds(RNGpar); | ||||||
| @@ -91,8 +96,6 @@ int main(int argc, char **argv) { | |||||||
|   // need wrappers of the fermionic classes |   // need wrappers of the fermionic classes | ||||||
|   // that have a complex construction |   // that have a complex construction | ||||||
|   // standard |   // standard | ||||||
|   ActionParameters WilsonPar(Reader); |  | ||||||
|   //RealD beta = 6.4 ; |  | ||||||
|   WilsonGaugeActionR Waction(WilsonPar.beta); |   WilsonGaugeActionR Waction(WilsonPar.beta); | ||||||
|  |  | ||||||
|   ActionLevel<HMCWrapper::Field> Level1(1); |   ActionLevel<HMCWrapper::Field> Level1(1); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user