164 строки
5.1 KiB
C++
164 строки
5.1 KiB
C++
//
|
|
// Copyright (c) 2008 Advanced Micro Devices, Inc. All rights reserved.
|
|
//
|
|
#ifndef WITHOUT_FSA_BACKEND
|
|
|
|
#include <string>
|
|
#include <sstream>
|
|
#include <fstream>
|
|
#include <iostream>
|
|
|
|
#include "os/os.hpp"
|
|
#include "hsadevice.hpp"
|
|
#include "hsaprogram.hpp"
|
|
#include "hsacompilerlib.hpp"
|
|
//#include "gpukernel.hpp"
|
|
//#include "compiler/compiler.hpp"
|
|
#include "utils/options.hpp"
|
|
#include <cstdio>
|
|
|
|
//CLC_IN_PROCESS_CHANGE
|
|
extern int openclFrontEnd(const char* cmdline, std::string*, std::string* typeInfo = NULL);
|
|
|
|
|
|
namespace oclhsa {
|
|
|
|
|
|
/* Temporary log function for the compiler library */
|
|
static void logFunction(const char* msg, size_t size)
|
|
{
|
|
std::cout<< "Compiler Log: " << msg << std::endl;
|
|
}
|
|
|
|
static int programsCount = 0;
|
|
|
|
|
|
bool
|
|
FSAILProgram::compileImpl(const std::string& sourceCode,
|
|
const std::vector<const std::string*>& headers,
|
|
const char** headerIncludeNames,
|
|
amd::option::Options* options)
|
|
{
|
|
|
|
acl_error errorCode;
|
|
aclTargetInfo target;
|
|
target = g_complibApi._aclGetTargetInfo(LP64_SWITCH("hsail","hsail-64"),
|
|
dev().deviceInfo().targetName_,
|
|
&errorCode);
|
|
|
|
//end if asic info is ready
|
|
// We dump the source code for each program (param: headers)
|
|
// into their filenames (headerIncludeNames) into the TEMP
|
|
// folder specific to the OS and add the include path while
|
|
// compiling
|
|
|
|
//Find the temp folder for the OS
|
|
std::string tempFolder = amd::Os::getEnvironment("TEMP");
|
|
if (tempFolder.empty()) {
|
|
tempFolder = amd::Os::getEnvironment("TMP");
|
|
if (tempFolder.empty()) {
|
|
tempFolder = WINDOWS_SWITCH(".","/tmp");;
|
|
}
|
|
}
|
|
//Iterate through each source code and dump it into tmp
|
|
std::fstream f;
|
|
std::vector<std::string> headerFileNames(headers.size());
|
|
std::vector<std::string> newDirs;
|
|
for (size_t i = 0; i < headers.size(); ++i) {
|
|
std::string headerPath = tempFolder;
|
|
std::string headerIncludeName(headerIncludeNames[i]);
|
|
// replace / in path with current os's file separator
|
|
if ( amd::Os::fileSeparator() != '/') {
|
|
for (std::string::iterator it = headerIncludeName.begin(),
|
|
end = headerIncludeName.end();
|
|
it != end;
|
|
++it) {
|
|
if (*it == '/') *it = amd::Os::fileSeparator();
|
|
}
|
|
}
|
|
size_t pos = headerIncludeName.rfind(amd::Os::fileSeparator());
|
|
if (pos != std::string::npos) {
|
|
headerPath += amd::Os::fileSeparator();
|
|
headerPath += headerIncludeName.substr(0, pos);
|
|
headerIncludeName = headerIncludeName.substr(pos+1);
|
|
}
|
|
if (!amd::Os::pathExists(headerPath)) {
|
|
bool ret = amd::Os::createPath(headerPath);
|
|
assert(ret && "failed creating path!");
|
|
newDirs.push_back(headerPath);
|
|
}
|
|
std::string headerFullName
|
|
= headerPath + amd::Os::fileSeparator() + headerIncludeName;
|
|
headerFileNames[i] = headerFullName;
|
|
f.open(headerFullName.c_str(), std::fstream::out);
|
|
//Should we allow asserts
|
|
assert(!f.fail() && "failed creating header file!");
|
|
f.write(headers[i]->c_str(), headers[i]->length());
|
|
f.close();
|
|
}
|
|
|
|
//Create Binary
|
|
binaryElf_ = g_complibApi._aclBinaryInit(sizeof(aclBinary),
|
|
&target,
|
|
&binOpts_,
|
|
&errorCode);
|
|
|
|
if( errorCode!=ACL_SUCCESS ) {
|
|
buildLog_ += "Error while compiling opencl source:\
|
|
aclBinary init failure \n";
|
|
LogWarning("aclBinaryInit failed");
|
|
return false;
|
|
}
|
|
|
|
//Insert opencl into binary
|
|
errorCode = g_complibApi._aclInsertSection(device().compiler(),
|
|
binaryElf_,
|
|
sourceCode.c_str(),
|
|
strlen(sourceCode.c_str()),
|
|
aclSOURCE);
|
|
|
|
if ( errorCode != ACL_SUCCESS ) {
|
|
buildLog_ += "Error while converting to BRIG: \
|
|
Inserting openCl Source \n";
|
|
}
|
|
|
|
|
|
//Set the options for the compiler
|
|
//Set the include path for the temp folder that contains the includes
|
|
if(!headers.empty()) {
|
|
this->compileOptions_.append(" -I");
|
|
this->compileOptions_.append(tempFolder);
|
|
}
|
|
|
|
//Add only for CL2.0 and later
|
|
if (options->oVariables->CLStd[2] >= '2') {
|
|
std::stringstream opts;
|
|
opts << " -D" << "CL_DEVICE_MAX_GLOBAL_VARIABLE_SIZE="
|
|
<< device().info().maxGlobalVariableSize_;
|
|
compileOptions_.append(opts.str());
|
|
}
|
|
|
|
//Compile source to IR
|
|
this->compileOptions_.append(hsailOptions());
|
|
errorCode = g_complibApi._aclCompile(device().compiler(),
|
|
binaryElf_,
|
|
//"-Wf,--support_all_extensions",
|
|
this->compileOptions_.c_str(),
|
|
ACL_TYPE_OPENCL,
|
|
ACL_TYPE_LLVMIR_BINARY,
|
|
logFunction);
|
|
buildLog_ += g_complibApi._aclGetCompilerLog(device().compiler());
|
|
if( errorCode!=ACL_SUCCESS ) {
|
|
LogWarning("aclCompile failed");
|
|
buildLog_ += "Error while compiling \
|
|
opencl source: Compiling CL to IR";
|
|
return false;
|
|
}
|
|
// Save the binary in the interface class
|
|
saveBinaryAndSetType(TYPE_COMPILED);
|
|
return true;
|
|
|
|
}
|
|
}
|
|
#endif // WITHOUT_GPU_BACKEND
|