Compare commits
2 Commits
libpredict
...
fdab7096d4
| Author | SHA1 | Date | |
|---|---|---|---|
| fdab7096d4 | |||
| b5c923193c |
@@ -17,6 +17,83 @@ using std::pair;
|
|||||||
|
|
||||||
#define DEBUG_TRACE 0
|
#define DEBUG_TRACE 0
|
||||||
|
|
||||||
|
static bool findAlloatableKeyword(SgExpression* exp)
|
||||||
|
{
|
||||||
|
if (exp)
|
||||||
|
{
|
||||||
|
if (exp->variant() == ALLOCATABLE_OP)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return findAlloatableKeyword(exp->lhs()) || findAlloatableKeyword(exp->rhs());
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static bool checkDynamicArray(DIST::Array *array)
|
||||||
|
{
|
||||||
|
for (const auto &bounds : array->GetSizes())
|
||||||
|
if (bounds.first == -1 || bounds.second == -1)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static SgExpression* findExprWithVariant(SgExpression* exp, int variant)
|
||||||
|
{
|
||||||
|
if (exp)
|
||||||
|
{
|
||||||
|
if (exp->variant() == variant)
|
||||||
|
return exp;
|
||||||
|
|
||||||
|
auto *l = findExprWithVariant(exp->lhs(), variant);
|
||||||
|
if (l)
|
||||||
|
return l;
|
||||||
|
|
||||||
|
auto *r = findExprWithVariant(exp->rhs(), variant);
|
||||||
|
if (r)
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool checkAssumedSize(SgStatement *st, const string &arrayName, const string ¤tFile)
|
||||||
|
{
|
||||||
|
bool found = false;
|
||||||
|
|
||||||
|
DIST::Array* array_p = getArrayFromDeclarated(st, arrayName);
|
||||||
|
|
||||||
|
auto place = *array_p->GetDeclInfo().begin();
|
||||||
|
auto decl_file_name = place.first;
|
||||||
|
|
||||||
|
SgFile::switchToFile(decl_file_name);
|
||||||
|
|
||||||
|
SgExpression* list = st->expr(0);
|
||||||
|
while (list)
|
||||||
|
{
|
||||||
|
if (list->lhs() && list->lhs()->symbol()->identifier() == arrayName)
|
||||||
|
{
|
||||||
|
if(findExprWithVariant(list->lhs(), STAR_RANGE))
|
||||||
|
found = true;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
auto *dim_expr = findExprWithVariant(st->expr(2), DIMENSION_OP);
|
||||||
|
if (dim_expr && findExprWithVariant(dim_expr, STAR_RANGE))
|
||||||
|
found = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SgFile::switchToFile(currentFile);
|
||||||
|
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
static void findArrays(SgExpression* exp, set<SgSymbol*>& arrays)
|
static void findArrays(SgExpression* exp, set<SgSymbol*>& arrays)
|
||||||
{
|
{
|
||||||
if (exp)
|
if (exp)
|
||||||
@@ -29,7 +106,7 @@ static void findArrays(SgExpression* exp, set<SgSymbol*>& arrays)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void populateDistributedIoArrays(map<SgSymbol*, set<SgStatement*>>& arrays, SgStatement* stat)
|
static void populateDistributedIoArrays(map<SgSymbol*, set<SgStatement*>>& arrays, SgStatement* stat, const string& current_file)
|
||||||
{
|
{
|
||||||
auto var = stat->variant();
|
auto var = stat->variant();
|
||||||
|
|
||||||
@@ -130,8 +207,14 @@ static void populateDistributedIoArrays(map<SgSymbol*, set<SgStatement*>>& array
|
|||||||
{
|
{
|
||||||
string array_name = string(by_symb->identifier());
|
string array_name = string(by_symb->identifier());
|
||||||
DIST::Array* array_p = getArrayFromDeclarated(declaratedInStmt(by_symb), array_name);
|
DIST::Array* array_p = getArrayFromDeclarated(declaratedInStmt(by_symb), array_name);
|
||||||
if (array_p && array_p->GetDistributeFlagVal() == Distribution::distFlag::IO_PRIV && arrays[by_symb].insert(stat).second)
|
if (array_p &&
|
||||||
|
array_p->GetDistributeFlagVal() == Distribution::distFlag::IO_PRIV &&
|
||||||
|
!checkAssumedSize(declaratedInStmt(by_symb), array_name, current_file) &&
|
||||||
|
arrays[by_symb].insert(stat).second
|
||||||
|
)
|
||||||
|
{
|
||||||
__spf_print(DEBUG_TRACE, "[%d]: add array %s\n", stat->lineNumber(), array_p->GetName().c_str());
|
__spf_print(DEBUG_TRACE, "[%d]: add array %s\n", stat->lineNumber(), array_p->GetName().c_str());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
__spf_print(DEBUG_TRACE, "[replace]\n");
|
__spf_print(DEBUG_TRACE, "[replace]\n");
|
||||||
@@ -204,7 +287,12 @@ static void replaceArrayRec(SgSymbol* arr, SgSymbol* replace_by, SgStatement* st
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void copyArrayBetweenStatements(SgSymbol* replace_symb, SgSymbol* replace_by, SgStatement* start, SgStatement* last, bool start_is_scope)
|
static void copyArrayBetweenStatements(SgSymbol* replace_symb,
|
||||||
|
SgSymbol* replace_by,
|
||||||
|
SgStatement* start,
|
||||||
|
SgStatement* last,
|
||||||
|
bool start_is_scope,
|
||||||
|
FuncInfo *func_info)
|
||||||
{
|
{
|
||||||
while (start->lexNext() && !isSgExecutableStatement(start->lexNext()))
|
while (start->lexNext() && !isSgExecutableStatement(start->lexNext()))
|
||||||
start = start->lexNext();
|
start = start->lexNext();
|
||||||
@@ -227,6 +315,18 @@ static void copyArrayBetweenStatements(SgSymbol* replace_symb, SgSymbol* replace
|
|||||||
start->insertStmtAfter(*assign, *parent);
|
start->insertStmtAfter(*assign, *parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (has_write)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < func_info->funcParams.identificators.size(); i++)
|
||||||
|
{
|
||||||
|
if (func_info->funcParams.identificators[i] == replace_symb->identifier())
|
||||||
|
{
|
||||||
|
has_write &= func_info->funcParams.isArgOut(i);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (has_write)
|
if (has_write)
|
||||||
{
|
{
|
||||||
// A = A_reg
|
// A = A_reg
|
||||||
@@ -237,9 +337,15 @@ static void copyArrayBetweenStatements(SgSymbol* replace_symb, SgSymbol* replace
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void replaceArrayInFragment(SgSymbol* replace_symb, const set<SgStatement*> usages, SgSymbol* replace_by, SgStatement* start, SgStatement* last, const string& filename)
|
static void replaceArrayInFragment(SgSymbol* replace_symb,
|
||||||
|
const set<SgStatement*> usages,
|
||||||
|
SgSymbol* replace_by,
|
||||||
|
SgStatement* start,
|
||||||
|
SgStatement* last,
|
||||||
|
FuncInfo *func_info,
|
||||||
|
const string& filename)
|
||||||
{
|
{
|
||||||
while (start->lexNext() && !isSgExecutableStatement(start->lexNext()))
|
while (start->lexNext() && (!isSgExecutableStatement(start->lexNext()) || start->lexNext()->variant() == ALLOCATE_STMT))
|
||||||
start = start->lexNext();
|
start = start->lexNext();
|
||||||
|
|
||||||
set<SgStatement*> not_opened, not_closed, copied;
|
set<SgStatement*> not_opened, not_closed, copied;
|
||||||
@@ -306,7 +412,7 @@ static void replaceArrayInFragment(SgSymbol* replace_symb, const set<SgStatement
|
|||||||
}
|
}
|
||||||
|
|
||||||
__spf_print(DEBUG_TRACE, "[copy %s] [%d, %d]\n", replace_symb->identifier(), scope_start->lineNumber(), scope_end->lineNumber());
|
__spf_print(DEBUG_TRACE, "[copy %s] [%d, %d]\n", replace_symb->identifier(), scope_start->lineNumber(), scope_end->lineNumber());
|
||||||
copyArrayBetweenStatements(replace_symb, replace_by, scope_start, scope_end, copy_scope == scope_start);
|
copyArrayBetweenStatements(replace_symb, replace_by, scope_start, scope_end, copy_scope == scope_start, func_info);
|
||||||
copied.insert(copy_scope);
|
copied.insert(copy_scope);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -365,6 +471,16 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
if (SgFile::switchToFile(filename) == -1)
|
if (SgFile::switchToFile(filename) == -1)
|
||||||
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
||||||
|
|
||||||
|
auto func_info_it = allFuncInfo.find(filename);
|
||||||
|
if (func_info_it == allFuncInfo.end())
|
||||||
|
{
|
||||||
|
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto *lbound_symb = new SgSymbol(PROCEDURE_NAME, "lbound");
|
||||||
|
auto *ubound_symb = new SgSymbol(PROCEDURE_NAME, "ubound");
|
||||||
|
|
||||||
for (auto& lines : linesByFile.second)
|
for (auto& lines : linesByFile.second)
|
||||||
{
|
{
|
||||||
__spf_print(DEBUG_TRACE, "[fragment] %s: %d:%d %d\n", filename.c_str(), lines.lines.first,
|
__spf_print(DEBUG_TRACE, "[fragment] %s: %d:%d %d\n", filename.c_str(), lines.lines.first,
|
||||||
@@ -374,7 +490,6 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
|
|
||||||
if (lines.isImplicit())
|
if (lines.isImplicit())
|
||||||
{
|
{
|
||||||
|
|
||||||
curr_stmt = current_file->SgStatementAtLine(lines.lines.first);
|
curr_stmt = current_file->SgStatementAtLine(lines.lines.first);
|
||||||
end = current_file->SgStatementAtLine(lines.lines.second);
|
end = current_file->SgStatementAtLine(lines.lines.second);
|
||||||
|
|
||||||
@@ -390,6 +505,7 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
map<SgSymbol*, set<SgStatement*>> need_replace;
|
map<SgSymbol*, set<SgStatement*>> need_replace;
|
||||||
SgStatement* last_io_bound = NULL;
|
SgStatement* last_io_bound = NULL;
|
||||||
|
|
||||||
|
FuncInfo *current_func_info = NULL;
|
||||||
|
|
||||||
while (curr_stmt != end)
|
while (curr_stmt != end)
|
||||||
{
|
{
|
||||||
@@ -398,9 +514,24 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
|
|
||||||
auto var = curr_stmt->variant();
|
auto var = curr_stmt->variant();
|
||||||
|
|
||||||
|
|
||||||
if (var == PROC_HEDR || var == PROG_HEDR || var == FUNC_HEDR)
|
if (var == PROC_HEDR || var == PROG_HEDR || var == FUNC_HEDR)
|
||||||
{
|
{
|
||||||
|
current_func_info = NULL;
|
||||||
|
for (auto *func_info : func_info_it->second)
|
||||||
|
{
|
||||||
|
if (func_info->funcName == curr_stmt->symbol()->identifier())
|
||||||
|
{
|
||||||
|
current_func_info = func_info;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!current_func_info)
|
||||||
|
{
|
||||||
|
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
curr_stmt = curr_stmt->lexNext();
|
curr_stmt = curr_stmt->lexNext();
|
||||||
while (curr_stmt && !isSgExecutableStatement(curr_stmt))
|
while (curr_stmt && !isSgExecutableStatement(curr_stmt))
|
||||||
{
|
{
|
||||||
@@ -428,7 +559,7 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
const string locationName = array_p->GetLocation().second;
|
const string locationName = array_p->GetLocation().second;
|
||||||
|
|
||||||
auto place = *array_p->GetDeclInfo().begin();
|
auto place = *array_p->GetDeclInfo().begin();
|
||||||
string fileName = place.first;
|
auto decl_file_name = place.first;
|
||||||
string suffix = "_io_l";
|
string suffix = "_io_l";
|
||||||
|
|
||||||
if (fromModule)
|
if (fromModule)
|
||||||
@@ -437,7 +568,7 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
pair<SgSymbol*, SgSymbol*> copied;
|
pair<SgSymbol*, SgSymbol*> copied;
|
||||||
copied.first = array_to_copy;
|
copied.first = array_to_copy;
|
||||||
|
|
||||||
if (SgFile::switchToFile(fileName) == -1)
|
if (SgFile::switchToFile(decl_file_name) == -1)
|
||||||
{
|
{
|
||||||
auto* func_stmt = curr_stmt->getScopeForDeclare();
|
auto* func_stmt = curr_stmt->getScopeForDeclare();
|
||||||
|
|
||||||
@@ -472,7 +603,9 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
insertPlace->insertStmtAfter(*stat, *st);
|
insertPlace->insertStmtAfter(*stat, *st);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
copied = copyArray(place, array_p, linesByFile.second, suffix + to_string(region->GetId()), fileName, newDeclsToInclude, copied_syms);
|
{
|
||||||
|
copied = copyArray(place, array_p, linesByFile.second, suffix + to_string(region->GetId()), decl_file_name, newDeclsToInclude, copied_syms);
|
||||||
|
}
|
||||||
|
|
||||||
SgStatement* decl = SgStatement::getStatementByFileAndLine(place.first, place.second);
|
SgStatement* decl = SgStatement::getStatementByFileAndLine(place.first, place.second);
|
||||||
|
|
||||||
@@ -492,10 +625,124 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
dir_str += "!$SPF ANALYSIS(PROCESS_PRIVATE(" + string(copied.second->identifier()) + "))\n";
|
dir_str += "!$SPF ANALYSIS(PROCESS_PRIVATE(" + string(copied.second->identifier()) + "))\n";
|
||||||
decl->addComment(dir_str.c_str());
|
decl->addComment(dir_str.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
created_copies.insert({ array_to_copy, copied.second });
|
created_copies.insert({ array_to_copy, copied.second });
|
||||||
|
|
||||||
if (curr_stmt->switchToFile() == -1)
|
// make array-copy allocatable in case of main array shape not constant
|
||||||
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
if(checkDynamicArray(array_p))
|
||||||
|
{
|
||||||
|
// insert allocatable keyword in declaration
|
||||||
|
auto *kword_list = decl->expr(2);
|
||||||
|
if (!findAlloatableKeyword(kword_list))
|
||||||
|
{
|
||||||
|
if (!kword_list)
|
||||||
|
{
|
||||||
|
kword_list = new SgExprListExp();
|
||||||
|
decl->setExpression(2, *kword_list);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (kword_list->rhs())
|
||||||
|
kword_list = kword_list->rhs();
|
||||||
|
|
||||||
|
if (kword_list->lhs())
|
||||||
|
{
|
||||||
|
kword_list->setRhs(new SgExprListExp());
|
||||||
|
kword_list = kword_list->rhs();
|
||||||
|
}
|
||||||
|
|
||||||
|
kword_list->setLhs(new SgExpression(ALLOCATABLE_OP));
|
||||||
|
}
|
||||||
|
|
||||||
|
// insert allocate(a_l(lbound(a, 1):ubound(a,1),...)) statement
|
||||||
|
SgFile::switchToFile(filename);
|
||||||
|
SgStatement* insertPlace = NULL;
|
||||||
|
|
||||||
|
auto* func_stmt = curr_stmt->getScopeForDeclare();
|
||||||
|
for (auto iterator = func_stmt->lexNext();
|
||||||
|
!isSgExecutableStatement(iterator) || isSPF_stat(iterator) &&
|
||||||
|
!(iterator->variant() == SPF_PARALLEL_REG_DIR || iterator->variant() == SPF_END_PARALLEL_REG_DIR);
|
||||||
|
iterator = iterator->lexNext())
|
||||||
|
{
|
||||||
|
insertPlace = iterator;
|
||||||
|
}
|
||||||
|
|
||||||
|
//NULL - no decl stats in function!
|
||||||
|
if (!insertPlace)
|
||||||
|
insertPlace = func_stmt;
|
||||||
|
|
||||||
|
auto st = insertPlace->controlParent();
|
||||||
|
if (st->variant() == GLOBAL)
|
||||||
|
st = insertPlace;
|
||||||
|
|
||||||
|
auto *stat = new SgStatement(ALLOCATE_STMT);
|
||||||
|
auto *created_array_ref = new SgArrayRefExp(*copied.second);
|
||||||
|
|
||||||
|
auto* dim_list = new SgExprListExp();
|
||||||
|
created_array_ref->setLhs(dim_list);
|
||||||
|
|
||||||
|
int dim_len = array_p->GetSizes().size();
|
||||||
|
for (int i = 1; i <= dim_len; i++)
|
||||||
|
{
|
||||||
|
auto *lcall = new SgFunctionCallExp(*lbound_symb);
|
||||||
|
auto *rcall = new SgFunctionCallExp(*ubound_symb);
|
||||||
|
|
||||||
|
for (auto *call : {lcall, rcall})
|
||||||
|
{
|
||||||
|
call->setLhs(new SgExprListExp());
|
||||||
|
|
||||||
|
call->lhs()->setLhs(new SgArrayRefExp(*array_to_copy));
|
||||||
|
call->lhs()->setRhs(new SgValueExp(i));
|
||||||
|
}
|
||||||
|
|
||||||
|
auto *dot_expr = new SgExpression(DDOT);
|
||||||
|
dot_expr->setLhs(lcall);
|
||||||
|
dot_expr->setRhs(rcall);
|
||||||
|
|
||||||
|
dim_list->setLhs(dot_expr);
|
||||||
|
|
||||||
|
if (i < dim_len)
|
||||||
|
{
|
||||||
|
auto *next = new SgExprListExp();
|
||||||
|
dim_list->setRhs(next);
|
||||||
|
dim_list = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stat->setExpression(0, created_array_ref);
|
||||||
|
|
||||||
|
insertPlace->insertStmtAfter(*stat, *st);
|
||||||
|
|
||||||
|
// insert deallocate statemens before all returns
|
||||||
|
auto *find_return_stmt = func_stmt;
|
||||||
|
while (find_return_stmt != func_stmt->lastNodeOfStmt())
|
||||||
|
{
|
||||||
|
auto *next = find_return_stmt->lexNext();
|
||||||
|
if (next && (isSgReturnStmt(next) || next == func_stmt->lastNodeOfStmt()))
|
||||||
|
{
|
||||||
|
if (next->hasLabel())
|
||||||
|
{
|
||||||
|
moveLabelBefore(next);
|
||||||
|
find_return_stmt = next->lexPrev();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto *dealloc_stmt = new SgStatement(DEALLOCATE_STMT);
|
||||||
|
|
||||||
|
dealloc_stmt->setExpression(0, new SgExprListExp());
|
||||||
|
dealloc_stmt->expr(0)->setLhs(new SgArrayRefExp(*copied.second));
|
||||||
|
|
||||||
|
find_return_stmt->insertStmtAfter(*dealloc_stmt, *next->controlParent());
|
||||||
|
|
||||||
|
if (next == curr_stmt)
|
||||||
|
curr_stmt = dealloc_stmt;
|
||||||
|
}
|
||||||
|
|
||||||
|
find_return_stmt = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SgFile::switchToFile(filename);
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -507,7 +754,7 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
{
|
{
|
||||||
auto it = created_copies.find(p.first);
|
auto it = created_copies.find(p.first);
|
||||||
if (it != created_copies.end())
|
if (it != created_copies.end())
|
||||||
replaceArrayInFragment(p.first, p.second, it->second, last_io_bound, curr_stmt, filename);
|
replaceArrayInFragment(p.first, p.second, it->second, last_io_bound, curr_stmt, current_func_info, filename);
|
||||||
else
|
else
|
||||||
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
printInternalError(convertFileName(__FILE__).c_str(), __LINE__);
|
||||||
}
|
}
|
||||||
@@ -527,7 +774,7 @@ void replaceDistributedArraysInIO(vector<ParallelRegion*>& regions,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
populateDistributedIoArrays(need_replace, curr_stmt);
|
populateDistributedIoArrays(need_replace, curr_stmt, filename);
|
||||||
curr_stmt = curr_stmt->lexNext();
|
curr_stmt = curr_stmt->lexNext();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user