/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** ** GNU Lesser General Public License Usage ** ** 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. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** Other Usage ** ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include "cpplocalsymbols.h" #include "cppsemanticinfo.h" #include #include #include #include #include #include #include #include using namespace CPlusPlus; using namespace CppEditor::Internal; namespace { class FindLocalSymbols: protected ASTVisitor { Scope *_functionScope; Document::Ptr _doc; public: FindLocalSymbols(Document::Ptr doc) : ASTVisitor(doc->translationUnit()), _doc(doc), hasD(false), hasQ(false) { } // local and external uses. SemanticInfo::LocalUseMap localUses; bool hasD; bool hasQ; void operator()(DeclarationAST *ast) { localUses.clear(); if (!ast) return; if (FunctionDefinitionAST *def = ast->asFunctionDefinition()) { if (def->symbol) { _functionScope = def->symbol; accept(ast); } } else if (ObjCMethodDeclarationAST *decl = ast->asObjCMethodDeclaration()) { if (decl->method_prototype->symbol) { _functionScope = decl->method_prototype->symbol; accept(ast); } } } protected: using ASTVisitor::visit; using ASTVisitor::endVisit; void enterScope(Scope *scope) { _scopeStack.append(scope); for (unsigned i = 0; i < scope->memberCount(); ++i) { if (Symbol *member = scope->memberAt(i)) { if (member->isTypedef()) continue; else if (! member->isGenerated() && (member->isDeclaration() || member->isArgument())) { if (member->name() && member->name()->isNameId()) { const Identifier *id = member->identifier(); unsigned line, column; getTokenStartPosition(member->sourceLocation(), &line, &column); localUses[member].append(SemanticInfo::Use(line, column, id->size(), SemanticInfo::LocalUse)); } } } } } bool checkLocalUse(NameAST *nameAst, unsigned firstToken) { if (SimpleNameAST *simpleName = nameAst->asSimpleName()) { const Identifier *id = identifier(simpleName->identifier_token); for (int i = _scopeStack.size() - 1; i != -1; --i) { if (Symbol *member = _scopeStack.at(i)->find(id)) { if (member->isTypedef() || !(member->isDeclaration() || member->isArgument())) continue; else if (!member->isGenerated() && (member->sourceLocation() < firstToken || member->enclosingScope()->isFunction())) { unsigned line, column; getTokenStartPosition(simpleName->identifier_token, &line, &column); localUses[member].append(SemanticInfo::Use(line, column, id->size(), SemanticInfo::LocalUse)); return false; } } } } return true; } virtual bool visit(IdExpressionAST *ast) { return checkLocalUse(ast->name, ast->firstToken()); } virtual bool visit(SizeofExpressionAST *ast) { if (ast->expression && ast->expression->asTypeId()) { TypeIdAST *typeId = ast->expression->asTypeId(); if (!typeId->declarator && typeId->type_specifier_list && !typeId->type_specifier_list->next) { if (NamedTypeSpecifierAST *namedTypeSpec = typeId->type_specifier_list->value->asNamedTypeSpecifier()) { if (checkLocalUse(namedTypeSpec->name, namedTypeSpec->firstToken())) return false; } } } return true; } virtual bool visit(CastExpressionAST *ast) { if (ast->expression && ast->expression->asUnaryExpression()) { TypeIdAST *typeId = ast->type_id->asTypeId(); if (typeId && !typeId->declarator && typeId->type_specifier_list && !typeId->type_specifier_list->next) { if (NamedTypeSpecifierAST *namedTypeSpec = typeId->type_specifier_list->value->asNamedTypeSpecifier()) { if (checkLocalUse(namedTypeSpec->name, namedTypeSpec->firstToken())) { accept(ast->expression); return false; } } } } return true; } virtual bool visit(QtMemberDeclarationAST *ast) { if (tokenKind(ast->q_token) == T_Q_D) hasD = true; else hasQ = true; return true; } virtual bool visit(FunctionDefinitionAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(FunctionDefinitionAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(CompoundStatementAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(CompoundStatementAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(IfStatementAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(IfStatementAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(WhileStatementAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(WhileStatementAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(ForStatementAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(ForStatementAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(ForeachStatementAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(ForeachStatementAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(SwitchStatementAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(SwitchStatementAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(CatchClauseAST *ast) { if (ast->symbol) enterScope(ast->symbol); return true; } virtual void endVisit(CatchClauseAST *ast) { if (ast->symbol) _scopeStack.removeLast(); } virtual bool visit(ExpressionOrDeclarationStatementAST *ast) { accept(ast->declaration); return false; } private: QList _scopeStack; }; } // end of anonymous namespace LocalSymbols::LocalSymbols(CPlusPlus::Document::Ptr doc, CPlusPlus::DeclarationAST *ast) { FindLocalSymbols findLocalSymbols(doc); findLocalSymbols(ast); hasD = findLocalSymbols.hasD; hasQ = findLocalSymbols.hasQ; uses = findLocalSymbols.localUses; }