forked from qt-creator/qt-creator
		
	Added generator for the accept0 methods
Done with Erik Verbruggen
This commit is contained in:
		@@ -16,12 +16,237 @@
 | 
			
		||||
#include <Literals.h>
 | 
			
		||||
#include <CppDocument.h>
 | 
			
		||||
#include <Overview.h>
 | 
			
		||||
#include <Names.h>
 | 
			
		||||
#include <Scope.h>
 | 
			
		||||
 | 
			
		||||
#include <iostream>
 | 
			
		||||
#include <cstdlib>
 | 
			
		||||
 | 
			
		||||
using namespace CPlusPlus;
 | 
			
		||||
 | 
			
		||||
class SearchListNodes: protected ASTVisitor
 | 
			
		||||
{
 | 
			
		||||
    QList<QByteArray> _listNodes;
 | 
			
		||||
 | 
			
		||||
public:
 | 
			
		||||
    SearchListNodes(Control *control)
 | 
			
		||||
        : ASTVisitor(control)
 | 
			
		||||
    { }
 | 
			
		||||
 | 
			
		||||
    QList<QByteArray> operator()(AST *ast)
 | 
			
		||||
    {
 | 
			
		||||
        _listNodes.clear();
 | 
			
		||||
        accept(ast);
 | 
			
		||||
        return _listNodes;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
protected:
 | 
			
		||||
    virtual bool visit(ClassSpecifierAST *ast)
 | 
			
		||||
    {
 | 
			
		||||
        const QString className = oo(ast->symbol->name());
 | 
			
		||||
        
 | 
			
		||||
        if (! (className.length() > 3 && className.endsWith(QLatin1String("AST"))))
 | 
			
		||||
            return true;
 | 
			
		||||
 | 
			
		||||
        for (unsigned i = 0; i < ast->symbol->memberCount(); ++i) {
 | 
			
		||||
            Symbol *member = ast->symbol->memberAt(i);
 | 
			
		||||
            Name *memberName = member->name();
 | 
			
		||||
 | 
			
		||||
            if (! memberName)
 | 
			
		||||
                continue;
 | 
			
		||||
            else if (! memberName->identifier())
 | 
			
		||||
                continue;
 | 
			
		||||
            
 | 
			
		||||
            if (! qstrcmp("next", memberName->identifier()->chars())) {
 | 
			
		||||
                _listNodes.append(className.toUtf8());
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
private:
 | 
			
		||||
    Overview oo;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
class VisitCG: protected ASTVisitor
 | 
			
		||||
{
 | 
			
		||||
    QDir _cplusplusDir;
 | 
			
		||||
    QList<QByteArray> _listNodes;
 | 
			
		||||
    QTextStream *out;
 | 
			
		||||
 | 
			
		||||
public:
 | 
			
		||||
    VisitCG(const QDir &cplusplusDir, Control *control)
 | 
			
		||||
        : ASTVisitor(control), _cplusplusDir(cplusplusDir), out(0)
 | 
			
		||||
    { }
 | 
			
		||||
 | 
			
		||||
    void operator()(AST *ast)
 | 
			
		||||
    {
 | 
			
		||||
        QFileInfo fileInfo(_cplusplusDir, QLatin1String("ASTVisit.cpp"));
 | 
			
		||||
 | 
			
		||||
        QFile file(fileInfo.absoluteFilePath());
 | 
			
		||||
        if (! file.open(QFile::WriteOnly))
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        QTextStream output(&file);
 | 
			
		||||
        out = &output;
 | 
			
		||||
 | 
			
		||||
        *out <<
 | 
			
		||||
            "/**************************************************************************\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** This file is part of Qt Creator\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** Contact: Nokia Corporation (qt-info@nokia.com)\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** Commercial Usage\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** Licensees holding valid Qt Commercial licenses may use this file in\n"
 | 
			
		||||
            "** accordance with the Qt Commercial License Agreement provided with the\n"
 | 
			
		||||
            "** Software or, alternatively, in accordance with the terms contained in\n"
 | 
			
		||||
            "** a written agreement between you and Nokia.\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** GNU Lesser General Public License Usage\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** Alternatively, this file may be used under the terms of the GNU Lesser\n"
 | 
			
		||||
            "** General Public License version 2.1 as published by the Free Software\n"
 | 
			
		||||
            "** Foundation and appearing in the file LICENSE.LGPL included in the\n"
 | 
			
		||||
            "** packaging of this file.  Please review the following information to\n"
 | 
			
		||||
            "** ensure the GNU Lesser General Public License version 2.1 requirements\n"
 | 
			
		||||
            "** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "** If you are unsure which license is appropriate for your use, please\n"
 | 
			
		||||
            "** contact the sales department at http://qt.nokia.com/contact.\n"
 | 
			
		||||
            "**\n"
 | 
			
		||||
            "**************************************************************************/\n"
 | 
			
		||||
            "\n"
 | 
			
		||||
            "#include \"AST.h\"\n"
 | 
			
		||||
            "#include \"ASTVisitor.h\"\n"
 | 
			
		||||
            "\n"
 | 
			
		||||
            "using namespace CPlusPlus;\n" << endl;
 | 
			
		||||
 | 
			
		||||
        SearchListNodes listNodes(control());
 | 
			
		||||
        _listNodes = listNodes(ast);
 | 
			
		||||
 | 
			
		||||
        accept(ast);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
protected:
 | 
			
		||||
    using ASTVisitor::visit;
 | 
			
		||||
 | 
			
		||||
    QMap<QByteArray, ClassSpecifierAST *> classMap;
 | 
			
		||||
 | 
			
		||||
    QByteArray id_cast(NameAST *name)
 | 
			
		||||
    {
 | 
			
		||||
        if (! name)
 | 
			
		||||
            return QByteArray();
 | 
			
		||||
 | 
			
		||||
        Identifier *id = identifier(name->asSimpleName()->identifier_token);
 | 
			
		||||
 | 
			
		||||
        return QByteArray::fromRawData(id->chars(), id->size());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    void visitMembers(Class *klass)
 | 
			
		||||
    {
 | 
			
		||||
        const QByteArray className = klass->name()->identifier()->chars();
 | 
			
		||||
 | 
			
		||||
        // *out << "        // visit " << className.constData() << endl;
 | 
			
		||||
        for (unsigned i = 0; i < klass->memberCount(); ++i) {
 | 
			
		||||
            Symbol *member = klass->memberAt(i);
 | 
			
		||||
            if (! member->name())
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            Identifier *id = member->name()->identifier();
 | 
			
		||||
 | 
			
		||||
            if (! id)
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            const QByteArray memberName = QByteArray::fromRawData(id->chars(), id->size());
 | 
			
		||||
            if (member->type().isUnsigned() && memberName.endsWith("_token")) {
 | 
			
		||||
                // nothing to do. The member is a token.
 | 
			
		||||
 | 
			
		||||
            } else if (PointerType *ptrTy = member->type()->asPointerType()) {
 | 
			
		||||
 | 
			
		||||
                if (NamedType *namedTy = ptrTy->elementType()->asNamedType()) {
 | 
			
		||||
                    QByteArray typeName = namedTy->name()->identifier()->chars();
 | 
			
		||||
 | 
			
		||||
                    if (_listNodes.contains(typeName) && memberName != "next") {
 | 
			
		||||
                        *out
 | 
			
		||||
                                << "        for (" << typeName.constData() << " *it = "
 | 
			
		||||
                                << memberName.constData() << "; it; it = it->next)" << endl
 | 
			
		||||
                                << "            accept(it, visitor);" << endl;
 | 
			
		||||
 | 
			
		||||
                    } else if (typeName.endsWith("AST") && memberName != "next") {
 | 
			
		||||
                        *out << "        accept(" << memberName.constData() << ", visitor);" << endl;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        for (unsigned i = 0; i < klass->baseClassCount(); ++i) {
 | 
			
		||||
            const QByteArray baseClassName = klass->baseClassAt(i)->identifier()->chars();
 | 
			
		||||
 | 
			
		||||
            if (ClassSpecifierAST *baseClassSpec = classMap.value(baseClassName, 0)) {
 | 
			
		||||
                visitMembers(baseClassSpec->symbol);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    bool checkMethod(Symbol *accept0Method) const
 | 
			
		||||
    {
 | 
			
		||||
        Declaration *decl = accept0Method->asDeclaration();
 | 
			
		||||
        if (! decl)
 | 
			
		||||
            return false;
 | 
			
		||||
 | 
			
		||||
        Function *funTy = decl->type()->asFunctionType();
 | 
			
		||||
        if (! funTy)
 | 
			
		||||
            return false;
 | 
			
		||||
 | 
			
		||||
        else if (funTy->isPureVirtual())
 | 
			
		||||
            return false;
 | 
			
		||||
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    virtual bool visit(ClassSpecifierAST *ast)
 | 
			
		||||
    {
 | 
			
		||||
        Class *klass = ast->symbol;
 | 
			
		||||
        const QByteArray className = id_cast(ast->name);
 | 
			
		||||
 | 
			
		||||
        Identifier *visit_id = control()->findOrInsertIdentifier("accept0");
 | 
			
		||||
        Symbol *accept0Method = klass->members()->lookat(visit_id);
 | 
			
		||||
        for (; accept0Method; accept0Method = accept0Method->next()) {
 | 
			
		||||
            if (accept0Method->identifier() != visit_id)
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            if (checkMethod(accept0Method))
 | 
			
		||||
                break;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (! accept0Method)
 | 
			
		||||
            return true;
 | 
			
		||||
 | 
			
		||||
        classMap.insert(className, ast);
 | 
			
		||||
 | 
			
		||||
        *out
 | 
			
		||||
                << "void " << className.constData() << "::accept0(ASTVisitor *visitor)" << endl
 | 
			
		||||
                << "{" << endl
 | 
			
		||||
                << "    if (visitor->visit(this)) {" << endl;
 | 
			
		||||
 | 
			
		||||
        visitMembers(klass);
 | 
			
		||||
 | 
			
		||||
        *out
 | 
			
		||||
                << "    }" << endl
 | 
			
		||||
                << "    visitor->endVisit(this);" << endl
 | 
			
		||||
                << "}" << endl
 | 
			
		||||
                << endl;
 | 
			
		||||
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
QTextCursor createCursor(TranslationUnit *unit, AST *ast, QTextDocument *document)
 | 
			
		||||
{
 | 
			
		||||
    unsigned startLine, startColumn, endLine, endColumn;
 | 
			
		||||
@@ -226,6 +451,9 @@ QStringList generateAST_H(const Snapshot &snapshot, const QDir &cplusplusDir)
 | 
			
		||||
        out << document.toPlainText();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    VisitCG cg(cplusplusDir, doc->control());
 | 
			
		||||
    cg(doc->translationUnit()->ast());
 | 
			
		||||
 | 
			
		||||
    return astDerivedClasses;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user