451 lines
12 KiB
C++
451 lines
12 KiB
C++
/*
|
|
* Copyright 2011,2015 Sven Verdoolaege. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the following
|
|
* disclaimer in the documentation and/or other materials provided
|
|
* with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY SVEN VERDOOLAEGE ''AS IS'' AND ANY
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL SVEN VERDOOLAEGE OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* The views and conclusions contained in the software and documentation
|
|
* are those of the authors and should not be interpreted as
|
|
* representing official policies, either expressed or implied, of
|
|
* Sven Verdoolaege.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <iostream>
|
|
|
|
#include <clang/AST/Attr.h>
|
|
#include <clang/Basic/SourceManager.h>
|
|
|
|
#include "isl_config.h"
|
|
#include "extract_interface.h"
|
|
#include "generator.h"
|
|
|
|
/* Compare the prefix of "s" to "prefix" up to the length of "prefix".
|
|
*/
|
|
static int prefixcmp(const char *s, const char *prefix)
|
|
{
|
|
return strncmp(s, prefix, strlen(prefix));
|
|
}
|
|
|
|
/* Should "method" be considered to be a static method?
|
|
* That is, is the first argument something other than
|
|
* an instance of the class?
|
|
*/
|
|
bool generator::is_static(const isl_class &clazz, FunctionDecl *method)
|
|
{
|
|
ParmVarDecl *param = method->getParamDecl(0);
|
|
QualType type = param->getOriginalType();
|
|
|
|
if (!is_isl_type(type))
|
|
return true;
|
|
return extract_type(type) != clazz.name;
|
|
}
|
|
|
|
/* Find the FunctionDecl with name "name",
|
|
* returning NULL if there is no such FunctionDecl.
|
|
* If "required" is set, then error out if no FunctionDecl can be found.
|
|
*/
|
|
FunctionDecl *generator::find_by_name(const string &name, bool required)
|
|
{
|
|
map<string, FunctionDecl *>::iterator i;
|
|
|
|
i = functions_by_name.find(name);
|
|
if (i != functions_by_name.end())
|
|
return i->second;
|
|
if (required)
|
|
die("No " + name + " function found");
|
|
return NULL;
|
|
}
|
|
|
|
/* Collect all functions that belong to a certain type, separating
|
|
* constructors from regular methods and keeping track of the _to_str,
|
|
* _copy and _free functions, if any, separately. If there are any overloaded
|
|
* functions, then they are grouped based on their name after removing the
|
|
* argument type suffix.
|
|
*/
|
|
generator::generator(SourceManager &SM, set<RecordDecl *> &exported_types,
|
|
set<FunctionDecl *> exported_functions, set<FunctionDecl *> functions) :
|
|
SM(SM)
|
|
{
|
|
map<string, isl_class>::iterator ci;
|
|
|
|
set<FunctionDecl *>::iterator in;
|
|
for (in = functions.begin(); in != functions.end(); ++in) {
|
|
FunctionDecl *decl = *in;
|
|
functions_by_name[decl->getName()] = decl;
|
|
}
|
|
|
|
set<RecordDecl *>::iterator it;
|
|
for (it = exported_types.begin(); it != exported_types.end(); ++it) {
|
|
RecordDecl *decl = *it;
|
|
string name = decl->getName();
|
|
classes[name].name = name;
|
|
classes[name].type = decl;
|
|
classes[name].fn_to_str = find_by_name(name + "_to_str", false);
|
|
classes[name].fn_copy = find_by_name(name + "_copy", true);
|
|
classes[name].fn_free = find_by_name(name + "_free", true);
|
|
}
|
|
|
|
for (in = exported_functions.begin(); in != exported_functions.end();
|
|
++in) {
|
|
FunctionDecl *method = *in;
|
|
isl_class *c = method2class(method);
|
|
|
|
if (!c)
|
|
continue;
|
|
if (is_constructor(method)) {
|
|
c->constructors.insert(method);
|
|
} else {
|
|
string fullname = c->name_without_type_suffix(method);
|
|
c->methods[fullname].insert(method);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Print error message "msg" and abort.
|
|
*/
|
|
void generator::die(const char *msg)
|
|
{
|
|
fprintf(stderr, "%s\n", msg);
|
|
abort();
|
|
}
|
|
|
|
/* Print error message "msg" and abort.
|
|
*/
|
|
void generator::die(string msg)
|
|
{
|
|
die(msg.c_str());
|
|
}
|
|
|
|
/* Return a sequence of the types of which the given type declaration is
|
|
* marked as being a subtype.
|
|
* The order of the types is the opposite of the order in which they
|
|
* appear in the source. In particular, the first annotation
|
|
* is the one that is closest to the annotated type and the corresponding
|
|
* type is then also the first that will appear in the sequence of types.
|
|
*/
|
|
std::vector<string> generator::find_superclasses(RecordDecl *decl)
|
|
{
|
|
vector<string> super;
|
|
|
|
if (!decl->hasAttrs())
|
|
return super;
|
|
|
|
string sub = "isl_subclass";
|
|
size_t len = sub.length();
|
|
AttrVec attrs = decl->getAttrs();
|
|
for (AttrVec::const_iterator i = attrs.begin(); i != attrs.end(); ++i) {
|
|
const AnnotateAttr *ann = dyn_cast<AnnotateAttr>(*i);
|
|
if (!ann)
|
|
continue;
|
|
string s = ann->getAnnotation().str();
|
|
if (s.substr(0, len) == sub) {
|
|
s = s.substr(len + 1, s.length() - len - 2);
|
|
super.push_back(s);
|
|
}
|
|
}
|
|
|
|
return super;
|
|
}
|
|
|
|
/* Is decl marked as being part of an overloaded method?
|
|
*/
|
|
bool generator::is_overload(Decl *decl)
|
|
{
|
|
return has_annotation(decl, "isl_overload");
|
|
}
|
|
|
|
/* Is decl marked as a constructor?
|
|
*/
|
|
bool generator::is_constructor(Decl *decl)
|
|
{
|
|
return has_annotation(decl, "isl_constructor");
|
|
}
|
|
|
|
/* Is decl marked as consuming a reference?
|
|
*/
|
|
bool generator::takes(Decl *decl)
|
|
{
|
|
return has_annotation(decl, "isl_take");
|
|
}
|
|
|
|
/* Is decl marked as preserving a reference?
|
|
*/
|
|
bool generator::keeps(Decl *decl)
|
|
{
|
|
return has_annotation(decl, "isl_keep");
|
|
}
|
|
|
|
/* Is decl marked as returning a reference that is required to be freed.
|
|
*/
|
|
bool generator::gives(Decl *decl)
|
|
{
|
|
return has_annotation(decl, "isl_give");
|
|
}
|
|
|
|
/* Return the class that has a name that matches the initial part
|
|
* of the name of function "fd" or NULL if no such class could be found.
|
|
*/
|
|
isl_class *generator::method2class(FunctionDecl *fd)
|
|
{
|
|
string best;
|
|
map<string, isl_class>::iterator ci;
|
|
string name = fd->getNameAsString();
|
|
|
|
for (ci = classes.begin(); ci != classes.end(); ++ci) {
|
|
if (name.substr(0, ci->first.length()) == ci->first)
|
|
best = ci->first;
|
|
}
|
|
|
|
if (classes.find(best) == classes.end()) {
|
|
cerr << "Unable to find class of " << name << endl;
|
|
return NULL;
|
|
}
|
|
|
|
return &classes[best];
|
|
}
|
|
|
|
/* Is "type" the type "isl_ctx *"?
|
|
*/
|
|
bool generator::is_isl_ctx(QualType type)
|
|
{
|
|
if (!type->isPointerType())
|
|
return 0;
|
|
type = type->getPointeeType();
|
|
if (type.getAsString() != "isl_ctx")
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
/* Is the first argument of "fd" of type "isl_ctx *"?
|
|
*/
|
|
bool generator::first_arg_is_isl_ctx(FunctionDecl *fd)
|
|
{
|
|
ParmVarDecl *param;
|
|
|
|
if (fd->getNumParams() < 1)
|
|
return false;
|
|
|
|
param = fd->getParamDecl(0);
|
|
return is_isl_ctx(param->getOriginalType());
|
|
}
|
|
|
|
namespace {
|
|
|
|
struct ClangAPI {
|
|
/* Return the first location in the range returned by
|
|
* clang::SourceManager::getImmediateExpansionRange.
|
|
* Older versions of clang return a pair of SourceLocation objects.
|
|
* More recent versions return a CharSourceRange.
|
|
*/
|
|
static SourceLocation range_begin(
|
|
const std::pair<SourceLocation,SourceLocation> &p) {
|
|
return p.first;
|
|
}
|
|
static SourceLocation range_begin(const CharSourceRange &range) {
|
|
return range.getBegin();
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
/* Does the callback argument "param" take its argument at position "pos"?
|
|
*
|
|
* The memory management annotations of arguments to function pointers
|
|
* are not recorded by clang, so the information cannot be extracted
|
|
* from the type of "param".
|
|
* Instead, go to the location in the source where the callback argument
|
|
* is declared, look for the right argument of the callback itself and
|
|
* then check if it has an "__isl_take" memory management annotation.
|
|
*
|
|
* If the return value of the function has a memory management annotation,
|
|
* then the spelling of "param" will point to the spelling
|
|
* of this memory management annotation. Since the macro is defined
|
|
* on the command line (in main), this location does not have a file entry.
|
|
* In this case, move up one level in the macro expansion to the location
|
|
* where the memory management annotation is used.
|
|
*/
|
|
bool generator::callback_takes_argument(ParmVarDecl *param,
|
|
int pos)
|
|
{
|
|
SourceLocation loc;
|
|
const char *s, *end, *next;
|
|
bool takes, keeps;
|
|
|
|
loc = param->getSourceRange().getBegin();
|
|
if (!SM.getFileEntryForID(SM.getFileID(SM.getSpellingLoc(loc))))
|
|
loc = ClangAPI::range_begin(SM.getImmediateExpansionRange(loc));
|
|
s = SM.getCharacterData(loc);
|
|
if (!s)
|
|
die("No character data");
|
|
s = strchr(s, '(');
|
|
if (!s)
|
|
die("Cannot find function pointer");
|
|
s = strchr(s + 1, '(');
|
|
if (!s)
|
|
die("Cannot find function pointer arguments");
|
|
end = strchr(s + 1, ')');
|
|
if (!end)
|
|
die("Cannot find end of function pointer arguments");
|
|
while (pos-- > 0) {
|
|
s = strchr(s + 1, ',');
|
|
if (!s || s > end)
|
|
die("Cannot find function pointer argument");
|
|
}
|
|
next = strchr(s + 1, ',');
|
|
if (next && next < end)
|
|
end = next;
|
|
s = strchr(s + 1, '_');
|
|
if (!s || s > end)
|
|
die("Cannot find function pointer argument annotation");
|
|
takes = prefixcmp(s, "__isl_take") == 0;
|
|
keeps = prefixcmp(s, "__isl_keep") == 0;
|
|
if (!takes && !keeps)
|
|
die("Cannot find function pointer argument annotation");
|
|
|
|
return takes;
|
|
}
|
|
|
|
/* Is "type" that of a pointer to an isl_* structure?
|
|
*/
|
|
bool generator::is_isl_type(QualType type)
|
|
{
|
|
if (type->isPointerType()) {
|
|
string s;
|
|
|
|
type = type->getPointeeType();
|
|
if (type->isFunctionType())
|
|
return false;
|
|
s = type.getAsString();
|
|
return s.substr(0, 4) == "isl_";
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/* Is "type" the type isl_bool?
|
|
*/
|
|
bool generator::is_isl_bool(QualType type)
|
|
{
|
|
string s;
|
|
|
|
if (type->isPointerType())
|
|
return false;
|
|
|
|
s = type.getAsString();
|
|
return s == "isl_bool";
|
|
}
|
|
|
|
/* Is "type" the type isl_stat?
|
|
*/
|
|
bool generator::is_isl_stat(QualType type)
|
|
{
|
|
string s;
|
|
|
|
if (type->isPointerType())
|
|
return false;
|
|
|
|
s = type.getAsString();
|
|
return s == "isl_stat";
|
|
}
|
|
|
|
|
|
/* Is "type" that of a pointer to a function?
|
|
*/
|
|
bool generator::is_callback(QualType type)
|
|
{
|
|
if (!type->isPointerType())
|
|
return false;
|
|
type = type->getPointeeType();
|
|
return type->isFunctionType();
|
|
}
|
|
|
|
/* Is "type" that of "char *" of "const char *"?
|
|
*/
|
|
bool generator::is_string(QualType type)
|
|
{
|
|
if (type->isPointerType()) {
|
|
string s = type->getPointeeType().getAsString();
|
|
return s == "const char" || s == "char";
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/* Is "type" that of "long"?
|
|
*/
|
|
bool generator::is_long(QualType type)
|
|
{
|
|
const BuiltinType *builtin = type->getAs<BuiltinType>();
|
|
return builtin && builtin->getKind() == BuiltinType::Long;
|
|
}
|
|
|
|
/* Return the name of the type that "type" points to.
|
|
* The input "type" is assumed to be a pointer type.
|
|
*/
|
|
string generator::extract_type(QualType type)
|
|
{
|
|
if (type->isPointerType())
|
|
return type->getPointeeType().getAsString();
|
|
die("Cannot extract type from non-pointer type");
|
|
}
|
|
|
|
/* Given the type of a function pointer, return the corresponding
|
|
* function prototype.
|
|
*/
|
|
const FunctionProtoType *generator::extract_prototype(QualType type)
|
|
{
|
|
return type->getPointeeType()->getAs<FunctionProtoType>();
|
|
}
|
|
|
|
/* If "method" is overloaded, then return its name with the suffix
|
|
* corresponding to the type of the final argument removed.
|
|
* Otherwise, simply return the name of the function.
|
|
*/
|
|
string isl_class::name_without_type_suffix(FunctionDecl *method)
|
|
{
|
|
int num_params;
|
|
ParmVarDecl *param;
|
|
string name, type;
|
|
size_t name_len, type_len;
|
|
|
|
name = method->getName();
|
|
if (!generator::is_overload(method))
|
|
return name;
|
|
|
|
num_params = method->getNumParams();
|
|
param = method->getParamDecl(num_params - 1);
|
|
type = generator::extract_type(param->getOriginalType());
|
|
type = type.substr(4);
|
|
name_len = name.length();
|
|
type_len = type.length();
|
|
|
|
if (name_len > type_len && name.substr(name_len - type_len) == type)
|
|
name = name.substr(0, name_len - type_len - 1);
|
|
|
|
return name;
|
|
}
|