/* * 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 #include #include #include #include #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::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 &exported_types, set exported_functions, set functions) : SM(SM) { map::iterator ci; set::iterator in; for (in = functions.begin(); in != functions.end(); ++in) { FunctionDecl *decl = *in; functions_by_name[decl->getName()] = decl; } set::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 generator::find_superclasses(RecordDecl *decl) { vector 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(*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::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 &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(); 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(); } /* 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; }