forked from N-BodyShop/changa
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSorter.h
200 lines (168 loc) · 6.69 KB
/
Sorter.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
/** \file Sorter.h
This class manages a parallel sort of particles using a DataManager to hold
suggested splitter keys.
\author Graeme Lufkin (gwl@u.washington.edu)
*/
#ifndef SORTER_H
#define SORTER_H
#include <list>
#include <vector>
#include <set>
#include <numeric>
#include "ParallelGravity.h"
/**
\brief Class to hold the Oct decomposition tree
*/
struct OctDecompNode {
static int maxNumChildren;
static int lgMaxNumChildren;
OctDecompNode() : children(NULL), nchildren(0), nparticles(0)
{
}
NodeKey key;
OctDecompNode *children;
int nchildren;
int64_t nparticles;
void makeSubTree(int refineLevel, CkVec<OctDecompNode*> *active);
int64_t buildCounts();
void deleteBeneath();
void combine(int thresh, vector<NodeKey> &finalKeys, vector<uint64_t> &counts);
void getLeafNodes(CkVec<OctDecompNode*> *activeNodes);
};
/**
\brief Domain decomposition of particles via a parallel sort
*/
/** Domain decomposition is handled differently depending on the type
of decomposition requested. SFC (either Morton ordering or
Peano-Hibert) or Oct (where domains are complete nodes within the
tree) is described first.
In the SFC and Oct case, the Sorter controls a parallel sorting of particles
based on their keys. The goal is to split the particles up into n
pieces of roughly equal sizes.
The Sorter generates a list of possible splitter keys, and broadcasts this
list to the TreePieces. The TreePieces evaluate the list, determining
how many particles they have between each splitter key. These
bin counts are then reduced back to the Sorter::collectEvaluations,
which uses the information to create a new list. When the
last iteration of splitter keys is deemed satisfactory, counts for
the final keys are evaluated, and the keys and counts are broadcast
to DataManager::acceptFinalKeys(), which coordinates the
shuffling of the particles to the correct owners.
*/
class Sorter : public CBase_Sorter {
double decompTime;
/// The total number of keys we're sorting.
int64_t numKeys;
/// The number of chares to sort into.
int numChares;
/// The indexes of the chares that are responsible for each segment of data.
std::vector<int> chareIDs;
/// A list of chare elements to which nothing is assigned
std::vector<int> availableChares;
// Total size of the keys allocated (allows a margin to increase)
//int keysSize;
/// The percent tolerance to sort keys within.
double tolerance;
/// The number of particles on either side of a splitter that corresponds to the requested tolerance.
int closeEnough;
/// The number of iterations completed.
int numIterations;
/// A flag telling if we're done yet.
bool sorted;
std::vector<NodeKey> nodeKeys;
/// The histogram of counts for the last round of splitter keys.
std::vector<uint64_t> binCounts;
std::vector<unsigned int> binCountsGas;
std::vector<unsigned int> binCountsStar;
/// The number of bins in the histogram.
int numCounts;
/// The keys I've decided on that divide the objects evenly (within the tolerance).
std::vector<SFC::Key> keyBoundaries;
std::vector<uint64_t> accumulatedBinCounts;
/// The keys I'm sending out to be evaluated.
std::vector<SFC::Key> splitters;
CkBitVector binsToSplit;
/// The list of object number splits not yet met.
int64_t *goals;
int numGoalsPending;
/// The DataManager I broadcast candidate keys to.
CProxy_DataManager dm;
/// The callback for when the sort is complete.
CkCallback sortingCallback;
/// Variables to decide when to split or join a TreePiece in the Oct decomposition
int joinThreshold, splitThreshold;
/// Specify what is the level of refinement of nodes sent out
/// for histogramming in Oct decomposition.
int refineLevel;
// root of the full tree of node keys for decomposition
OctDecompNode *root;
// node keys for the initial bins
OctDecompNode *decompRoots;
int numDecompRoots;
CkVec<OctDecompNode*> *activeNodes;
CkVec<OctDecompNode*> *tmpActiveNodes;
///Variables added for ORB decomposition
typedef struct DivData{
OrientedBox<float> boundingBox;
double curLow;
double curHigh;
double curDivision;
char curDim;
} ORBData;
/// Contains the split information for the domain decomposition.
/// Splits are inserted into the list as the decomposition tree is
/// built down from the top.
std::list<ORBData> orbData;
/// The weights for all the keys returned by the weightBalance routine
CkVec<int> zeros;
/// The list of nodes opened by the last invocation of weightBalance
CkVec<NodeKey> nodesOpened;
void adjustSplitters();
bool refineOctSplitting(int n, int64_t *count);
public:
Sorter() {
root = NULL;
decompRoots = NULL;
numDecompRoots = 0;
joinThreshold = 0;
activeNodes = new CkVec<OctDecompNode*>;
tmpActiveNodes = new CkVec<OctDecompNode*>;
chareIDs.resize(numTreePieces, 1);
chareIDs[0] = 0;
partial_sum(chareIDs.begin(), chareIDs.end(), chareIDs.begin());
};
Sorter(CkMigrateMessage* m) : CBase_Sorter(m) {
root = NULL;
decompRoots = NULL;
numDecompRoots = 0;
joinThreshold = 0;
chareIDs.resize(numTreePieces, 1);
chareIDs[0] = 0;
partial_sum(chareIDs.begin(), chareIDs.end(), chareIDs.begin());
};
/** Sort the particles in an array of TreePieces using a histogram-probing method.
The the Sorter sends splitters to the TreePieces for them to evaluate.
The evaluation results in a histogram back to the Sorter,
which uses the information to generate new guesses for the splitter keys.
When all the splitter keys have been found (within the percent tolerance per chare)
the TreePieces move the data around to the proper sorted location. The boundary
keys are shared between adjacent TreePieces, as these will be needed in the
tree building phase.
The callback will receive a CkReductionMsg containing no data.
*/
void startSorting(const CkGroupID& dataManagerID, const double toler, const CkCallback& cb, bool decompose);
void convertNodesToSplitters();
SFC::Key * convertNodesToSplittersRefine(int num, NodeKey* keys);
//void convertNodesToSplittersNoZeros(int num, NodeKey* nodeKeys, CkVec<int> &zero);
/// @brief Collect the counts of particles in each domain
void collectEvaluations(CkReductionMsg* m);
void collectEvaluationsSFC(CkReductionMsg* m);
void collectEvaluationsOct(CkReductionMsg* m);
//ORB Decomposition
void doORBDecomposition(CkReductionMsg* m);
void finishPhase(CkReductionMsg *m);
void collectORBCounts(CkReductionMsg* m);
void readytoSendORB(CkReductionMsg* m);
//void sendBoundingBoxes(CkReductionMsg* m);
};
#endif //SORTER_H