Urho3D/Source/Clang-Tools/Annotator/Annotator.cpp
2021-07-17 16:43:46 +00:00

197 lines
7.4 KiB
C++

//
// Copyright (c) 2008-2021 the Urho3D project.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
//
#include <clang/ASTMatchers/ASTMatchFinder.h>
#include <clang/Tooling/CommonOptionsParser.h>
#include <clang/Tooling/Refactoring.h>
#include <unordered_map>
#include <unordered_set>
using namespace clang;
using namespace clang::ast_matchers;
using namespace clang::driver;
using namespace clang::tooling;
using namespace llvm;
static cl::extrahelp commonHelp(CommonOptionsParser::HelpMessage);
static cl::extrahelp moreHelp(
"\tFor example, to run Annotator on all files in a subtree of the\n"
"\tsource tree, use:\n"
"\n"
"\t find path/in/substree -name '*.cpp'|xargs Annotator -p build/path\n"
"\n"
"\tNote, that path/in/subtree and current directory should follow the\n"
"\trules described above.\n"
"\n"
"Most probably you want to invoke 'annotate' built-in target instead of invoking this tool\n"
"directly. The 'annotate' target invokes this tool in a right context prepared by build system.\n"
"\n"
);
static cl::OptionCategory annotatorCategory("Annotator options");
// ClangTool only takes a reference to the array list without owning it, so we need to keep the filtered list ourselves
class PathFilter
{
public:
template <typename Fn> PathFilter(const std::vector<std::string> sourcePaths, Fn fn)
{
std::copy_if(sourcePaths.begin(), sourcePaths.end(), std::back_inserter(pathList_), fn);
}
std::vector<std::string> GetPathList() {
return pathList_;
}
private:
std::vector<std::string> pathList_;
};
struct Data
{
std::unordered_set<std::string> exposedSymbols_;
std::unordered_set<std::string> annotatedSymbols_;
};
static const std::string categories_[] = {"class", "enum"};
static std::unordered_map<std::string, Data> categoryData_;
class ExtractCallback : public MatchFinder::MatchCallback
{
public :
virtual void run(const MatchFinder::MatchResult& result)
{
for (auto& i: categories_)
{
auto symbol = result.Nodes.getNodeAs<StringLiteral>(i);
if (symbol)
categoryData_[i].exposedSymbols_.insert(symbol->getString());
}
}
virtual void onStartOfTranslationUnit()
{
static unsigned count = sizeof("Extracting") / sizeof(char) - 1;
outs() << '.' << (++count % 100 ? "" : "\n"); // Sending a heart beat
}
};
class AnnotateCallback : public MatchFinder::MatchCallback
{
public :
AnnotateCallback(Replacements& replacements) :
replacements_(replacements)
{
}
virtual void run(const MatchFinder::MatchResult& result)
{
for (auto& i: categories_)
{
auto symbol = result.Nodes.getNodeAs<NamedDecl>(i);
if (symbol)
{
auto& data = categoryData_[i];
if (data.annotatedSymbols_.find(symbol->getName()) == data.annotatedSymbols_.end() &&
data.exposedSymbols_.find(symbol->getName()) == data.exposedSymbols_.end())
{
replacements_.insert(Replacement(*result.SourceManager, symbol->getLocation(), 0, "NONSCRIPTABLE "));
data.annotatedSymbols_.insert(symbol->getName());
}
}
}
}
virtual void onStartOfTranslationUnit()
{
static unsigned count = sizeof("Annotating") / sizeof(char) - 1;
outs() << '.' << (++count % 100 ? "" : "\n");
}
private:
Replacements& replacements_;
};
int main(int argc, const char** argv)
{
// Parse the arguments and pass them to the the internal sub-tools
CommonOptionsParser optionsParser(argc, argv, annotatorCategory);
PathFilter bindingPathFilter
(optionsParser.getSourcePathList(), [](const std::string& path) { return path.find("API.cpp") != std::string::npos; });
PathFilter nonBindingPathFilter
(optionsParser.getSourcePathList(), [](const std::string& path) { return path.find("API.cpp") == std::string::npos; });
ClangTool bindingExtractor(optionsParser.getCompilations(), bindingPathFilter.GetPathList());
RefactoringTool annotator(optionsParser.getCompilations(), nonBindingPathFilter.GetPathList());
// Setup finder to match against AST nodes from existing AngelScript binding source files
ExtractCallback extractCallback;
MatchFinder bindingFinder;
// Find exposed classes (they are registered through RegisterObjectType(), RegisterRefCounted(), RegisterObject(), etc)
bindingFinder.addMatcher(
memberCallExpr(
callee(
methodDecl(hasName("RegisterObjectType"))),
hasArgument(0, stringLiteral().bind("class"))), &extractCallback);
bindingFinder.addMatcher(
callExpr(
hasDeclaration(
functionDecl(hasParameter(1, hasName("className")))),
hasArgument(1, stringLiteral().bind("class"))), &extractCallback);
// Find exposed enums
bindingFinder.addMatcher(
memberCallExpr(
callee(
methodDecl(hasName("RegisterEnum"))),
hasArgument(0, stringLiteral().bind("enum"))), &extractCallback);
// Setup finder to match against AST nodes for annotating Urho3D library source files
AnnotateCallback annotateCallback(annotator.getReplacements());
MatchFinder annotateFinder;
// Find exported class declarations with Urho3D namespace
annotateFinder.addMatcher(
recordDecl(
unless(hasAttr(attr::Annotate)),
#ifndef _MSC_VER
hasAttr(attr::Visibility),
#else
hasAttr(attr::DLLExport),
#endif
matchesName("^::Urho3D::")).bind("class"), &annotateCallback);
// Find enum declarations with Urho3D namespace
annotateFinder.addMatcher(
enumDecl(
unless(hasAttr(attr::Annotate)),
matchesName("^::Urho3D::")).bind("enum"), &annotateCallback);
// Unbuffered stdout stream to keep the Travis-CI's log flowing and thus prevent it from killing a potentially long running job
outs().SetUnbuffered();
// Success when both sub-tools are run successfully
return (outs() << "Extracting", true) &&
bindingExtractor.run(newFrontendActionFactory(&bindingFinder).get()) == EXIT_SUCCESS &&
(outs() << "\nAnnotating", true) &&
annotator.runAndSave(newFrontendActionFactory(&annotateFinder).get()) == EXIT_SUCCESS &&
(outs() << "\n", true) ?
EXIT_SUCCESS : EXIT_FAILURE;
}