|
|
@ -164,6 +164,7 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
for (int i = 0; i < numProcs; i++)
|
|
|
|
for (int i = 0; i < numProcs; i++)
|
|
|
|
PCounter[i] = 0;
|
|
|
|
PCounter[i] = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MilanLongInt NumMessagesBundled;
|
|
|
|
MilanLongInt NumMessagesBundled;
|
|
|
|
MilanInt ghostOwner; // Changed by Fabio to be an integer, addresses needs to be integers!
|
|
|
|
MilanInt ghostOwner; // Changed by Fabio to be an integer, addresses needs to be integers!
|
|
|
|
//vector<MilanLongInt> candidateMate;
|
|
|
|
//vector<MilanLongInt> candidateMate;
|
|
|
@ -213,15 +214,12 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
MilanLongInt privateMyCard = 0;
|
|
|
|
MilanLongInt privateMyCard = 0;
|
|
|
|
staticQueue U, privateU, privateQLocalVtx, privateQGhostVtx, privateQMsgType, privateQOwner;
|
|
|
|
staticQueue U, privateU, privateQLocalVtx, privateQGhostVtx, privateQMsgType, privateQOwner;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
staticQueue privateReqQLocalVtx, privateReqQGhostVtx, privateReqQMsgType, privateReqQOwner;
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool isEmpty;
|
|
|
|
bool isEmpty;
|
|
|
|
#ifdef TIME_TRACKER
|
|
|
|
#ifdef TIME_TRACKER
|
|
|
|
double Ghost2LocalInitialization = MPI_Wtime();
|
|
|
|
double Ghost2LocalInitialization = MPI_Wtime();
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#pragma omp parallel private(insertMe, k, u, w, v, k1, adj1, adj2, adj11, adj12, heaviestEdgeWt, ghostOwner, privateU, privateMyCard, isEmpty, privateQLocalVtx, privateQGhostVtx, privateQMsgType, privateQOwner /*, privateReqQLocalVtx, privateReqQGhostVtx, privateReqQMsgType, privateReqQOwner*/) firstprivate(StartIndex, EndIndex) default(shared) num_threads(4)
|
|
|
|
#pragma omp parallel private(insertMe, k, u, w, v, k1, adj1, adj2, adj11, adj12, heaviestEdgeWt, ghostOwner, privateU, privateMyCard, isEmpty, privateQLocalVtx, privateQGhostVtx, privateQMsgType, privateQOwner) firstprivate(StartIndex, EndIndex) default(shared) num_threads(4)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
// TODO comments about the reduction
|
|
|
|
// TODO comments about the reduction
|
|
|
@ -402,7 +400,7 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
* Create the Queue Data Structure for the Dominating Set
|
|
|
|
* Create the Queue Data Structure for the Dominating Set
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* I had to declare the staticuQueue U before the parallel region
|
|
|
|
* I had to declare the staticuQueue U before the parallel region
|
|
|
|
* to have it in the correct scope. Since we can't chane the dimension
|
|
|
|
* to have it in the correct scope. Since we can't change the dimension
|
|
|
|
* of a staticQueue I had to destroy the previous object and instantiate
|
|
|
|
* of a staticQueue I had to destroy the previous object and instantiate
|
|
|
|
* a new one of the correct size.
|
|
|
|
* a new one of the correct size.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
@ -462,33 +460,22 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
* in parallel.
|
|
|
|
* in parallel.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
MilanLongInt size = numGhostEdges; //TODO how can I decide a meaningfull size?
|
|
|
|
MilanLongInt size = numGhostVertices; //TODO how can I decide a more meaningfull size?
|
|
|
|
//Fail messages
|
|
|
|
//Fail messages
|
|
|
|
privateQLocalVtx.~staticQueue();
|
|
|
|
privateQLocalVtx.~staticQueue();
|
|
|
|
privateQGhostVtx.~staticQueue();
|
|
|
|
privateQGhostVtx.~staticQueue();
|
|
|
|
privateQMsgType.~staticQueue();
|
|
|
|
privateQMsgType.~staticQueue();
|
|
|
|
privateQOwner.~staticQueue();
|
|
|
|
privateQOwner.~staticQueue();
|
|
|
|
//Request messages
|
|
|
|
privateU.~staticQueue();
|
|
|
|
/*
|
|
|
|
|
|
|
|
privateReqQLocalVtx.~staticQueue();
|
|
|
|
new(&privateU) staticQueue(NLVer + numGhostVertices); //TODO how can I put a meaningfull size?
|
|
|
|
privateReqQGhostVtx.~staticQueue();
|
|
|
|
|
|
|
|
privateReqQMsgType.~staticQueue();
|
|
|
|
|
|
|
|
privateReqQOwner.~staticQueue();
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
new(&privateQLocalVtx) staticQueue(size);
|
|
|
|
new(&privateQLocalVtx) staticQueue(size);
|
|
|
|
new(&privateQGhostVtx) staticQueue(size);
|
|
|
|
new(&privateQGhostVtx) staticQueue(size);
|
|
|
|
new(&privateQMsgType) staticQueue(size);
|
|
|
|
new(&privateQMsgType) staticQueue(size);
|
|
|
|
new(&privateQOwner) staticQueue(size);
|
|
|
|
new(&privateQOwner) staticQueue(size);
|
|
|
|
/*
|
|
|
|
|
|
|
|
new(&privateReqQLocalVtx) staticQueue(size);
|
|
|
|
|
|
|
|
new(&privateReqQGhostVtx) staticQueue(size);
|
|
|
|
|
|
|
|
new(&privateReqQMsgType) staticQueue(size);
|
|
|
|
|
|
|
|
new(&privateReqQOwner) staticQueue(size);
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#pragma omp for reduction(+: msgInd, NumMessagesBundled, myCard, PCounter[:numProcs]) schedule(static)
|
|
|
|
#pragma omp for reduction(+: msgInd, NumMessagesBundled, myCard, PCounter[:numProcs]) schedule(static)
|
|
|
|
for ( v=0; v < NLVer; v++ )
|
|
|
|
for (v = 0; v < NLVer; v++) {
|
|
|
|
{
|
|
|
|
|
|
|
|
//Start: PARALLEL_PROCESS_EXPOSED_VERTEX_B(v)
|
|
|
|
//Start: PARALLEL_PROCESS_EXPOSED_VERTEX_B(v)
|
|
|
|
k = candidateMate[v];
|
|
|
|
k = candidateMate[v];
|
|
|
|
candidateMate[v] = verLocInd[k];
|
|
|
|
candidateMate[v] = verLocInd[k];
|
|
|
@ -501,12 +488,9 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
cout<<"\n("<<myRank<<")"<<v+StartIndex<<" Points to: "<<w; fflush(stdout);
|
|
|
|
cout<<"\n("<<myRank<<")"<<v+StartIndex<<" Points to: "<<w; fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
//If found a dominating edge:
|
|
|
|
//If found a dominating edge:
|
|
|
|
if (w >= 0) {
|
|
|
|
if (w >= 0) {
|
|
|
|
|
|
|
|
|
|
|
|
//This piece of code is actually executed under 0.01% of the times
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (isAlreadyMatched(verLocInd[k], StartIndex, EndIndex, GMate, Mate, Ghost2LocalMap)) {
|
|
|
|
if (isAlreadyMatched(verLocInd[k], StartIndex, EndIndex, GMate, Mate, Ghost2LocalMap)) {
|
|
|
|
w = computeCandidateMate(verLocPtr[v],
|
|
|
|
w = computeCandidateMate(verLocPtr[v],
|
|
|
|
verLocPtr[v + 1],
|
|
|
|
verLocPtr[v + 1],
|
|
|
@ -536,28 +520,43 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
assert(ghostOwner != -1);
|
|
|
|
assert(ghostOwner != -1);
|
|
|
|
assert(ghostOwner != myRank);
|
|
|
|
assert(ghostOwner != myRank);
|
|
|
|
PCounter[ghostOwner]++;
|
|
|
|
PCounter[ghostOwner]++;
|
|
|
|
#pragma omp critical(Mate)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//TODO whyyyyy does it fail if I use a private data structure???
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
privateQLocalVtx.push_back(v + StartIndex);
|
|
|
|
|
|
|
|
privateQGhostVtx.push_back(w);
|
|
|
|
|
|
|
|
privateQMsgType.push_back(REQUEST);
|
|
|
|
|
|
|
|
privateQOwner.push_back(ghostOwner);
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#pragma omp critical(MSG)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
QLocalVtx.push_back(v + StartIndex);
|
|
|
|
QLocalVtx.push_back(v + StartIndex);
|
|
|
|
QGhostVtx.push_back(w);
|
|
|
|
QGhostVtx.push_back(w);
|
|
|
|
QMsgType.push_back(REQUEST);
|
|
|
|
QMsgType.push_back(REQUEST);
|
|
|
|
QOwner.push_back(ghostOwner);
|
|
|
|
QOwner.push_back(ghostOwner);
|
|
|
|
|
|
|
|
} // end of critical region
|
|
|
|
|
|
|
|
|
|
|
|
if (candidateMate[NLVer + Ghost2LocalMap[w]] == v + StartIndex) {
|
|
|
|
if (candidateMate[NLVer + Ghost2LocalMap[w]] == v + StartIndex) {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
privateU.push_back(v + StartIndex);
|
|
|
|
|
|
|
|
privateU.push_back(w);
|
|
|
|
Mate[v] = w;
|
|
|
|
Mate[v] = w;
|
|
|
|
|
|
|
|
//FIXME could this instruction create errors?
|
|
|
|
GMate[Ghost2LocalMap[w]] = v + StartIndex; //w is a Ghost
|
|
|
|
GMate[Ghost2LocalMap[w]] = v + StartIndex; //w is a Ghost
|
|
|
|
U.push_back(v + StartIndex);
|
|
|
|
|
|
|
|
U.push_back(w);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
cout<<"\n("<<myRank<<")MATCH: ("<<v+StartIndex<<","<<w<<")"; fflush(stdout);
|
|
|
|
cout<<"\n("<<myRank<<")MATCH: ("<<v+StartIndex<<","<<w<<")"; fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
//Decrement the counter:
|
|
|
|
//Decrement the counter:
|
|
|
|
//Start: PARALLEL_PROCESS_CROSS_EDGE_B(v)
|
|
|
|
//Start: PARALLEL_PROCESS_CROSS_EDGE_B(v)
|
|
|
|
|
|
|
|
#pragma omp critical
|
|
|
|
|
|
|
|
{
|
|
|
|
if (Counter[Ghost2LocalMap[w]] > 0) {
|
|
|
|
if (Counter[Ghost2LocalMap[w]] > 0) {
|
|
|
|
|
|
|
|
|
|
|
|
Counter[Ghost2LocalMap[w]] = Counter[Ghost2LocalMap[w]] - 1; //Decrement
|
|
|
|
Counter[Ghost2LocalMap[w]] -= 1; //Decrement
|
|
|
|
if (Counter[Ghost2LocalMap[w]] == 0) {
|
|
|
|
if (Counter[Ghost2LocalMap[w]] == 0) {
|
|
|
|
S--; //Decrement S
|
|
|
|
S--; //Decrement S
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
@ -565,77 +564,84 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
fflush(stdout);
|
|
|
|
fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
} //End of if Counter[w] > 0
|
|
|
|
} //End of if Counter[w] > 0
|
|
|
|
//End: PARALLEL_PROCESS_CROSS_EDGE_B(v)
|
|
|
|
//End: PARALLEL_PROCESS_CROSS_EDGE_B(v)
|
|
|
|
} //End of if CandidateMate[w] = v
|
|
|
|
} //End of if CandidateMate[w] = v
|
|
|
|
} // end of critical region
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} //End of if a Ghost Vertex
|
|
|
|
} //End of if a Ghost Vertex
|
|
|
|
else { // w is a local vertex
|
|
|
|
else { // w is a local vertex
|
|
|
|
|
|
|
|
|
|
|
|
if (candidateMate[w - StartIndex] == (v + StartIndex)) {
|
|
|
|
if (candidateMate[w - StartIndex] == (v + StartIndex)) {
|
|
|
|
#pragma omp critical(Mate)
|
|
|
|
privateU.push_back(v + StartIndex);
|
|
|
|
{
|
|
|
|
privateU.push_back(w);
|
|
|
|
|
|
|
|
|
|
|
|
Mate[v] = w; //v is local
|
|
|
|
Mate[v] = w; //v is local
|
|
|
|
|
|
|
|
//FIXME this instruction could create errors
|
|
|
|
Mate[w - StartIndex] = v + StartIndex; //w is local
|
|
|
|
Mate[w - StartIndex] = v + StartIndex; //w is local
|
|
|
|
//Q.push_back(u);
|
|
|
|
|
|
|
|
U.push_back(v + StartIndex);
|
|
|
|
|
|
|
|
U.push_back(w);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
cout<<"\n("<<myRank<<")MATCH: ("<<v+StartIndex<<","<<w<<") "; fflush(stdout);
|
|
|
|
cout<<"\n("<<myRank<<")MATCH: ("<<v+StartIndex<<","<<w<<") "; fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
} //End of critical
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} //End of if ( candidateMate[w-StartIndex] == (v+StartIndex) )
|
|
|
|
} //End of if ( candidateMate[w-StartIndex] == (v+StartIndex) )
|
|
|
|
} //End of Else
|
|
|
|
} //End of Else
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
continue;
|
|
|
|
} //End of second if
|
|
|
|
} //End of second if
|
|
|
|
|
|
|
|
|
|
|
|
} //End of if(w >=0)
|
|
|
|
} //End of if(w >=0)
|
|
|
|
|
|
|
|
|
|
|
|
//if (w < 0) { -- if it arrives here this one if is useless, it is certainly -1
|
|
|
|
//This piece of code is executed a really small amount of times, I will not allocate a
|
|
|
|
|
|
|
|
//huge amount of memory to the private data structures.
|
|
|
|
adj11 = verLocPtr[v];
|
|
|
|
adj11 = verLocPtr[v];
|
|
|
|
adj12 = verLocPtr[v + 1];
|
|
|
|
adj12 = verLocPtr[v + 1];
|
|
|
|
for (k1 = adj11; k1 < adj12; k1++) {
|
|
|
|
for (k1 = adj11; k1 < adj12; k1++) {
|
|
|
|
w = verLocInd[k1];
|
|
|
|
w = verLocInd[k1];
|
|
|
|
if ((w < StartIndex) || (w > EndIndex)) { //A ghost
|
|
|
|
if ((w < StartIndex) || (w > EndIndex)) { //A ghost
|
|
|
|
//Build the Message Packet:
|
|
|
|
|
|
|
|
//Message[0] = v+StartIndex; //LOCAL
|
|
|
|
|
|
|
|
//Message[1] = w; //GHOST
|
|
|
|
|
|
|
|
//Message[2] = FAILURE; //TYPE
|
|
|
|
|
|
|
|
//Send a Request (Asynchronous)
|
|
|
|
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
#ifdef PRINT_DEBUG_INFO_
|
|
|
|
cout<<"\n("<<myRank<<")Sending a failure message: ";
|
|
|
|
cout<<"\n("<<myRank<<")Sending a failure message: ";
|
|
|
|
cout<<"\n("<<myRank<<")Ghost is "<<w<<" Owner is: "<<findOwnerOfGhost(w, verDistance, myRank, numProcs);
|
|
|
|
cout<<"\n("<<myRank<<")Ghost is "<<w<<" Owner is: "<<findOwnerOfGhost(w, verDistance, myRank, numProcs);
|
|
|
|
fflush(stdout);
|
|
|
|
fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
/* MPI_Bsend(&Message[0], 3, MPI_INT, inputSubGraph.findOwner(w),
|
|
|
|
|
|
|
|
ComputeTag, comm); */
|
|
|
|
|
|
|
|
NumMessagesBundled++;
|
|
|
|
|
|
|
|
msgInd++;
|
|
|
|
msgInd++;
|
|
|
|
|
|
|
|
NumMessagesBundled++;
|
|
|
|
ghostOwner = findOwnerOfGhost(w, verDistance, myRank, numProcs);
|
|
|
|
ghostOwner = findOwnerOfGhost(w, verDistance, myRank, numProcs);
|
|
|
|
assert(ghostOwner != -1);
|
|
|
|
assert(ghostOwner != -1);
|
|
|
|
assert(ghostOwner != myRank);
|
|
|
|
assert(ghostOwner != myRank);
|
|
|
|
PCounter[ghostOwner]++;
|
|
|
|
PCounter[ghostOwner]++;
|
|
|
|
privateQLocalVtx.push_back(v + StartIndex);
|
|
|
|
QLocalVtx.push_back(v + StartIndex);
|
|
|
|
privateQGhostVtx.push_back(w);
|
|
|
|
QGhostVtx.push_back(w);
|
|
|
|
privateQMsgType.push_back(FAILURE);
|
|
|
|
QMsgType.push_back(FAILURE);
|
|
|
|
privateQOwner.push_back(ghostOwner);
|
|
|
|
QOwner.push_back(ghostOwner);
|
|
|
|
|
|
|
|
|
|
|
|
} //End of if(GHOST)
|
|
|
|
} //End of if(GHOST)
|
|
|
|
} //End of for loop
|
|
|
|
} //End of for loop
|
|
|
|
//} // End of Else: w == -1
|
|
|
|
|
|
|
|
//End: PARALLEL_PROCESS_EXPOSED_VERTEX_B(v)
|
|
|
|
//End: PARALLEL_PROCESS_EXPOSED_VERTEX_B(v)
|
|
|
|
} //End of for ( v=0; v < NLVer; v++ )
|
|
|
|
} //End of for ( v=0; v < NLVer; v++ )
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#pragma omp critical(privateMsg)
|
|
|
|
#pragma omp critical(privateMsg)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
while (!privateQLocalVtx.empty())
|
|
|
|
while (!privateQLocalVtx.empty())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
QLocalVtx.push_back(privateQLocalVtx.pop_front());
|
|
|
|
|
|
|
|
QGhostVtx.push_back(privateQGhostVtx.pop_front());
|
|
|
|
QLocalVtx.push_back(privateQLocalVtx.pop_back());
|
|
|
|
QMsgType.push_back(privateQMsgType.pop_front());
|
|
|
|
QGhostVtx.push_back(privateQGhostVtx.pop_back());
|
|
|
|
QOwner.push_back(privateQOwner.pop_front());
|
|
|
|
QMsgType.push_back(privateQMsgType.pop_back());
|
|
|
|
|
|
|
|
QOwner.push_back(privateQOwner.pop_back());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#pragma omp critical(U)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
while (!privateU.empty())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
U.push_back(privateU.pop_front());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -649,10 +655,6 @@ void dalgoDistEdgeApproxDomEdgesLinearSearchMesgBndlSmallMateCMP(
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/////////////////////////// PROCESS MATCHED VERTICES //////////////////////////////
|
|
|
|
/////////////////////////// PROCESS MATCHED VERTICES //////////////////////////////
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
privateU.~staticQueue();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
new(&privateU) staticQueue(1000); //TODO how can I put a meaningfull size?
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
isEmpty = false;
|
|
|
|
isEmpty = false;
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef COUNT_LOCAL_VERTEX
|
|
|
|
#ifdef COUNT_LOCAL_VERTEX
|
|
|
|