forked from qt-creator/qt-creator
		
	This means at the moment the PrettyPrinter isn't doing anything useful, but the idea is that from here we can improve it to adapt the code to a certain style. Reviewed-by: Roberto Raggi
		
			
				
	
	
		
			279 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			279 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/**************************************************************************
 | 
						|
**
 | 
						|
** This file is part of Qt Creator
 | 
						|
**
 | 
						|
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
 | 
						|
**
 | 
						|
** Contact:  Qt Software Information (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 qt-sales@nokia.com.
 | 
						|
**
 | 
						|
**************************************************************************/
 | 
						|
 | 
						|
#include <AST.h>
 | 
						|
#include <ASTVisitor.h>
 | 
						|
#include <Control.h>
 | 
						|
#include <Scope.h>
 | 
						|
#include <Semantic.h>
 | 
						|
#include <TranslationUnit.h>
 | 
						|
#include <PrettyPrinter.h>
 | 
						|
 | 
						|
#include <QFile>
 | 
						|
#include <QList>
 | 
						|
#include <QCoreApplication>
 | 
						|
#include <QStringList>
 | 
						|
#include <QFileInfo>
 | 
						|
#include <QtDebug>
 | 
						|
 | 
						|
#include <cstdio>
 | 
						|
#include <cstdlib>
 | 
						|
#include <iostream>
 | 
						|
#include <sstream>
 | 
						|
 | 
						|
class Rewrite
 | 
						|
{
 | 
						|
    QMultiMap<unsigned, QByteArray> _insertBefore;
 | 
						|
    QMultiMap<unsigned, QByteArray> _insertAfter;
 | 
						|
    QSet<unsigned> _removed;
 | 
						|
 | 
						|
public:
 | 
						|
    void remove(unsigned index)
 | 
						|
    { remove(index, index + 1); }
 | 
						|
 | 
						|
    void remove(unsigned first, unsigned last)
 | 
						|
    {
 | 
						|
        Q_ASSERT(first < last);
 | 
						|
 | 
						|
        for (; first != last; ++first)
 | 
						|
            _removed.insert(first);
 | 
						|
    }
 | 
						|
 | 
						|
    void insertTextBefore(unsigned index, const QByteArray &text)
 | 
						|
    { _insertBefore.insert(index, text); }
 | 
						|
 | 
						|
    void insertTextAfter(unsigned index, const QByteArray &text)
 | 
						|
    { _insertAfter.insert(index, text); }
 | 
						|
 | 
						|
    void rewrite(const TranslationUnit *unit,
 | 
						|
                 const QByteArray &contents,
 | 
						|
                 QByteArray *out)
 | 
						|
    {
 | 
						|
        _source = contents;
 | 
						|
        const char *source = contents.constData();
 | 
						|
        unsigned previousTokenEndPosition = 0;
 | 
						|
        for (unsigned i = 0; i < unit->tokenCount(); ++i) {
 | 
						|
            const Token &tk = unit->tokenAt(i);
 | 
						|
 | 
						|
            if (previousTokenEndPosition != tk.begin()) {
 | 
						|
                Q_ASSERT(previousTokenEndPosition < tk.begin());
 | 
						|
                out->append(source + previousTokenEndPosition,
 | 
						|
                            tk.begin() - previousTokenEndPosition);
 | 
						|
            }
 | 
						|
 | 
						|
            QMultiMap<unsigned, QByteArray>::const_iterator it;
 | 
						|
 | 
						|
            it = _insertBefore.constFind(i);
 | 
						|
            for (; it != _insertBefore.constEnd() && it.key() == i; ++it) {
 | 
						|
                out->append(it.value());
 | 
						|
            }
 | 
						|
 | 
						|
            if (! _removed.contains(i))
 | 
						|
                out->append(source + tk.begin(), tk.length);
 | 
						|
 | 
						|
            it = _insertAfter.constFind(i);
 | 
						|
            for (; it != _insertAfter.constEnd() && it.key() == i; ++it) {
 | 
						|
                out->append(it.value());
 | 
						|
            }
 | 
						|
 | 
						|
            previousTokenEndPosition = tk.end();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
protected:
 | 
						|
    QByteArray _source;
 | 
						|
};
 | 
						|
 | 
						|
class SimpleRefactor: protected ASTVisitor, Rewrite {
 | 
						|
public:
 | 
						|
    SimpleRefactor(Control *control)
 | 
						|
        : ASTVisitor(control)
 | 
						|
    { }
 | 
						|
 | 
						|
    void operator()(const TranslationUnit *unit,
 | 
						|
                    const QByteArray &source,
 | 
						|
                    QByteArray *out)
 | 
						|
    {
 | 
						|
        accept(unit->ast());
 | 
						|
        rewrite(unit, source, out);
 | 
						|
    }
 | 
						|
 | 
						|
protected:
 | 
						|
    bool isEnumOrTypedefEnum(SpecifierAST *spec) {
 | 
						|
        if (! spec)
 | 
						|
            return false;
 | 
						|
        if (SimpleSpecifierAST *simpleSpec = spec->asSimpleSpecifier()) {
 | 
						|
            if (tokenKind(simpleSpec->specifier_token) == T_TYPEDEF)
 | 
						|
                return isEnumOrTypedefEnum(spec->next);
 | 
						|
        }
 | 
						|
        return spec->asEnumSpecifier() != 0;
 | 
						|
    }
 | 
						|
    virtual bool visit(SimpleDeclarationAST *ast) {
 | 
						|
        if (isEnumOrTypedefEnum(ast->decl_specifier_seq)) {
 | 
						|
            //remove(ast->firstToken(), ast->lastToken());
 | 
						|
            insertTextBefore(ast->firstToken(), "/* #REF# removed ");
 | 
						|
            insertTextAfter(ast->lastToken() - 1, "*/");
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    virtual bool visit(AccessDeclarationAST *ast)
 | 
						|
    {
 | 
						|
        if (tokenKind(ast->access_specifier_token) == T_PRIVATE) {
 | 
						|
            // change visibility from `private' to `public'.
 | 
						|
            remove(ast->access_specifier_token);
 | 
						|
            insertTextAfter(ast->access_specifier_token, "public /* #REF# private->public */");
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    virtual bool visit(FunctionDefinitionAST *ast)
 | 
						|
    {
 | 
						|
        bool isInline = false;
 | 
						|
        for (SpecifierAST *spec = ast->decl_specifier_seq; spec; spec = spec->next) {
 | 
						|
            if (SimpleSpecifierAST *simpleSpec = spec->asSimpleSpecifier()) {
 | 
						|
                if (tokenKind(simpleSpec->specifier_token) == T_INLINE) {
 | 
						|
                    isInline = true;
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // force the `inline' specifier.
 | 
						|
        if (! isInline)
 | 
						|
            insertTextBefore(ast->firstToken(), "inline /* #REF# made inline */ ");
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    virtual bool visit(ClassSpecifierAST *ast)
 | 
						|
    {
 | 
						|
        // export/import the class using the macro MY_EXPORT.
 | 
						|
        if (ast->name)
 | 
						|
            insertTextBefore(ast->name->firstToken(), "MY_EXPORT ");
 | 
						|
 | 
						|
        // add QObject to the base clause.
 | 
						|
        if (ast->colon_token)
 | 
						|
            insertTextAfter(ast->colon_token, " public QObject,");
 | 
						|
        else if (ast->lbrace_token)
 | 
						|
            insertTextBefore(ast->lbrace_token, ": public QObject ");
 | 
						|
 | 
						|
        // mark the class as Q_OBJECT.
 | 
						|
        if (ast->lbrace_token)
 | 
						|
            insertTextAfter(ast->lbrace_token, " Q_OBJECT\n");
 | 
						|
 | 
						|
        for (DeclarationAST *it = ast->member_specifiers; it; it = it->next) {
 | 
						|
            accept(it);
 | 
						|
        }
 | 
						|
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    virtual bool visit(CppCastExpressionAST *ast)
 | 
						|
    {
 | 
						|
        // Replace the C++ cast expression (e.g. static_cast<foo>(a)) with
 | 
						|
        // the one generated by the pretty printer.
 | 
						|
        std::ostringstream o;
 | 
						|
        PrettyPrinter pp(control(), o);
 | 
						|
        pp(ast, _source);
 | 
						|
        remove(ast->firstToken(), ast->lastToken());
 | 
						|
        const std::string str = o.str();
 | 
						|
        insertTextBefore(ast->firstToken(), str.c_str());
 | 
						|
        insertTextBefore(ast->firstToken(), "/* #REF# beautiful cast */ ");
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
int main(int argc, char *argv[])
 | 
						|
{
 | 
						|
    QCoreApplication app(argc, argv);
 | 
						|
 | 
						|
    QStringList args = app.arguments();
 | 
						|
    const QString appName = args.first();
 | 
						|
    args.removeFirst();
 | 
						|
 | 
						|
    bool test_rewriter = false;
 | 
						|
    bool test_pretty_printer = false;
 | 
						|
 | 
						|
    foreach (QString arg, args) {
 | 
						|
        if (arg == QLatin1String("--test-rewriter"))
 | 
						|
            test_rewriter = true;
 | 
						|
        else if (arg == QLatin1String("--test-pretty-printer"))
 | 
						|
            test_pretty_printer = true;
 | 
						|
        else if (arg == QLatin1String("--help")) {
 | 
						|
            const QFileInfo appInfo(appName);
 | 
						|
            const QByteArray appFileName = QFile::encodeName(appInfo.fileName());
 | 
						|
 | 
						|
            printf("Usage: %s [options]\n"
 | 
						|
                   "  --help                    Display this information\n"
 | 
						|
                   "  --test-rewriter           Test the tree rewriter\n"
 | 
						|
                   "  --test-pretty-printer     Test the pretty printer\n",
 | 
						|
                   appFileName.constData());
 | 
						|
            return EXIT_SUCCESS;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    QFile in;
 | 
						|
    if (! in.open(stdin, QFile::ReadOnly))
 | 
						|
        return EXIT_FAILURE;
 | 
						|
 | 
						|
    const QByteArray source = in.readAll();
 | 
						|
 | 
						|
    Control control;
 | 
						|
    StringLiteral *fileId = control.findOrInsertFileName("<stdin>");
 | 
						|
    TranslationUnit unit(&control, fileId);
 | 
						|
    unit.setObjCEnabled(true);
 | 
						|
    unit.setSource(source.constData(), source.size());
 | 
						|
    unit.parse();
 | 
						|
    if (! unit.ast())
 | 
						|
        return EXIT_FAILURE;
 | 
						|
 | 
						|
    TranslationUnitAST *ast = unit.ast()->asTranslationUnit();
 | 
						|
    Q_ASSERT(ast != 0);
 | 
						|
 | 
						|
    Scope globalScope;
 | 
						|
    Semantic sem(&control);
 | 
						|
    for (DeclarationAST *decl = ast->declarations; decl; decl = decl->next) {
 | 
						|
        sem.check(decl, &globalScope);
 | 
						|
    }
 | 
						|
 | 
						|
    // test the rewriter
 | 
						|
    if (test_rewriter) {
 | 
						|
        QByteArray out;
 | 
						|
        SimpleRefactor refactor(&control);
 | 
						|
        refactor(&unit, source, &out);
 | 
						|
        printf("%s\n", out.constData());
 | 
						|
    } else if (test_pretty_printer) {
 | 
						|
        PrettyPrinter pp(&control, std::cout);
 | 
						|
        pp(unit.ast(), source);
 | 
						|
    }
 | 
						|
    return EXIT_SUCCESS;
 | 
						|
}
 |