2011-04-29 14:27:02 +08:00
|
|
|
//===-- OpenScopImporter.cpp - Import Scops with openscop library --------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Import modified .scop files into Polly. This allows to change the schedule of
|
|
|
|
// statements.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "polly/LinkAllPasses.h"
|
|
|
|
|
|
|
|
#include "polly/Dependences.h"
|
|
|
|
#include "polly/ScopInfo.h"
|
|
|
|
#include "polly/ScopPass.h"
|
|
|
|
|
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Assembly/Writer.h"
|
|
|
|
|
|
|
|
#ifdef OPENSCOP_FOUND
|
|
|
|
|
|
|
|
#define OPENSCOP_INT_T_IS_MP
|
|
|
|
#include "openscop/openscop.h"
|
|
|
|
|
2011-07-09 11:10:22 +08:00
|
|
|
#include "isl/map.h"
|
2011-04-29 14:27:02 +08:00
|
|
|
#include "isl/set.h"
|
|
|
|
#include "isl/constraint.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace polly;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
static cl::opt<std::string>
|
|
|
|
ImportDir("polly-import-dir",
|
|
|
|
cl::desc("The directory to import the .scop files from."),
|
|
|
|
cl::Hidden, cl::value_desc("Directory path"), cl::ValueRequired,
|
|
|
|
cl::init("."));
|
|
|
|
static cl::opt<std::string>
|
|
|
|
ImportPostfix("polly-import-postfix",
|
|
|
|
cl::desc("Postfix to append to the import .scop files."),
|
|
|
|
cl::Hidden, cl::value_desc("File postfix"), cl::ValueRequired,
|
|
|
|
cl::init(""));
|
|
|
|
|
|
|
|
struct ScopImporter : public ScopPass {
|
|
|
|
static char ID;
|
|
|
|
Scop *S;
|
|
|
|
Dependences *D;
|
|
|
|
explicit ScopImporter() : ScopPass(ID) {}
|
|
|
|
bool updateScattering(Scop *S, openscop_scop_p OScop);
|
|
|
|
|
|
|
|
std::string getFileName(Scop *S) const;
|
|
|
|
virtual bool runOnScop(Scop &S);
|
|
|
|
virtual void printScop(raw_ostream &OS) const;
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
char ScopImporter::ID = 0;
|
|
|
|
|
|
|
|
/// @brief Create an isl constraint from a row of OpenScop integers.
|
|
|
|
///
|
|
|
|
/// @param row An array of isl/OpenScop integers.
|
2011-10-06 08:03:35 +08:00
|
|
|
/// @param Space An isl space object, describing how to spilt the dimensions.
|
2011-04-29 14:27:02 +08:00
|
|
|
///
|
|
|
|
/// @return An isl constraint representing this integer array.
|
2011-10-06 08:03:35 +08:00
|
|
|
isl_constraint *constraintFromMatrixRow(isl_int *row, isl_space *Space) {
|
2011-04-29 14:27:02 +08:00
|
|
|
isl_constraint *c;
|
|
|
|
|
2011-10-06 08:03:35 +08:00
|
|
|
unsigned NbOut = isl_space_size(Space, isl_dim_out);
|
|
|
|
unsigned NbIn = isl_space_size(Space, isl_dim_in);
|
|
|
|
unsigned NbParam = isl_space_size(Space, isl_dim_param);
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
if (isl_int_is_zero(row[0]))
|
2011-10-06 08:03:35 +08:00
|
|
|
c = isl_equality_alloc(isl_space_copy(Space));
|
2011-04-29 14:27:02 +08:00
|
|
|
else
|
2011-10-06 08:03:35 +08:00
|
|
|
c = isl_inequality_alloc(isl_space_copy(Space));
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
unsigned current_column = 1;
|
|
|
|
|
|
|
|
for (unsigned j = 0; j < NbOut; ++j)
|
2011-10-06 08:03:35 +08:00
|
|
|
isl_constraint_set_coefficient(c, isl_space_out, j, row[current_column++]);
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
for (unsigned j = 0; j < NbIn; ++j)
|
2011-10-06 08:03:35 +08:00
|
|
|
isl_constraint_set_coefficient(c, isl_space_in, j, row[current_column++]);
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
for (unsigned j = 0; j < NbParam; ++j)
|
2011-10-06 08:03:35 +08:00
|
|
|
isl_constraint_set_coefficient(c, isl_space_param, j, row[current_column++]);
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
isl_constraint_set_constant(c, row[current_column]);
|
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @brief Create an isl map from a OpenScop matrix.
|
|
|
|
///
|
|
|
|
/// @param m The OpenScop matrix to translate.
|
2011-10-06 08:03:35 +08:00
|
|
|
/// @param Space The dimensions that are contained in the OpenScop matrix.
|
2011-04-29 14:27:02 +08:00
|
|
|
///
|
|
|
|
/// @return An isl map representing m.
|
2011-10-06 08:03:35 +08:00
|
|
|
isl_map *mapFromMatrix(openscop_matrix_p m, isl_space *Space) {
|
|
|
|
isl_basic_map *bmap = isl_basic_map_universe(isl_space_copy(Space));
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
for (unsigned i = 0; i < m->NbRows; ++i) {
|
|
|
|
isl_constraint *c;
|
|
|
|
|
2011-10-06 08:03:35 +08:00
|
|
|
c = constraintFromMatrixRow(m->p[i], Space);
|
2011-04-29 14:27:02 +08:00
|
|
|
bmap = isl_basic_map_add_constraint(bmap, c);
|
|
|
|
}
|
|
|
|
|
|
|
|
return isl_map_from_basic_map(bmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @brief Create a new scattering for PollyStmt.
|
|
|
|
///
|
|
|
|
/// @param m The matrix describing the new scattering.
|
|
|
|
/// @param PollyStmt The statement to create the scattering for.
|
|
|
|
///
|
|
|
|
/// @return An isl_map describing the scattering.
|
|
|
|
isl_map *scatteringForStmt(openscop_matrix_p m, ScopStmt *PollyStmt) {
|
|
|
|
|
|
|
|
unsigned NbParam = PollyStmt->getNumParams();
|
|
|
|
unsigned NbIterators = PollyStmt->getNumIterators();
|
|
|
|
unsigned NbScattering = m->NbColumns - 2 - NbParam - NbIterators;
|
|
|
|
|
|
|
|
isl_ctx *ctx = PollyStmt->getParent()->getCtx();
|
2011-10-06 08:03:35 +08:00
|
|
|
isl_space *Space = isl_dim_alloc(ctx, NbParam, NbIterators, NbScattering);
|
|
|
|
Space = isl_space_set_tuple_name(Space, isl_dim_out, "scattering");
|
|
|
|
Space = isl_space_set_tuple_name(Space, isl_dim_in, PollyStmt->getBaseName());
|
|
|
|
isl_map *map = mapFromMatrix(m, Space);
|
|
|
|
isl_space_free(Space);
|
2011-04-29 14:27:02 +08:00
|
|
|
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef Dependences::StatementToIslMapTy StatementToIslMapTy;
|
|
|
|
|
|
|
|
/// @brief Read the new scattering from the OpenScop description.
|
|
|
|
///
|
|
|
|
/// @S The Scop to update
|
|
|
|
/// @OScop The OpenScop data structure describing the new scattering.
|
|
|
|
/// @return A map that contains for each Statement the new scattering.
|
|
|
|
StatementToIslMapTy *readScattering(Scop *S, openscop_scop_p OScop) {
|
|
|
|
StatementToIslMapTy &NewScattering = *(new StatementToIslMapTy());
|
|
|
|
openscop_statement_p stmt = OScop->statement;
|
|
|
|
|
|
|
|
for (Scop::iterator SI = S->begin(), SE = S->end(); SI != SE; ++SI) {
|
|
|
|
|
|
|
|
if ((*SI)->isFinalRead())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!stmt) {
|
|
|
|
errs() << "Not enough statements available in OpenScop file\n";
|
|
|
|
delete &NewScattering;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
NewScattering[*SI] = scatteringForStmt(stmt->schedule, *SI);
|
|
|
|
stmt = stmt->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stmt) {
|
|
|
|
errs() << "Too many statements in OpenScop file\n";
|
|
|
|
delete &NewScattering;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return &NewScattering;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @brief Update the scattering in a Scop using the OpenScop description of
|
|
|
|
/// the scattering.
|
|
|
|
///
|
|
|
|
/// @S The Scop to update
|
|
|
|
/// @OScop The OpenScop data structure describing the new scattering.
|
|
|
|
/// @return Returns false, if the update failed.
|
|
|
|
bool ScopImporter::updateScattering(Scop *S, openscop_scop_p OScop) {
|
|
|
|
StatementToIslMapTy *NewScattering = readScattering(S, OScop);
|
|
|
|
|
|
|
|
if (!NewScattering)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!D->isValidScattering(NewScattering)) {
|
|
|
|
errs() << "OpenScop file contains a scattering that changes the "
|
|
|
|
<< "dependences. Use -disable-polly-legality to continue anyways\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Scop::iterator SI = S->begin(), SE = S->end(); SI != SE; ++SI) {
|
|
|
|
ScopStmt *Stmt = *SI;
|
|
|
|
|
|
|
|
if (NewScattering->find(Stmt) != NewScattering->end())
|
|
|
|
Stmt->setScattering((*NewScattering)[Stmt]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
std::string ScopImporter::getFileName(Scop *S) const {
|
|
|
|
std::string FunctionName =
|
|
|
|
S->getRegion().getEntry()->getParent()->getNameStr();
|
|
|
|
std::string FileName = FunctionName + "___" + S->getNameStr() + ".scop";
|
|
|
|
return FileName;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScopImporter::printScop(raw_ostream &OS) const {
|
|
|
|
S->print(OS);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScopImporter::runOnScop(Scop &scop) {
|
|
|
|
S = &scop;
|
|
|
|
Region &R = scop.getRegion();
|
|
|
|
D = &getAnalysis<Dependences>();
|
|
|
|
|
|
|
|
std::string FileName = ImportDir + "/" + getFileName(S) + ImportPostfix;
|
|
|
|
FILE *F = fopen(FileName.c_str(), "r");
|
|
|
|
|
|
|
|
if (!F) {
|
|
|
|
errs() << "Cannot open file: " << FileName << "\n";
|
|
|
|
errs() << "Skipping import.\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
openscop_scop_p openscop = openscop_scop_read(F);
|
|
|
|
fclose(F);
|
|
|
|
|
|
|
|
std::string FunctionName = R.getEntry()->getParent()->getNameStr();
|
|
|
|
errs() << "Reading Scop '" << R.getNameStr() << "' in function '"
|
|
|
|
<< FunctionName << "' from '" << FileName << "'.\n";
|
|
|
|
|
|
|
|
bool UpdateSuccessfull = updateScattering(S, openscop);
|
|
|
|
|
|
|
|
if (!UpdateSuccessfull) {
|
|
|
|
errs() << "Update failed" << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScopImporter::getAnalysisUsage(AnalysisUsage &AU) const {
|
|
|
|
ScopPass::getAnalysisUsage(AU);
|
|
|
|
AU.addRequired<Dependences>();
|
|
|
|
}
|
|
|
|
|
|
|
|
static RegisterPass<ScopImporter> A("polly-import",
|
|
|
|
"Polly - Import Scops with OpenScop library"
|
|
|
|
" (Reads a .scop file for each Scop)"
|
|
|
|
);
|
|
|
|
|
|
|
|
Pass *polly::createScopImporterPass() {
|
|
|
|
return new ScopImporter();
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|