2010-03-08 12:12:58 +01:00
|
|
|
/**************************************************************************
|
|
|
|
**
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
**
|
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
|
|
|
**
|
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
|
|
**
|
|
|
|
** Commercial Usage
|
|
|
|
**
|
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Nokia.
|
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
**
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
** General Public License version 2.1 as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
|
|
**
|
|
|
|
** If you are unsure which license is appropriate for your use, please
|
|
|
|
** contact the sales department at http://qt.nokia.com/contact.
|
|
|
|
**
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
#include <AST.h>
|
|
|
|
#include <ASTVisitor.h>
|
|
|
|
#include <ASTPatternBuilder.h>
|
|
|
|
#include <ASTMatcher.h>
|
|
|
|
#include <Control.h>
|
|
|
|
#include <Scope.h>
|
|
|
|
#include <Semantic.h>
|
|
|
|
#include <TranslationUnit.h>
|
|
|
|
#include <Literals.h>
|
|
|
|
#include <Symbols.h>
|
|
|
|
#include <Names.h>
|
|
|
|
#include <CoreTypes.h>
|
|
|
|
#include <CppDocument.h>
|
2010-03-08 16:42:14 +01:00
|
|
|
#include <SymbolVisitor.h>
|
2010-03-08 17:31:14 +01:00
|
|
|
#include <Overview.h>
|
2010-03-08 12:12:58 +01:00
|
|
|
|
|
|
|
#include <QFile>
|
|
|
|
#include <QList>
|
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QStringList>
|
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QTime>
|
|
|
|
#include <QtDebug>
|
|
|
|
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2010-03-08 16:42:14 +01:00
|
|
|
#include <fstream>
|
2010-03-08 12:12:58 +01:00
|
|
|
#include <iostream>
|
|
|
|
#include <cxxabi.h>
|
|
|
|
|
|
|
|
using namespace CPlusPlus;
|
|
|
|
|
|
|
|
class ASTDump: protected ASTVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ASTDump(TranslationUnit *unit)
|
|
|
|
: ASTVisitor(unit) {}
|
|
|
|
|
|
|
|
void operator()(AST *ast) {
|
2010-03-08 16:42:14 +01:00
|
|
|
QByteArray basename = translationUnit()->fileName();
|
|
|
|
int dotIdx = basename.lastIndexOf('.');
|
|
|
|
if (dotIdx != -1)
|
|
|
|
basename.truncate(dotIdx);
|
|
|
|
basename.append(".ast.dot");
|
|
|
|
out.open(basename.constData());
|
|
|
|
|
|
|
|
out << "digraph AST {" << std::endl;
|
2010-03-08 12:21:42 +01:00
|
|
|
// std::cout << "rankdir = \"LR\";" << std::endl;
|
2010-03-08 12:12:58 +01:00
|
|
|
accept(ast);
|
2010-03-08 16:42:14 +01:00
|
|
|
|
|
|
|
foreach (const QByteArray &terminalShape, _terminalShapes) {
|
|
|
|
out << std::string(terminalShape) << " [shape=rect]" << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
alignTerminals();
|
|
|
|
|
|
|
|
out << "}" << std::endl;
|
|
|
|
out.close();
|
|
|
|
std::cout << basename.constData() << std::endl;
|
2010-03-08 12:12:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2010-03-08 16:42:14 +01:00
|
|
|
void alignTerminals() {
|
2010-03-08 17:34:32 +01:00
|
|
|
out<<"{ rank=same;" << std::endl;
|
2010-03-08 16:42:14 +01:00
|
|
|
foreach (const QByteArray &terminalShape, _terminalShapes) {
|
2010-03-08 17:34:32 +01:00
|
|
|
out << " " << std::string(terminalShape) << ";" << std::endl;
|
2010-03-08 16:42:14 +01:00
|
|
|
}
|
|
|
|
out<<"}"<<std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray addTerminalInfo(AST *ast) const {
|
|
|
|
if (SimpleNameAST *simpleName = ast->asSimpleName()) {
|
|
|
|
return spell(simpleName->identifier_token);
|
|
|
|
} else if (BinaryExpressionAST *binExpr = ast->asBinaryExpression()) {
|
|
|
|
return spell(binExpr->binary_op_token);
|
|
|
|
} else if (SimpleSpecifierAST *simpleSpec = ast->asSimpleSpecifier()) {
|
|
|
|
return spell(simpleSpec->specifier_token);
|
|
|
|
} else if (NumericLiteralAST *numLit = ast->asNumericLiteral()) {
|
|
|
|
return spell(numLit->literal_token);
|
|
|
|
} else if (StringLiteralAST *strLit = ast->asStringLiteral()) {
|
|
|
|
return spell(strLit->literal_token);
|
|
|
|
} else if (BoolLiteralAST *boolLit = ast->asBoolLiteral()) {
|
|
|
|
return spell(boolLit->literal_token);
|
2010-03-12 11:35:09 +01:00
|
|
|
} else if (ObjCSelectorArgumentAST *selArg = ast->asObjCSelectorArgument()) {
|
|
|
|
QByteArray t = spell(selArg->name_token);
|
|
|
|
if (selArg->colon_token)
|
|
|
|
t.append(spell(selArg->colon_token));
|
|
|
|
return t;
|
|
|
|
} else if (AttributeAST *attr = ast->asAttribute()) {
|
|
|
|
return spell(attr->identifier_token);
|
2010-03-08 16:42:14 +01:00
|
|
|
} else {
|
|
|
|
return QByteArray();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string name(AST *ast) {
|
2010-03-08 12:12:58 +01:00
|
|
|
QByteArray name = abi::__cxa_demangle(typeid(*ast).name(), 0, 0, 0) + 11;
|
|
|
|
name.truncate(name.length() - 3);
|
|
|
|
name = QByteArray::number(_id.value(ast)) + ". " + name;
|
2010-03-08 12:21:42 +01:00
|
|
|
|
2010-03-08 16:42:14 +01:00
|
|
|
QByteArray info = addTerminalInfo(ast);
|
|
|
|
if (!info.isEmpty()) {
|
|
|
|
name.append("\\n");
|
|
|
|
name.append(info);
|
2010-03-08 12:21:42 +01:00
|
|
|
}
|
2010-03-08 12:12:58 +01:00
|
|
|
name.prepend('"');
|
2010-03-08 16:42:14 +01:00
|
|
|
name.append('"');
|
|
|
|
|
|
|
|
if (!info.isEmpty())
|
|
|
|
_terminalShapes.insert(name);
|
2010-03-08 12:12:58 +01:00
|
|
|
|
|
|
|
return std::string(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool preVisit(AST *ast) {
|
|
|
|
static int count = 1;
|
|
|
|
_id[ast] = count++;
|
|
|
|
|
|
|
|
if (! _stack.isEmpty())
|
2010-03-08 16:42:14 +01:00
|
|
|
out << name(_stack.last()) << " -> " << name(ast) << ";" << std::endl;
|
2010-03-08 12:12:58 +01:00
|
|
|
|
|
|
|
_stack.append(ast);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-03-08 12:21:42 +01:00
|
|
|
virtual void postVisit(AST *) {
|
2010-03-08 16:42:14 +01:00
|
|
|
_stack.removeLast();
|
2010-03-08 12:12:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
QHash<AST *, int> _id;
|
|
|
|
QList<AST *> _stack;
|
2010-03-08 16:42:14 +01:00
|
|
|
QSet<QByteArray> _terminalShapes;
|
|
|
|
std::ofstream out;
|
|
|
|
};
|
|
|
|
|
|
|
|
class SymbolDump: protected SymbolVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SymbolDump(TranslationUnit *unit)
|
|
|
|
: translationUnit(unit)
|
2010-03-08 17:31:14 +01:00
|
|
|
{
|
|
|
|
o.setShowArgumentNames(true);
|
|
|
|
o.setShowFunctionSignatures(true);
|
|
|
|
o.setShowReturnTypes(true);
|
|
|
|
}
|
2010-03-08 16:42:14 +01:00
|
|
|
|
|
|
|
void operator()(Symbol *s) {
|
|
|
|
QByteArray basename = translationUnit->fileName();
|
|
|
|
int dotIdx = basename.lastIndexOf('.');
|
|
|
|
if (dotIdx != -1)
|
|
|
|
basename.truncate(dotIdx);
|
|
|
|
basename.append(".symbols.dot");
|
|
|
|
out.open(basename.constData());
|
|
|
|
|
|
|
|
out << "digraph Symbols {" << std::endl;
|
|
|
|
// std::cout << "rankdir = \"LR\";" << std::endl;
|
|
|
|
accept(s);
|
|
|
|
|
|
|
|
for (int i = 0; i < _connections.size(); ++i) {
|
|
|
|
QPair<Symbol*,Symbol*> connection = _connections.at(i);
|
|
|
|
QByteArray from = _id.value(connection.first);
|
|
|
|
if (from.isEmpty())
|
|
|
|
from = name(connection.first);
|
|
|
|
QByteArray to = _id.value(connection.second);
|
|
|
|
if (to.isEmpty())
|
|
|
|
to = name(connection.second);
|
|
|
|
out << from.constData() << " -> " << to.constData() << ";" << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
out << "}" << std::endl;
|
|
|
|
out.close();
|
|
|
|
std::cout << basename.constData() << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
QByteArray name(Symbol *s) {
|
|
|
|
QByteArray name = abi::__cxa_demangle(typeid(*s).name(), 0, 0, 0) + 11;
|
|
|
|
if (s->identifier()) {
|
|
|
|
name.append("\\nid: ");
|
|
|
|
name.append(s->identifier()->chars());
|
|
|
|
}
|
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool preVisit(Symbol *s) {
|
|
|
|
static int count = 0;
|
|
|
|
QByteArray nodeId("s");
|
|
|
|
nodeId.append(QByteArray::number(++count));
|
|
|
|
_id[s] = nodeId;
|
|
|
|
|
|
|
|
if (!_stack.isEmpty())
|
|
|
|
_connections.append(qMakePair(_stack.last(), s));
|
|
|
|
|
|
|
|
_stack.append(s);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void postVisit(Symbol *) {
|
|
|
|
_stack.removeLast();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void simpleNode(Symbol *symbol) {
|
|
|
|
out << _id[symbol].constData() << " [label=\"" << name(symbol).constData() << "\"];" << std::endl;
|
|
|
|
}
|
|
|
|
|
2010-03-08 17:31:14 +01:00
|
|
|
void generateTemplateParams(TemplateParameters *params) {
|
|
|
|
if (!params || params->scope()->symbolCount() == 0)
|
2010-03-08 16:42:14 +01:00
|
|
|
return;
|
|
|
|
|
2010-03-08 17:31:14 +01:00
|
|
|
out << "<";
|
2010-03-08 16:42:14 +01:00
|
|
|
for (unsigned i = 0; i < params->scope()->symbolCount(); ++i) {
|
|
|
|
if (i > 0)
|
2010-03-08 17:31:14 +01:00
|
|
|
out << ",";
|
2010-03-08 16:42:14 +01:00
|
|
|
Symbol *s = params->scope()->symbolAt(i);
|
2010-03-08 17:31:14 +01:00
|
|
|
out << qPrintable(o(s->name()));
|
|
|
|
if (s->type())
|
|
|
|
out << ":" << qPrintable(o(s->type()));
|
2010-03-08 16:42:14 +01:00
|
|
|
}
|
2010-03-08 17:31:14 +01:00
|
|
|
out << ">";
|
2010-03-08 16:42:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool visit(Class *symbol) {
|
|
|
|
const char *id = _id.value(symbol).constData();
|
|
|
|
out << id << " [label=\"";
|
|
|
|
if (symbol->isClass()) {
|
|
|
|
out << "class";
|
|
|
|
} else if (symbol->isStruct()) {
|
|
|
|
out << "struct";
|
|
|
|
} else if (symbol->isUnion()) {
|
|
|
|
out << "union";
|
|
|
|
} else {
|
|
|
|
out << "UNKNOWN";
|
|
|
|
}
|
2010-03-08 17:31:14 +01:00
|
|
|
generateTemplateParams(symbol->templateParameters());
|
|
|
|
|
2010-03-08 16:42:14 +01:00
|
|
|
out << "\\nid: ";
|
|
|
|
if (symbol->identifier()) {
|
|
|
|
out << symbol->identifier()->chars();
|
|
|
|
} else {
|
|
|
|
out << "NO ID";
|
|
|
|
}
|
|
|
|
out << "\"];" << std::endl;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool visit(UsingNamespaceDirective *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(UsingDeclaration *symbol) { simpleNode(symbol); return true; }
|
2010-03-08 17:31:14 +01:00
|
|
|
|
|
|
|
virtual bool visit(Declaration *symbol) {
|
|
|
|
out << _id[symbol].constData() << " [label=\"";
|
|
|
|
out << "Declaration\\n";
|
|
|
|
out << qPrintable(o(symbol->name()));
|
|
|
|
out << ": ";
|
|
|
|
out << qPrintable(o(symbol->type()));
|
|
|
|
out << "\"];" << std::endl;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-03-08 16:42:14 +01:00
|
|
|
virtual bool visit(Argument *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(TypenameArgument *symbol) { simpleNode(symbol); return true; }
|
2010-03-08 17:31:14 +01:00
|
|
|
|
|
|
|
virtual bool visit(BaseClass *symbol) {
|
|
|
|
out << _id[symbol].constData() << " [label=\"BaseClass\\n";
|
|
|
|
out << qPrintable(o(symbol->name()));
|
|
|
|
out << "\"];" << std::endl;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-03-08 16:42:14 +01:00
|
|
|
virtual bool visit(Enum *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(Function *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(Namespace *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(Block *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ForwardClassDeclaration *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCBaseClass *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCBaseProtocol *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCClass *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCForwardClassDeclaration *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCProtocol *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCForwardProtocolDeclaration *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCMethod *symbol) { simpleNode(symbol); return true; }
|
|
|
|
virtual bool visit(ObjCPropertyDeclaration *symbol) { simpleNode(symbol); return true; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
TranslationUnit *translationUnit;
|
|
|
|
QHash<Symbol *, QByteArray> _id;
|
|
|
|
QList<QPair<Symbol *,Symbol*> >_connections;
|
|
|
|
QList<Symbol *> _stack;
|
|
|
|
std::ofstream out;
|
2010-03-08 17:31:14 +01:00
|
|
|
Overview o;
|
2010-03-08 12:12:58 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
QCoreApplication app(argc, argv);
|
|
|
|
|
|
|
|
QStringList files = app.arguments();
|
|
|
|
files.removeFirst();
|
|
|
|
|
|
|
|
foreach (const QString &fileName, files) {
|
|
|
|
QFile file(fileName);
|
2010-03-08 16:42:14 +01:00
|
|
|
if (! file.open(QFile::ReadOnly)) {
|
|
|
|
std::cerr << "Cannot open \"" << qPrintable(fileName)
|
|
|
|
<< "\", skipping it." << std::endl;
|
2010-03-08 12:12:58 +01:00
|
|
|
continue;
|
2010-03-08 16:42:14 +01:00
|
|
|
}
|
2010-03-08 12:12:58 +01:00
|
|
|
|
|
|
|
const QByteArray source = file.readAll();
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
Document::Ptr doc = Document::create(fileName);
|
|
|
|
doc->control()->setDiagnosticClient(0);
|
|
|
|
doc->setSource(source);
|
|
|
|
doc->parse();
|
|
|
|
|
2010-03-08 16:42:14 +01:00
|
|
|
doc->check();
|
|
|
|
|
2010-03-08 12:12:58 +01:00
|
|
|
ASTDump dump(doc->translationUnit());
|
|
|
|
dump(doc->translationUnit()->ast());
|
2010-03-08 16:42:14 +01:00
|
|
|
|
|
|
|
SymbolDump dump2(doc->translationUnit());
|
|
|
|
dump2(doc->globalNamespace());
|
2010-03-08 12:12:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|