findInteractingBlocks.cu 19.6 KB
Newer Older
1
#define GROUP_SIZE 256
2
#define BUFFER_SIZE 256
3
4
5
6

/**
 * Find a bounding box for the atoms in each block.
 */
7
8
9
extern "C" __global__ void findBlockBounds(int numAtoms, real4 periodicBoxSize, real4 invPeriodicBoxSize, real4 periodicBoxVecX, real4 periodicBoxVecY, real4 periodicBoxVecZ,
        const real4* __restrict__ posq, real4* __restrict__ blockCenter, real4* __restrict__ blockBoundingBox, int* __restrict__ rebuildNeighborList,
        real2* __restrict__ sortedBlocks) {
10
11
12
13
14
    int index = blockIdx.x*blockDim.x+threadIdx.x;
    int base = index*TILE_SIZE;
    while (base < numAtoms) {
        real4 pos = posq[base];
#ifdef USE_PERIODIC
15
        APPLY_PERIODIC_TO_POS(pos)
16
17
18
19
20
21
22
#endif
        real4 minPos = pos;
        real4 maxPos = pos;
        int last = min(base+TILE_SIZE, numAtoms);
        for (int i = base+1; i < last; i++) {
            pos = posq[i];
#ifdef USE_PERIODIC
23
            real4 center = 0.5f*(maxPos+minPos);
24
            APPLY_PERIODIC_TO_POS_WITH_CENTER(pos, center)
25
26
27
28
#endif
            minPos = make_real4(min(minPos.x,pos.x), min(minPos.y,pos.y), min(minPos.z,pos.z), 0);
            maxPos = make_real4(max(maxPos.x,pos.x), max(maxPos.y,pos.y), max(maxPos.z,pos.z), 0);
        }
29
        real4 blockSize = 0.5f*(maxPos-minPos);
Peter Eastman's avatar
Peter Eastman committed
30
        real4 center = 0.5f*(maxPos+minPos);
31
        center.w = 0;
32
        for (int i = base; i < last; i++) {
Peter Eastman's avatar
Peter Eastman committed
33
34
35
36
37
            pos = posq[i];
            real4 delta = posq[i]-center;
#ifdef USE_PERIODIC
            APPLY_PERIODIC_TO_DELTA(delta)
#endif
38
            center.w = max(center.w, delta.x*delta.x+delta.y*delta.y+delta.z*delta.z);
Peter Eastman's avatar
Peter Eastman committed
39
        }
40
        center.w = sqrt(center.w);
41
        blockBoundingBox[index] = blockSize;
Peter Eastman's avatar
Peter Eastman committed
42
        blockCenter[index] = center;
43
        sortedBlocks[index] = make_real2(blockSize.x+blockSize.y+blockSize.z, index);
44
45
46
47
        index += blockDim.x*gridDim.x;
        base = index*TILE_SIZE;
    }
    if (blockIdx.x == 0 && threadIdx.x == 0)
48
        rebuildNeighborList[0] = 0;
49
50
51
}

/**
52
 * Sort the data about bounding boxes so it can be accessed more efficiently in the next kernel.
53
 */
54
55
56
extern "C" __global__ void sortBoxData(const real2* __restrict__ sortedBlock, const real4* __restrict__ blockCenter,
        const real4* __restrict__ blockBoundingBox, real4* __restrict__ sortedBlockCenter,
        real4* __restrict__ sortedBlockBoundingBox, const real4* __restrict__ posq, const real4* __restrict__ oldPositions,
57
        unsigned int* __restrict__ interactionCount, int* __restrict__ rebuildNeighborList, bool forceRebuild) {
58
59
60
61
62
63
64
65
    for (int i = threadIdx.x+blockIdx.x*blockDim.x; i < NUM_BLOCKS; i += blockDim.x*gridDim.x) {
        int index = (int) sortedBlock[i].y;
        sortedBlockCenter[i] = blockCenter[index];
        sortedBlockBoundingBox[i] = blockBoundingBox[index];
    }
    
    // Also check whether any atom has moved enough so that we really need to rebuild the neighbor list.

66
    bool rebuild = forceRebuild;
67
68
69
70
71
72
73
74
    for (int i = threadIdx.x+blockIdx.x*blockDim.x; i < NUM_ATOMS; i += blockDim.x*gridDim.x) {
        real4 delta = oldPositions[i]-posq[i];
        if (delta.x*delta.x + delta.y*delta.y + delta.z*delta.z > 0.25f*PADDING*PADDING)
            rebuild = true;
    }
    if (rebuild) {
        rebuildNeighborList[0] = 1;
        interactionCount[0] = 0;
75
        interactionCount[1] = 0;
76
77
    }
}
78

79
__device__ int saveSinglePairs(int x, int* atoms, int* flags, int length, unsigned int maxSinglePairs, unsigned int* singlePairCount, int2* singlePairs, int* sumBuffer, volatile unsigned int& pairStartIndex) {
80
81
    // Record interactions that should be computed as single pairs rather than in blocks.
    
82
83
    const int indexInWarp = threadIdx.x%32;
    int sum = 0;
84
    #pragma unroll 8 // (GROUP_SIZE / TILE_SIZE)
85
86
    for (int i = indexInWarp; i < length; i += 32) {
        int count = __popc(flags[i]);
87
        sum += (count <= MAX_BITS_FOR_PAIRS ? count : 0);
88
    }
peastman's avatar
peastman committed
89
90
91
92
    for (int i = 1; i < 32; i *= 2) {
        int n = __shfl_up_sync(0xffffffff, sum, i);
        if (indexInWarp >= i)
            sum += n;
93
    }
peastman's avatar
peastman committed
94
95
96
97
    if (indexInWarp == 31)
        pairStartIndex = atomicAdd(singlePairCount,(unsigned int) sum);
    __syncwarp();
    int prevSum = __shfl_up_sync(0xffffffff, sum, 1);
98
    unsigned int pairIndex = pairStartIndex + (indexInWarp > 0 ? prevSum : 0);
99
100
    for (int i = indexInWarp; i < length; i += 32) {
        int count = __popc(flags[i]);
101
        if (count <= MAX_BITS_FOR_PAIRS && pairIndex+count <= maxSinglePairs) {
102
103
104
105
106
107
108
109
110
            int f = flags[i];
            while (f != 0) {
                singlePairs[pairIndex] = make_int2(atoms[i], x*TILE_SIZE+__ffs(f)-1);
                f &= f-1;
                pairIndex++;
            }
        }
    }
    
111
    // Compact the remaining interactions.
112
113
114
    
    const int warpMask = (1<<indexInWarp)-1;
    int numCompacted = 0;
115
116
    for (int start = 0; start < length; start += 32) {
        int i = start+indexInWarp;
117
118
        int atom = atoms[i];
        int flag = flags[i];
119
        bool include = (i < length && __popc(flags[i]) > MAX_BITS_FOR_PAIRS);
120
        int includeFlags = BALLOT(include);
121
122
123
124
125
126
127
128
129
130
        if (include) {
            int index = numCompacted+__popc(includeFlags&warpMask);
            atoms[index] = atom;
            flags[index] = flag;
        }
        numCompacted += __popc(includeFlags);
    }
    return numCompacted;
}

131
/**
132
133
134
135
136
 * Compare the bounding boxes for each pair of atom blocks (comprised of 32 atoms each), forming a tile. If the two
 * atom blocks are sufficiently far apart, mark them as non-interacting. There are two stages in the algorithm.
 *
 * STAGE 1:
 *
137
 * A coarse grained atom block against interacting atom block neighbour list is constructed. 
138
 *
139
140
141
142
 * Each warp first loads in some block X of interest. Each thread within the warp then loads 
 * in a different atom block Y. If Y has exclusions with X, then Y is not processed.  If the bounding boxes 
 * of the two atom blocks are within the cutoff distance, then the two atom blocks are considered to be
 * interacting and Y is added to the buffer for X.
143
144
145
 *
 * STAGE 2:
 *
146
 * A fine grained atom block against interacting atoms neighbour list is constructed.
147
 *
148
149
150
151
 * The warp loops over atom blocks Y that were found to (possibly) interact with atom block X.  Each thread
 * in the warp loops over the 32 atoms in X and compares their positions to one particular atom from block Y.
 * If it finds one closer than the cutoff distance, the atom is added to the list of atoms interacting with block X.
 * This continues until the buffer fills up, at which point the results are written to global memory.
152
153
154
155
156
157
 *
 * [in] periodicBoxSize        - size of the rectangular periodic box
 * [in] invPeriodicBoxSize     - inverse of the periodic box
 * [in] blockCenter            - the center of each bounding box
 * [in] blockBoundingBox       - bounding box of each atom block
 * [out] interactionCount      - total number of tiles that have interactions
158
 * [out] interactingTiles      - set of blocks that have interactions
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
 * [out] interactingAtoms      - a list of atoms that interact with each atom block
 * [in] posq                   - x,y,z coordinates of each atom and charge q
 * [in] maxTiles               - maximum number of tiles to process, used for multi-GPUs
 * [in] startBlockIndex        - first block to process, used for multi-GPUs,
 * [in] numBlocks              - total number of atom blocks
 * [in] sortedBlocks           - a sorted list of atom blocks based on volume
 * [in] sortedBlockCenter      - sorted centers, duplicated for fast access to avoid indexing
 * [in] sortedBlockBoundingBox - sorted bounding boxes, duplicated for fast access
 * [in] exclusionIndices       - maps into exclusionRowIndices with the starting position for a given atom
 * [in] exclusionRowIndices    - stores the a continuous list of exclusions
 *           eg: block 0 is excluded from atom 3,5,6
 *               block 1 is excluded from atom 3,4
 *               block 2 is excluded from atom 1,3,5,6
 *              exclusionIndices[0][3][5][8]
 *           exclusionRowIndices[3][5][6][3][4][1][3][5][6]
 *                         index 0  1  2  3  4  5  6  7  8 
 * [out] oldPos                - stores the positions of the atoms in which this neighbourlist was built on
 *                             - this is used to decide when to rebuild a neighbourlist
 * [in] rebuildNeighbourList   - whether or not to execute this kernel
 *
179
 */
180
extern "C" __global__ __launch_bounds__(GROUP_SIZE,3) void findBlocksWithInteractions(real4 periodicBoxSize, real4 invPeriodicBoxSize, real4 periodicBoxVecX, real4 periodicBoxVecY, real4 periodicBoxVecZ,
181
        unsigned int* __restrict__ interactionCount, int* __restrict__ interactingTiles, unsigned int* __restrict__ interactingAtoms,
182
183
        int2* __restrict__ singlePairs, const real4* __restrict__ posq, unsigned int maxTiles, unsigned int maxSinglePairs,
        unsigned int startBlockIndex, unsigned int numBlocks, real2* __restrict__ sortedBlocks, const real4* __restrict__ sortedBlockCenter,
184
185
        const real4* __restrict__ sortedBlockBoundingBox, const unsigned int* __restrict__ exclusionIndices, const unsigned int* __restrict__ exclusionRowIndices,
        real4* __restrict__ oldPositions, const int* __restrict__ rebuildNeighborList) {
186

187
188
    if (rebuildNeighborList[0] == 0)
        return; // The neighbor list doesn't need to be rebuilt.
189
190
191
192
193
194
195

    const int indexInWarp = threadIdx.x%32;
    const int warpStart = threadIdx.x-indexInWarp;
    const int totalWarps = blockDim.x*gridDim.x/32;
    const int warpIndex = (blockIdx.x*blockDim.x+threadIdx.x)/32;
    const int warpMask = (1<<indexInWarp)-1;
    __shared__ int workgroupBuffer[BUFFER_SIZE*(GROUP_SIZE/32)];
196
    __shared__ int workgroupFlagsBuffer[BUFFER_SIZE*(GROUP_SIZE/32)];
197
    __shared__ int warpExclusions[MAX_EXCLUSIONS*(GROUP_SIZE/32)];
198
    __shared__ real4 posBuffer[GROUP_SIZE];
199
200
    __shared__ volatile unsigned int workgroupTileIndex[GROUP_SIZE/32];
    __shared__ unsigned int workgroupPairStartIndex[GROUP_SIZE/32];
peastman's avatar
peastman committed
201
    int* sumBuffer = (int*) posBuffer; // Reuse the same buffer to save memory
202
    int* buffer = workgroupBuffer+BUFFER_SIZE*(warpStart/32);
203
    int* flagsBuffer = workgroupFlagsBuffer+BUFFER_SIZE*(warpStart/32);
204
    int* exclusionsForX = warpExclusions+MAX_EXCLUSIONS*(warpStart/32);
205
206
    volatile unsigned int& tileStartIndex = workgroupTileIndex[warpStart/32];
    volatile unsigned int& pairStartIndex = workgroupPairStartIndex[warpStart/32];
207
208

    // Loop over blocks.
209
    
210
211
212
213
    for (int block1 = startBlockIndex+warpIndex; block1 < startBlockIndex+numBlocks; block1 += totalWarps) {
        // Load data for this block.  Note that all threads in a warp are processing the same block.
        
        real2 sortedKey = sortedBlocks[block1];
214
        int x = (int) sortedKey.y;
215
216
217
        real4 blockCenterX = sortedBlockCenter[block1];
        real4 blockSizeX = sortedBlockBoundingBox[block1];
        int neighborsInBuffer = 0;
218
        real4 pos1 = posq[x*TILE_SIZE+indexInWarp];
219
220
221
222
223
224
225
226
#ifdef USE_PERIODIC
        const bool singlePeriodicCopy = (0.5f*periodicBoxSize.x-blockSizeX.x >= PADDED_CUTOFF &&
                                         0.5f*periodicBoxSize.y-blockSizeX.y >= PADDED_CUTOFF &&
                                         0.5f*periodicBoxSize.z-blockSizeX.z >= PADDED_CUTOFF);
        if (singlePeriodicCopy) {
            // The box is small enough that we can just translate all the atoms into a single periodic
            // box, then skip having to apply periodic boundary conditions later.
            
227
            APPLY_PERIODIC_TO_POS_WITH_CENTER(pos1, blockCenterX)
228
229
        }
#endif
230
        pos1.w = 0.5f * (pos1.x * pos1.x + pos1.y * pos1.y + pos1.z * pos1.z);
231
        posBuffer[threadIdx.x] = pos1;
232

233
234
235
236
237
        // Load exclusion data for block x.
        
        const int exclusionStart = exclusionRowIndices[x];
        const int exclusionEnd = exclusionRowIndices[x+1];
        const int numExclusions = exclusionEnd-exclusionStart;
238
        #pragma unroll 4 // (MAX_EXCLUSIONS)
239
        for (int j = indexInWarp; j < numExclusions; j += 32)
240
            exclusionsForX[j] = exclusionIndices[exclusionStart+j];
241
242
        if (MAX_EXCLUSIONS > 32)
            __syncthreads();
243
        
244
245
246
247
248
249
        // Loop over atom blocks to search for neighbors.  The threads in a warp compare block1 against 32
        // other blocks in parallel.

        for (int block2Base = block1+1; block2Base < NUM_BLOCKS; block2Base += 32) {
            int block2 = block2Base+indexInWarp;
            bool includeBlock2 = (block2 < NUM_BLOCKS);
250
            bool forceInclude = false;
251
            if (includeBlock2) {
252
253
                real4 blockCenterY = sortedBlockCenter[block2];
                real4 blockSizeY = sortedBlockBoundingBox[block2];
254
                real4 blockDelta = blockCenterX-blockCenterY;
255
#ifdef USE_PERIODIC
256
                APPLY_PERIODIC_TO_DELTA(blockDelta)
257
#endif
258
                includeBlock2 &= (blockDelta.x*blockDelta.x+blockDelta.y*blockDelta.y+blockDelta.z*blockDelta.z < (PADDED_CUTOFF+blockCenterX.w+blockCenterY.w)*(PADDED_CUTOFF+blockCenterX.w+blockCenterY.w));
259
260
261
262
                blockDelta.x = max(0.0f, fabs(blockDelta.x)-blockSizeX.x-blockSizeY.x);
                blockDelta.y = max(0.0f, fabs(blockDelta.y)-blockSizeX.y-blockSizeY.y);
                blockDelta.z = max(0.0f, fabs(blockDelta.z)-blockSizeX.z-blockSizeY.z);
                includeBlock2 &= (blockDelta.x*blockDelta.x+blockDelta.y*blockDelta.y+blockDelta.z*blockDelta.z < PADDED_CUTOFF_SQUARED);
263
264
265
266
267
#ifdef TRICLINIC
                // The calculation to find the nearest periodic copy is only guaranteed to work if the nearest copy is less than half a box width away.
                // If there's any possibility we might have missed it, do a detailed check.

                if (periodicBoxSize.z/2-blockSizeX.z-blockSizeY.z < PADDED_CUTOFF || periodicBoxSize.y/2-blockSizeX.y-blockSizeY.y < PADDED_CUTOFF)
268
                    includeBlock2 = forceInclude = true;
269
#endif
270
                if (includeBlock2) {
271
                    int y = (int) sortedBlocks[block2].y;
272
                    #pragma unroll 4 // (MAX_EXCLUSIONS)
273
274
275
276
277
278
279
                    for (int k = 0; k < numExclusions; k++)
                        includeBlock2 &= (exclusionsForX[k] != y);
                }
            }
            
            // Loop over any blocks we identified as potentially containing neighbors.
            
280
            int includeBlockFlags = BALLOT(includeBlock2);
Peter Eastman's avatar
Peter Eastman committed
281
            int forceIncludeFlags = BALLOT(forceInclude);
282
283
284
            while (includeBlockFlags != 0) {
                int i = __ffs(includeBlockFlags)-1;
                includeBlockFlags &= includeBlockFlags-1;
Peter Eastman's avatar
Peter Eastman committed
285
                forceInclude = (forceIncludeFlags>>i) & 1;
286
                int y = (int) sortedBlocks[block2Base+i].y;
287

288
                // Check each atom in block Y for interactions.
289

290
                int atom2 = y*TILE_SIZE+indexInWarp;
291
                real4 pos2 = posq[atom2];
292
293
#ifdef USE_PERIODIC
                if (singlePeriodicCopy) {
294
                    APPLY_PERIODIC_TO_POS_WITH_CENTER(pos2, blockCenterX)
295
296
                }
#endif
297
298
                pos2.w = 0.5f * (pos2.x * pos2.x + pos2.y * pos2.y + pos2.z * pos2.z);

299
                real4 blockCenterY = sortedBlockCenter[block2Base+i];
300
                real3 atomDelta = trimTo3(posBuffer[warpStart+indexInWarp])-trimTo3(blockCenterY);
301
302
303
#ifdef USE_PERIODIC
                APPLY_PERIODIC_TO_DELTA(atomDelta)
#endif
304
                int atomFlags = BALLOT(forceInclude || atomDelta.x*atomDelta.x+atomDelta.y*atomDelta.y+atomDelta.z*atomDelta.z < (PADDED_CUTOFF+blockCenterY.w)*(PADDED_CUTOFF+blockCenterY.w));
305
                int interacts = 0;
306
                if (atom2 < NUM_ATOMS && atomFlags != 0) {
307
308
#ifdef USE_PERIODIC
                    if (!singlePeriodicCopy) {
309
310
                        int first = __ffs(atomFlags)-1;
                        int last = 32-__clz(atomFlags);
311
                        for (int j = first; j < last; j++) {
312
                            real3 delta = trimTo3(pos2)-trimTo3(posBuffer[warpStart+j]);
313
                            APPLY_PERIODIC_TO_DELTA(delta)
314
                            interacts |= (delta.x*delta.x+delta.y*delta.y+delta.z*delta.z < PADDED_CUTOFF_SQUARED ? 1<<j : 0);
315
316
317
318
                        }
                    }
                    else {
#endif
319
320
321
322
323
                        #pragma unroll
                        for (int j = 0; j < 32; j++) {
                            real4 posj = posBuffer[warpStart+j];
                            real halfDist2 = posj.w + pos2.w - posj.x*pos2.x - posj.y*pos2.y - posj.z*pos2.z;
                            interacts |= (halfDist2 < 0.5f * PADDED_CUTOFF_SQUARED ? 1<<j : 0);
324
325
326
327
328
329
330
331
                        }
#ifdef USE_PERIODIC
                    }
#endif
                }
                
                // Add any interacting atoms to the buffer.
                
332
                int includeAtomFlags = BALLOT(interacts);
333
334
335
336
337
                if (interacts) {
                    int index = neighborsInBuffer+__popc(includeAtomFlags&warpMask);
                    buffer[index] = atom2;
                    flagsBuffer[index] = interacts;
                }
338
339
340
341
                neighborsInBuffer += __popc(includeAtomFlags);
                if (neighborsInBuffer > BUFFER_SIZE-TILE_SIZE) {
                    // Store the new tiles to memory.
                    
342
343
344
#if MAX_BITS_FOR_PAIRS > 0
                    neighborsInBuffer = saveSinglePairs(x, buffer, flagsBuffer, neighborsInBuffer, maxSinglePairs, &interactionCount[1], singlePairs, sumBuffer+warpStart, pairStartIndex);
#endif
345
                    unsigned int tilesToStore = neighborsInBuffer/TILE_SIZE;
346
347
                    if (tilesToStore > 0) {
                        if (indexInWarp == 0)
348
                            tileStartIndex = atomicAdd(&interactionCount[0], tilesToStore);
349
                        unsigned int newTileStartIndex = tileStartIndex;
350
351
352
                        if (newTileStartIndex+tilesToStore <= maxTiles) {
                            if (indexInWarp < tilesToStore)
                                interactingTiles[newTileStartIndex+indexInWarp] = x;
353
                            #pragma unroll 8 // (GROUP_SIZE / TILE_SIZE)
354
355
356
                            for (int j = 0; j < tilesToStore; j++)
                                interactingAtoms[(newTileStartIndex+j)*TILE_SIZE+indexInWarp] = buffer[indexInWarp+j*TILE_SIZE];
                        }
357
358
                        if (indexInWarp+TILE_SIZE*tilesToStore < BUFFER_SIZE)
                            buffer[indexInWarp] = buffer[indexInWarp+TILE_SIZE*tilesToStore];
359
                        neighborsInBuffer -= TILE_SIZE*tilesToStore;
360
361
                    }
                }
362
            }
363
364
365
366
        }
        
        // If we have a partially filled buffer,  store it to memory.
        
367
#if MAX_BITS_FOR_PAIRS > 0
368
        if (neighborsInBuffer > 32)
369
370
            neighborsInBuffer = saveSinglePairs(x, buffer, flagsBuffer, neighborsInBuffer, maxSinglePairs, &interactionCount[1], singlePairs, sumBuffer+warpStart, pairStartIndex);
#endif
371
        if (neighborsInBuffer > 0) {
372
            unsigned int tilesToStore = (neighborsInBuffer+TILE_SIZE-1)/TILE_SIZE;
373
            if (indexInWarp == 0)
374
                tileStartIndex = atomicAdd(&interactionCount[0], tilesToStore);
375
            unsigned int newTileStartIndex = tileStartIndex;
376
            if (newTileStartIndex+tilesToStore <= maxTiles) {
377
378
                if (indexInWarp < tilesToStore)
                    interactingTiles[newTileStartIndex+indexInWarp] = x;
379
                #pragma unroll 8 // (GROUP_SIZE / TILE_SIZE)
380
381
                for (int j = 0; j < tilesToStore; j++)
                    interactingAtoms[(newTileStartIndex+j)*TILE_SIZE+indexInWarp] = (indexInWarp+j*TILE_SIZE < neighborsInBuffer ? buffer[indexInWarp+j*TILE_SIZE] : NUM_ATOMS);
382
383
384
            }
        }
    }
385
386
387
388
389
    
    // Record the positions the neighbor list is based on.
    
    for (int i = threadIdx.x+blockIdx.x*blockDim.x; i < NUM_ATOMS; i += blockDim.x*gridDim.x)
        oldPositions[i] = posq[i];
390
}