mirror of
https://github.com/ldc-developers/ldc.git
synced 2025-04-29 22:50:53 +03:00
218 lines
5.3 KiB
C++
218 lines
5.3 KiB
C++
//===-- gen_gccbuiltins.cpp - GCC builtin module generator ----------------===//
|
||
//
|
||
// LDC – the LLVM D compiler
|
||
//
|
||
// This file is distributed under the BSD-style LDC license. See the LICENSE
|
||
// file for details.
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
//
|
||
// This tool reads the GCC builtin definitions from LLVM's Intrinsics.td for
|
||
// a given architecture and accordingly generates a ldc.gccbuiltins_<arch>
|
||
// module for using them from D code.
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
|
||
#include "llvm/TableGen/Main.h"
|
||
#include "llvm/ADT/SmallString.h"
|
||
#include "llvm/ADT/StringRef.h"
|
||
#include "llvm/Support/CommandLine.h"
|
||
#include "llvm/Support/Path.h"
|
||
#include "llvm/TableGen/Record.h"
|
||
#if LDC_LLVM_VER < 302
|
||
#include "llvm/TableGen/TableGenAction.h"
|
||
#endif
|
||
#include <algorithm>
|
||
#include <assert.h>
|
||
#include <map>
|
||
#include <stdio.h>
|
||
#include <string.h>
|
||
#include <string>
|
||
|
||
using namespace std;
|
||
using namespace llvm;
|
||
|
||
string dtype(Record* rec)
|
||
{
|
||
Init* typeInit = rec->getValueInit("VT");
|
||
if(!typeInit)
|
||
return "";
|
||
|
||
string type = typeInit->getAsString();
|
||
|
||
if(type == "iPTR")
|
||
return "void*";
|
||
|
||
string vec = "";
|
||
|
||
if(type[0] == 'v')
|
||
{
|
||
size_t i = 1;
|
||
while(i != type.size() && type[i] <= '9' && type[i] >= '0')
|
||
i++;
|
||
|
||
vec = type.substr(1, i - 1);
|
||
type = type.substr(i);
|
||
}
|
||
|
||
if(vec.size() > 0 && type.size() > 0)
|
||
{
|
||
int typeSize, vecElements;
|
||
if(
|
||
sscanf(vec.c_str(), "%d", &vecElements) == 1 &&
|
||
sscanf(type.c_str() + 1, "%d", &typeSize) == 1 &&
|
||
typeSize * vecElements > 256)
|
||
{
|
||
return "";
|
||
}
|
||
}
|
||
|
||
if(type == "i8")
|
||
return "byte" + vec;
|
||
else if(type == "i16")
|
||
return "short" + vec;
|
||
else if(type == "i32")
|
||
return "int" + vec;
|
||
else if(type == "i64")
|
||
return "long" + vec;
|
||
else if(type == "f32")
|
||
return "float" + vec;
|
||
else if(type == "f64")
|
||
return "double" + vec;
|
||
else
|
||
return "";
|
||
}
|
||
|
||
string attributes(ListInit* propertyList)
|
||
{
|
||
string prop = propertyList->getSize() ?
|
||
propertyList->getElementAsRecord(0)->getName() : "";
|
||
|
||
return
|
||
prop == "IntrNoMem" ? "nothrow pure @safe" :
|
||
prop == "IntrReadArgMem" ? "nothrow pure" :
|
||
prop == "IntrReadWriteArgMem" ? "nothrow pure" : "nothrow";
|
||
}
|
||
|
||
void processRecord(raw_ostream& os, Record& rec, string arch)
|
||
{
|
||
if(!rec.getValue("GCCBuiltinName"))
|
||
return;
|
||
|
||
string builtinName = rec.getValueAsString("GCCBuiltinName");
|
||
string name = rec.getName();
|
||
|
||
if(name.substr(0, 4) != "int_" || name.find(arch) == string::npos)
|
||
return;
|
||
|
||
name = name.substr(4);
|
||
replace(name.begin(), name.end(), '_', '.');
|
||
name = string("llvm.") + name;
|
||
|
||
ListInit* paramsList = rec.getValueAsListInit("ParamTypes");
|
||
vector<string> params;
|
||
for(unsigned int i = 0; i < paramsList->getSize(); i++)
|
||
{
|
||
string t = dtype(paramsList->getElementAsRecord(i));
|
||
if(t == "")
|
||
return;
|
||
|
||
params.push_back(t);
|
||
}
|
||
|
||
ListInit* retList = rec.getValueAsListInit("RetTypes");
|
||
string ret;
|
||
if(retList->getSize() == 0)
|
||
ret = "void";
|
||
else if(retList->getSize() == 1)
|
||
{
|
||
ret = dtype(retList->getElementAsRecord(0));
|
||
if(ret == "")
|
||
return;
|
||
}
|
||
else
|
||
return;
|
||
|
||
os << "pragma(LDC_intrinsic, \"" + name + "\")\n ";
|
||
os << ret + " " + builtinName + "(";
|
||
|
||
if(params.size())
|
||
os << params[0];
|
||
|
||
for(size_t i = 1; i < params.size(); i++)
|
||
os << ", " << params[i];
|
||
|
||
os << ")" + attributes(rec.getValueAsListInit("Properties")) + ";\n\n";
|
||
}
|
||
|
||
std::string arch;
|
||
|
||
bool emit(raw_ostream& os, RecordKeeper& records)
|
||
{
|
||
os << "module ldc.gccbuiltins_";
|
||
os << arch;
|
||
os << "; \n\nimport core.simd;\n\n";
|
||
|
||
#if LDC_LLVM_VER >= 306
|
||
const auto &defs = records.getDefs();
|
||
#else
|
||
map<string, Record*> defs = records.getDefs();
|
||
#endif
|
||
|
||
for(
|
||
#if LDC_LLVM_VER >= 306
|
||
auto it = defs.cbegin();
|
||
#else
|
||
map<string, Record* >::iterator it = defs.begin();
|
||
#endif
|
||
it != defs.end();
|
||
it++)
|
||
{
|
||
processRecord(os, *(*it).second, arch);
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
#if LDC_LLVM_VER < 302
|
||
struct ActionImpl : TableGenAction
|
||
{
|
||
bool operator()(raw_ostream& os, RecordKeeper& records)
|
||
{
|
||
return emit(os, records);
|
||
}
|
||
};
|
||
#endif
|
||
|
||
int main(int argc, char** argv)
|
||
{
|
||
if(argc != 3)
|
||
{
|
||
fprintf(stderr, "There must be exactly two command line arguments\n");
|
||
return 1;
|
||
}
|
||
|
||
llvm::SmallString<128> file(LLVM_INTRINSIC_TD_PATH);
|
||
sys::path::append(file, "llvm");
|
||
#if LDC_LLVM_VER >= 303
|
||
sys::path::append(file, "IR");
|
||
#endif
|
||
sys::path::append(file, "Intrinsics.td");
|
||
|
||
string iStr = string("-I=") + string(LLVM_INTRINSIC_TD_PATH);
|
||
string oStr = string("-o=") + argv[1];
|
||
|
||
vector<char*> args2(argv, argv + 1);
|
||
args2.push_back(const_cast<char*>(file.c_str()));
|
||
args2.push_back(const_cast<char*>(iStr.c_str()));
|
||
args2.push_back(const_cast<char*>(oStr.c_str()));
|
||
|
||
cl::ParseCommandLineOptions(args2.size(), &args2[0]);
|
||
arch = argv[2];
|
||
#if LDC_LLVM_VER >= 302
|
||
return TableGenMain(argv[0], emit);
|
||
#else
|
||
ActionImpl act;
|
||
return TableGenMain(argv[0], act);
|
||
#endif
|
||
}
|