| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2012-01-26 18:33:46 +01:00
										 |  |  | ** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-11-02 15:59:12 +01:00
										 |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** In addition, as a special exception, Nokia gives you certain additional | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** rights. These rights are described in the Nokia Qt LGPL Exception | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** 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. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** If you have questions regarding the use of this file, please contact | 
					
						
							| 
									
										
										
										
											2011-11-02 15:59:12 +01:00
										 |  |  | ** Nokia at qt-info@nokia.com. | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | #include "cppchecksymbols.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  | #include "cpplocalsymbols.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-05 17:35:54 +02:00
										 |  |  | #include <cplusplus/Overview.h>
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <Names.h>
 | 
					
						
							|  |  |  | #include <Literals.h>
 | 
					
						
							|  |  |  | #include <Symbols.h>
 | 
					
						
							|  |  |  | #include <TranslationUnit.h>
 | 
					
						
							|  |  |  | #include <Scope.h>
 | 
					
						
							|  |  |  | #include <AST.h>
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | #include <SymbolVisitor.h>
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-23 17:44:49 +01:00
										 |  |  | #include <utils/qtcassert.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | #include <QtCore/QCoreApplication>
 | 
					
						
							|  |  |  | #include <QtCore/QThreadPool>
 | 
					
						
							|  |  |  | #include <QtCore/QDebug>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 09:35:03 +01:00
										 |  |  | #include <utils/runextensions.h>
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  | using namespace CPlusPlus; | 
					
						
							| 
									
										
										
										
											2012-02-07 15:09:08 +01:00
										 |  |  | using namespace CppTools; | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-19 11:08:39 +02:00
										 |  |  | class FriendlyThread: public QThread | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using QThread::msleep; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 10:54:40 +02:00
										 |  |  | class CollectSymbols: protected SymbolVisitor | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     Document::Ptr _doc; | 
					
						
							|  |  |  |     Snapshot _snapshot; | 
					
						
							|  |  |  |     QSet<QByteArray> _types; | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     QSet<QByteArray> _members; | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |     QSet<QByteArray> _virtualMethods; | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     QSet<QByteArray> _statics; | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     bool _mainDocument; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2010-08-10 10:54:40 +02:00
										 |  |  |     CollectSymbols(Document::Ptr doc, const Snapshot &snapshot) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |         : _doc(doc), _snapshot(snapshot), _mainDocument(false) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QSet<Namespace *> processed; | 
					
						
							|  |  |  |         process(doc, &processed); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QSet<QByteArray> &types() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return _types; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     const QSet<QByteArray> &members() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return _members; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |     const QSet<QByteArray> &virtualMethods() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return _virtualMethods; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     const QSet<QByteArray> &statics() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return _statics; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | protected: | 
					
						
							|  |  |  |     void process(Document::Ptr doc, QSet<Namespace *> *processed) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (! doc) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         else if (! processed->contains(doc->globalNamespace())) { | 
					
						
							|  |  |  |             processed->insert(doc->globalNamespace()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             foreach (const Document::Include &i, doc->includes()) | 
					
						
							|  |  |  |                 process(_snapshot.document(i.fileName()), processed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             _mainDocument = (doc == _doc); // ### improve
 | 
					
						
							|  |  |  |             accept(doc->globalNamespace()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void addType(const Identifier *id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (id) | 
					
						
							|  |  |  |             _types.insert(QByteArray::fromRawData(id->chars(), id->size())); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void addType(const Name *name) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (! name) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } else if (const QualifiedNameId *q = name->asQualifiedNameId()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |             addType(q->base()); | 
					
						
							|  |  |  |             addType(q->name()); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (name->isNameId() || name->isTemplateNameId()) { | 
					
						
							|  |  |  |             addType(name->identifier()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     void addMember(const Name *name) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (! name) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } else if (name->isNameId()) { | 
					
						
							|  |  |  |             const Identifier *id = name->identifier(); | 
					
						
							|  |  |  |             _members.insert(QByteArray::fromRawData(id->chars(), id->size())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |     void addVirtualMethod(const Name *name) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (! name) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } else if (name->isNameId()) { | 
					
						
							|  |  |  |             const Identifier *id = name->identifier(); | 
					
						
							|  |  |  |             _virtualMethods.insert(QByteArray::fromRawData(id->chars(), id->size())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     void addStatic(const Name *name) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (! name) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } else if (name->isNameId() || name->isTemplateNameId()) { | 
					
						
							|  |  |  |             const Identifier *id = name->identifier(); | 
					
						
							|  |  |  |             _statics.insert(QByteArray::fromRawData(id->chars(), id->size())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     // nothing to do
 | 
					
						
							|  |  |  |     virtual bool visit(UsingNamespaceDirective *) { return true; } | 
					
						
							|  |  |  |     virtual bool visit(UsingDeclaration *) { return true; } | 
					
						
							|  |  |  |     virtual bool visit(Argument *) { return true; } | 
					
						
							|  |  |  |     virtual bool visit(BaseClass *) { return true; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(Function *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |         if (symbol->isVirtual()) | 
					
						
							|  |  |  |             addVirtualMethod(symbol->name()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     virtual bool visit(Block *) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(NamespaceAlias *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(Declaration *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         if (symbol->enclosingEnum() != 0) | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |             addStatic(symbol->name()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |         if (Function *funTy = symbol->type()->asFunctionType()) { | 
					
						
							|  |  |  |             if (funTy->isVirtual()) | 
					
						
							|  |  |  |                 addVirtualMethod(symbol->name()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |         if (symbol->isTypedef()) | 
					
						
							|  |  |  |             addType(symbol->name()); | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |         else if (! symbol->type()->isFunctionType() && symbol->enclosingScope()->isClass()) | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |             addMember(symbol->name()); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(TypenameArgument *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(Enum *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(Namespace *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 15:24:55 +02:00
										 |  |  |     virtual bool visit(Template *) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     virtual bool visit(Class *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(ForwardClassDeclaration *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Objective-C
 | 
					
						
							|  |  |  |     virtual bool visit(ObjCBaseClass *) { return true; } | 
					
						
							|  |  |  |     virtual bool visit(ObjCBaseProtocol *) { return true; } | 
					
						
							|  |  |  |     virtual bool visit(ObjCPropertyDeclaration *) { return true; } | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     virtual bool visit(ObjCMethod *) { return true; } | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(ObjCClass *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(ObjCForwardClassDeclaration *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(ObjCProtocol *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual bool visit(ObjCForwardProtocolDeclaration *symbol) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         addType(symbol->name()); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // end of anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | CheckSymbols::Future CheckSymbols::go(Document::Ptr doc, const LookupContext &context) | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-01-23 17:44:49 +01:00
										 |  |  |     QTC_ASSERT(doc, return Future()); | 
					
						
							| 
									
										
										
										
											2010-07-19 11:08:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  |     return (new CheckSymbols(doc, context))->start(); | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | CheckSymbols::CheckSymbols(Document::Ptr doc, const LookupContext &context) | 
					
						
							| 
									
										
										
										
											2010-07-07 09:42:56 +02:00
										 |  |  |     : ASTVisitor(doc->translationUnit()), _doc(doc), _context(context) | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  |     , _lineOfLastUsage(0) | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-10 10:54:40 +02:00
										 |  |  |     CollectSymbols collectTypes(doc, context.snapshot()); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     _fileName = doc->fileName(); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     _potentialTypes = collectTypes.types(); | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     _potentialMembers = collectTypes.members(); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |     _potentialVirtualMethods = collectTypes.virtualMethods(); | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     _potentialStatics = collectTypes.statics(); | 
					
						
							| 
									
										
										
										
											2010-08-03 12:22:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     typeOfExpression.init(_doc, _context.snapshot(), _context.bindings()); | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | CheckSymbols::~CheckSymbols() | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  | { } | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | void CheckSymbols::run() | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     _diagnosticMessages.clear(); | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-19 11:08:39 +02:00
										 |  |  |     if (! isCanceled()) { | 
					
						
							|  |  |  |         if (_doc->translationUnit()) { | 
					
						
							|  |  |  |             accept(_doc->translationUnit()->ast()); | 
					
						
							|  |  |  |             flush(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-07-06 12:42:55 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-19 11:08:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     reportFinished(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::warning(unsigned line, unsigned column, const QString &text, unsigned length) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     Document::DiagnosticMessage m(Document::DiagnosticMessage::Warning, _fileName, line, column, text, length); | 
					
						
							|  |  |  |     _diagnosticMessages.append(m); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::warning(AST *ast, const QString &text) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     const Token &firstToken = tokenAt(ast->firstToken()); | 
					
						
							|  |  |  |     const Token &lastToken = tokenAt(ast->lastToken() - 1); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     const unsigned length = lastToken.end() - firstToken.begin(); | 
					
						
							|  |  |  |     unsigned line = 1, column = 1; | 
					
						
							|  |  |  |     getTokenStartPosition(ast->firstToken(), &line, &column); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     warning(line, column, text, length); | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-19 14:14:19 +01:00
										 |  |  | FunctionDefinitionAST *CheckSymbols::enclosingFunctionDefinition(bool skipTopOfStack) const | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-19 14:14:19 +01:00
										 |  |  |     int index = _astStack.size() - 1; | 
					
						
							|  |  |  |     if (skipTopOfStack && !_astStack.isEmpty()) | 
					
						
							|  |  |  |         --index; | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         AST *ast = _astStack.at(index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (FunctionDefinitionAST *funDef = ast->asFunctionDefinition()) | 
					
						
							|  |  |  |             return funDef; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TemplateDeclarationAST *CheckSymbols::enclosingTemplateDeclaration() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int index = _astStack.size() - 1; index != -1; --index) { | 
					
						
							|  |  |  |         AST *ast = _astStack.at(index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (TemplateDeclarationAST *funDef = ast->asTemplateDeclaration()) | 
					
						
							|  |  |  |             return funDef; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Scope *CheckSymbols::enclosingScope() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int index = _astStack.size() - 1; index != -1; --index) { | 
					
						
							|  |  |  |         AST *ast = _astStack.at(index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (NamespaceAST *ns = ast->asNamespace()) { | 
					
						
							|  |  |  |             if (ns->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return ns->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (ClassSpecifierAST *classSpec = ast->asClassSpecifier()) { | 
					
						
							|  |  |  |             if (classSpec->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return classSpec->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (FunctionDefinitionAST *funDef = ast->asFunctionDefinition()) { | 
					
						
							|  |  |  |             if (funDef->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return funDef->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (CompoundStatementAST *blockStmt = ast->asCompoundStatement()) { | 
					
						
							|  |  |  |             if (blockStmt->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return blockStmt->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (IfStatementAST *ifStmt = ast->asIfStatement()) { | 
					
						
							|  |  |  |             if (ifStmt->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return ifStmt->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (WhileStatementAST *whileStmt = ast->asWhileStatement()) { | 
					
						
							|  |  |  |             if (whileStmt->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return whileStmt->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (ForStatementAST *forStmt = ast->asForStatement()) { | 
					
						
							|  |  |  |             if (forStmt->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return forStmt->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (ForeachStatementAST *foreachStmt = ast->asForeachStatement()) { | 
					
						
							|  |  |  |             if (foreachStmt->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return foreachStmt->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-06 11:29:41 +02:00
										 |  |  |         } else if (SwitchStatementAST *switchStmt = ast->asSwitchStatement()) { | 
					
						
							|  |  |  |             if (switchStmt->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return switchStmt->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-06 11:29:41 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (CatchClauseAST *catchClause = ast->asCatchClause()) { | 
					
						
							|  |  |  |             if (catchClause->symbol) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 return catchClause->symbol; | 
					
						
							| 
									
										
										
										
											2010-08-06 11:29:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |     return _doc->globalNamespace(); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckSymbols::preVisit(AST *ast) | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     _astStack.append(ast); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-07 09:42:56 +02:00
										 |  |  |     if (isCanceled()) | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | void CheckSymbols::postVisit(AST *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _astStack.takeLast(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(NamespaceAST *ast) | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  |     if (ast->identifier_token) { | 
					
						
							|  |  |  |         const Token &tok = tokenAt(ast->identifier_token); | 
					
						
							|  |  |  |         if (! tok.generated()) { | 
					
						
							|  |  |  |             unsigned line, column; | 
					
						
							|  |  |  |             getTokenStartPosition(ast->identifier_token, &line, &column); | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |             Use use(line, column, tok.length(), SemanticInfo::TypeUse); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |             addUse(use); | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(UsingDirectiveAST *) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | bool CheckSymbols::visit(EnumeratorAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |     addUse(ast->identifier_token, SemanticInfo::StaticUse); | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | bool CheckSymbols::visit(SimpleDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->declarator_list && !ast->declarator_list->next) { | 
					
						
							|  |  |  |         if (ast->symbols && ! ast->symbols->next && !ast->symbols->value->isGenerated()) { | 
					
						
							|  |  |  |             Symbol *decl = ast->symbols->value; | 
					
						
							|  |  |  |             if (NameAST *declId = declaratorId(ast->declarator_list->value)) { | 
					
						
							|  |  |  |                 if (Function *funTy = decl->type()->asFunctionType()) { | 
					
						
							|  |  |  |                     if (funTy->isVirtual()) { | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |                         addUse(declId, SemanticInfo::VirtualMethodUse); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |                     } else if (maybeVirtualMethod(decl->name())) { | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |                         addVirtualMethod(_context.lookup(decl->name(), decl->enclosingScope()), declId, funTy->argumentCount()); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(NamedTypeSpecifierAST *) | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-11-11 09:32:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-12 11:57:38 +02:00
										 |  |  | bool CheckSymbols::visit(ElaboratedTypeSpecifierAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     accept(ast->attribute_list); | 
					
						
							|  |  |  |     accept(ast->name); | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |     addUse(ast->name, SemanticInfo::TypeUse); | 
					
						
							| 
									
										
										
										
											2010-08-12 11:57:38 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  | bool CheckSymbols::visit(MemberAccessAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     accept(ast->base_expression); | 
					
						
							| 
									
										
										
										
											2010-08-03 12:22:16 +02:00
										 |  |  |     if (! ast->member_name) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (const Name *name = ast->member_name->name) { | 
					
						
							|  |  |  |         if (const Identifier *ident = name->identifier()) { | 
					
						
							|  |  |  |             const QByteArray id = QByteArray::fromRawData(ident->chars(), ident->size()); | 
					
						
							|  |  |  |             if (_potentialMembers.contains(id)) { | 
					
						
							|  |  |  |                 const Token start = tokenAt(ast->firstToken()); | 
					
						
							|  |  |  |                 const Token end = tokenAt(ast->lastToken() - 1); | 
					
						
							| 
									
										
										
										
											2012-01-12 17:53:56 +01:00
										 |  |  |                 const QByteArray expression = _doc->utf8Source().mid(start.begin(), end.end() - start.begin()); | 
					
						
							| 
									
										
										
										
											2010-08-03 12:22:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-11 14:26:09 +01:00
										 |  |  |                 const QList<LookupItem> candidates = | 
					
						
							| 
									
										
										
										
											2012-01-12 17:35:37 +01:00
										 |  |  |                     typeOfExpression(expression, enclosingScope(), TypeOfExpression::Preprocess); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |                 addClassMember(candidates, ast->member_name); | 
					
						
							| 
									
										
										
										
											2010-08-03 12:22:16 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | bool CheckSymbols::visit(CallAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->base_expression) { | 
					
						
							|  |  |  |         accept(ast->base_expression); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         unsigned argumentCount = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (ExpressionListAST *it = ast->expression_list; it; it = it->next) | 
					
						
							|  |  |  |             ++argumentCount; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (MemberAccessAST *access = ast->base_expression->asMemberAccess()) { | 
					
						
							|  |  |  |             if (access->member_name && access->member_name->name) { | 
					
						
							|  |  |  |                 if (maybeVirtualMethod(access->member_name->name)) { | 
					
						
							|  |  |  |                     const QByteArray expression = textOf(access); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-11 14:26:09 +01:00
										 |  |  |                     const QList<LookupItem> candidates = | 
					
						
							| 
									
										
										
										
											2012-01-12 17:35:37 +01:00
										 |  |  |                         typeOfExpression(expression, enclosingScope(), | 
					
						
							|  |  |  |                                          TypeOfExpression::Preprocess); | 
					
						
							| 
									
										
										
										
											2010-08-03 18:14:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     NameAST *memberName = access->member_name; | 
					
						
							|  |  |  |                     if (QualifiedNameAST *q = memberName->asQualifiedName()) | 
					
						
							|  |  |  |                         memberName = q->unqualified_name; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |                     addVirtualMethod(candidates, memberName, argumentCount); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else if (IdExpressionAST *idExpr = ast->base_expression->asIdExpression()) { | 
					
						
							|  |  |  |             if (const Name *name = idExpr->name->name) { | 
					
						
							|  |  |  |                 if (maybeVirtualMethod(name)) { | 
					
						
							| 
									
										
										
										
											2010-08-03 18:14:34 +02:00
										 |  |  |                     NameAST *exprName = idExpr->name; | 
					
						
							|  |  |  |                     if (QualifiedNameAST *q = exprName->asQualifiedName()) | 
					
						
							|  |  |  |                         exprName = q->unqualified_name; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-11 14:26:09 +01:00
										 |  |  |                     const QList<LookupItem> candidates = | 
					
						
							| 
									
										
										
										
											2012-01-12 17:35:37 +01:00
										 |  |  |                         typeOfExpression(textOf(idExpr), enclosingScope(), | 
					
						
							| 
									
										
										
										
											2012-01-11 14:26:09 +01:00
										 |  |  |                                          TypeOfExpression::Preprocess); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |                     addVirtualMethod(candidates, exprName, argumentCount); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         accept(ast->expression_list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QByteArray CheckSymbols::textOf(AST *ast) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const Token start = tokenAt(ast->firstToken()); | 
					
						
							|  |  |  |     const Token end = tokenAt(ast->lastToken() - 1); | 
					
						
							| 
									
										
										
										
											2012-01-12 17:53:56 +01:00
										 |  |  |     const QByteArray text = _doc->utf8Source().mid(start.begin(), end.end() - start.begin()); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |     return text; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | void CheckSymbols::checkNamespace(NameAST *name) | 
					
						
							| 
									
										
										
										
											2009-11-11 09:32:05 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     if (! name) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-02-06 14:32:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     unsigned line, column; | 
					
						
							|  |  |  |     getTokenStartPosition(name->firstToken(), &line, &column); | 
					
						
							| 
									
										
										
										
											2010-02-06 14:32:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (ClassOrNamespace *b = _context.lookupType(name->name, enclosingScope())) { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |         foreach (Symbol *s, b->symbols()) { | 
					
						
							|  |  |  |             if (s->isNamespace()) | 
					
						
							|  |  |  |                 return; | 
					
						
							| 
									
										
										
										
											2010-02-06 14:32:25 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:35 +02:00
										 |  |  |     const unsigned length = tokenAt(name->lastToken() - 1).end() - tokenAt(name->firstToken()).begin(); | 
					
						
							|  |  |  |     warning(line, column, QCoreApplication::translate("CheckUndefinedSymbols", "Expected a namespace-name"), length); | 
					
						
							| 
									
										
										
										
											2010-02-06 14:32:25 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 17:06:00 +02:00
										 |  |  | bool CheckSymbols::hasVirtualDestructor(Class *klass) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! klass) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     const Identifier *id = klass->identifier(); | 
					
						
							|  |  |  |     if (! id) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |     for (Symbol *s = klass->find(id); s; s = s->next()) { | 
					
						
							| 
									
										
										
										
											2010-08-09 17:06:00 +02:00
										 |  |  |         if (! s->name()) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         else if (s->name()->isDestructorNameId()) { | 
					
						
							|  |  |  |             if (Function *funTy = s->type()->asFunctionType()) { | 
					
						
							|  |  |  |                 if (funTy->isVirtual() && id->isEqualTo(s->identifier())) | 
					
						
							|  |  |  |                     return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckSymbols::hasVirtualDestructor(ClassOrNamespace *binding) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QSet<ClassOrNamespace *> processed; | 
					
						
							|  |  |  |     QList<ClassOrNamespace *> todo; | 
					
						
							|  |  |  |     todo.append(binding); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (! todo.isEmpty()) { | 
					
						
							|  |  |  |         ClassOrNamespace *b = todo.takeFirst(); | 
					
						
							|  |  |  |         if (b && ! processed.contains(b)) { | 
					
						
							|  |  |  |             processed.insert(b); | 
					
						
							|  |  |  |             foreach (Symbol *s, b->symbols()) { | 
					
						
							|  |  |  |                 if (Class *k = s->asClass()) { | 
					
						
							|  |  |  |                     if (hasVirtualDestructor(k)) | 
					
						
							|  |  |  |                         return true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             todo += b->usings(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 13:01:24 +02:00
										 |  |  | void CheckSymbols::checkName(NameAST *ast, Scope *scope) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-09 13:45:12 +02:00
										 |  |  |     if (ast && ast->name) { | 
					
						
							| 
									
										
										
										
											2010-08-03 13:01:24 +02:00
										 |  |  |         if (! scope) | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |             scope = enclosingScope(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-19 17:19:03 +01:00
										 |  |  |         if (ast->asDestructorName() != 0) { | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |             Class *klass = scope->asClass(); | 
					
						
							| 
									
										
										
										
											2010-08-09 17:06:00 +02:00
										 |  |  |             if (hasVirtualDestructor(_context.lookupType(klass))) | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |                 addUse(ast, SemanticInfo::VirtualMethodUse); | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |         } else if (maybeType(ast->name) || maybeStatic(ast->name)) { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |             const QList<LookupItem> candidates = _context.lookup(ast->name, scope); | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |             addTypeOrStatic(candidates, ast); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         } else if (maybeMember(ast->name)) { | 
					
						
							|  |  |  |             const QList<LookupItem> candidates = _context.lookup(ast->name, scope); | 
					
						
							|  |  |  |             addClassMember(candidates, ast); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(SimpleNameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     checkName(ast); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(TemplateIdAST *ast) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  |     checkName(ast); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(DestructorNameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |     checkName(ast); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(QualifiedNameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (ast->name) { | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |         ClassOrNamespace *binding = 0; | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |         if (NestedNameSpecifierListAST *it = ast->nested_name_specifier_list) { | 
					
						
							|  |  |  |             NestedNameSpecifierAST *nested_name_specifier = it->value; | 
					
						
							| 
									
										
										
										
											2010-06-04 10:48:24 +02:00
										 |  |  |             if (NameAST *class_or_namespace_name = nested_name_specifier->class_or_namespace_name) { // ### remove shadowing
 | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |                 if (TemplateIdAST *template_id = class_or_namespace_name->asTemplateId()) | 
					
						
							|  |  |  |                     accept(template_id->template_argument_list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-04 10:48:24 +02:00
										 |  |  |                 const Name *name = class_or_namespace_name->name; | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                 binding = _context.lookupType(name, enclosingScope()); | 
					
						
							|  |  |  |                 addType(binding, class_or_namespace_name); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                 for (it = it->next; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2010-06-04 10:48:24 +02:00
										 |  |  |                     NestedNameSpecifierAST *nested_name_specifier = it->value; | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-04 10:48:24 +02:00
										 |  |  |                     if (NameAST *class_or_namespace_name = nested_name_specifier->class_or_namespace_name) { | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                         if (TemplateIdAST *template_id = class_or_namespace_name->asTemplateId()) { | 
					
						
							|  |  |  |                             if (template_id->template_token) { | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |                                 addUse(template_id, SemanticInfo::TypeUse); | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                                 binding = 0; // there's no way we can find a binding.
 | 
					
						
							|  |  |  |                             } | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                             accept(template_id->template_argument_list); | 
					
						
							|  |  |  |                             if (! binding) | 
					
						
							|  |  |  |                                 continue; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (binding) { | 
					
						
							|  |  |  |                             binding = binding->findType(class_or_namespace_name->name); | 
					
						
							|  |  |  |                             addType(binding, class_or_namespace_name); | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2010-06-04 10:48:24 +02:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |         if (binding && ast->unqualified_name) { | 
					
						
							| 
									
										
										
										
											2010-08-09 17:06:00 +02:00
										 |  |  |             if (ast->unqualified_name->asDestructorName() != 0) { | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                 if (hasVirtualDestructor(binding)) | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |                     addUse(ast->unqualified_name, SemanticInfo::VirtualMethodUse); | 
					
						
							| 
									
										
										
										
											2010-08-09 17:06:00 +02:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |                 addTypeOrStatic(binding->find(ast->unqualified_name->name), ast->unqualified_name); | 
					
						
							| 
									
										
										
										
											2010-08-09 17:06:00 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-09-09 16:06:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (TemplateIdAST *template_id = ast->unqualified_name->asTemplateId()) | 
					
						
							|  |  |  |                 accept(template_id->template_argument_list); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(TypenameTypeParameterAST *ast) | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |     addUse(ast->name, SemanticInfo::TypeUse); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     accept(ast->type_id); | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | bool CheckSymbols::visit(TemplateTypeParameterAST *ast) | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     accept(ast->template_parameter_list); | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |     addUse(ast->name, SemanticInfo::TypeUse); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     accept(ast->type_id); | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 13:01:24 +02:00
										 |  |  | bool CheckSymbols::visit(MemInitializerAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (FunctionDefinitionAST *enclosingFunction = enclosingFunctionDefinition()) { | 
					
						
							|  |  |  |         if (ast->name && enclosingFunction->symbol) { | 
					
						
							|  |  |  |             if (ClassOrNamespace *binding = _context.lookupType(enclosingFunction->symbol)) { | 
					
						
							|  |  |  |                 foreach (Symbol *s, binding->symbols()) { | 
					
						
							|  |  |  |                     if (Class *klass = s->asClass()){ | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                         checkName(ast->name, klass); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |                         break; | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2010-08-03 13:01:24 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         accept(ast->expression_list); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-08-03 13:01:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-07 15:59:08 +01:00
										 |  |  | bool CheckSymbols::visit(GotoStatementAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->identifier_token) | 
					
						
							|  |  |  |         addUse(ast->identifier_token, SemanticInfo::LabelUse); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckSymbols::visit(LabeledStatementAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->label_token) | 
					
						
							|  |  |  |         addUse(ast->label_token, SemanticInfo::LabelUse); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     accept(ast->statement); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  | bool CheckSymbols::visit(FunctionDefinitionAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:23:55 +02:00
										 |  |  |     AST *thisFunction = _astStack.takeLast(); | 
					
						
							| 
									
										
										
										
											2010-07-15 16:27:29 +02:00
										 |  |  |     accept(ast->decl_specifier_list); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:23:55 +02:00
										 |  |  |     _astStack.append(thisFunction); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 18:08:13 +02:00
										 |  |  |     if (ast->declarator && ast->symbol && ! ast->symbol->isGenerated()) { | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |         Function *fun = ast->symbol; | 
					
						
							|  |  |  |         if (NameAST *declId = declaratorId(ast->declarator)) { | 
					
						
							|  |  |  |             if (QualifiedNameAST *q = declId->asQualifiedName()) | 
					
						
							|  |  |  |                 declId = q->unqualified_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (fun->isVirtual()) { | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |                 addUse(declId, SemanticInfo::VirtualMethodUse); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |             } else if (maybeVirtualMethod(fun->name())) { | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |                 addVirtualMethod(_context.lookup(fun->name(), fun->enclosingScope()), declId, fun->argumentCount()); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:27:29 +02:00
										 |  |  |     accept(ast->declarator); | 
					
						
							|  |  |  |     accept(ast->ctor_initializer); | 
					
						
							|  |  |  |     accept(ast->function_body); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     const LocalSymbols locals(_doc, ast); | 
					
						
							| 
									
										
										
										
											2011-04-19 15:42:14 +02:00
										 |  |  |     foreach (const QList<SemanticInfo::Use> &uses, locals.uses) { | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |         foreach (const SemanticInfo::Use &u, uses) | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |             addUse(u); | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-19 14:14:19 +01:00
										 |  |  |     if (!enclosingFunctionDefinition(true)) | 
					
						
							|  |  |  |         flush(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  | void CheckSymbols::addUse(NameAST *ast, UseKind kind) | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! ast) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (QualifiedNameAST *q = ast->asQualifiedName()) | 
					
						
							|  |  |  |         ast = q->unqualified_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (! ast) | 
					
						
							|  |  |  |         return; // nothing to do
 | 
					
						
							|  |  |  |     else if (ast->asOperatorFunctionId() != 0 || ast->asConversionFunctionId() != 0) | 
					
						
							|  |  |  |         return; // nothing to do
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned startToken = ast->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (DestructorNameAST *dtor = ast->asDestructorName()) | 
					
						
							|  |  |  |         startToken = dtor->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-12 12:18:49 +02:00
										 |  |  |     else if (TemplateIdAST *templ = ast->asTemplateId()) | 
					
						
							|  |  |  |         startToken = templ->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     addUse(startToken, kind); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  | void CheckSymbols::addUse(unsigned tokenIndex, UseKind kind) | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! tokenIndex) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Token &tok = tokenAt(tokenIndex); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (tok.generated()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned line, column; | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |     getTokenStartPosition(tokenIndex, &line, &column); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     const unsigned length = tok.length(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Use use(line, column, length, kind); | 
					
						
							|  |  |  |     addUse(use); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  | static const int chunkSize = 50; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | void CheckSymbols::addUse(const Use &use) | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  |     if (!use.line) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (! enclosingFunctionDefinition()) { | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  |         if (_usages.size() >= chunkSize) { | 
					
						
							| 
									
										
										
										
											2011-09-01 09:22:50 +02:00
										 |  |  |             if (use.line > _lineOfLastUsage) | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |                 flush(); | 
					
						
							| 
									
										
										
										
											2010-07-15 13:41:51 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-06 12:42:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-01 09:22:50 +02:00
										 |  |  |     _lineOfLastUsage = qMax(_lineOfLastUsage, use.line); | 
					
						
							| 
									
										
										
										
											2010-07-16 18:00:54 +02:00
										 |  |  |     _usages.append(use); | 
					
						
							| 
									
										
										
										
											2010-07-05 18:52:52 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | void CheckSymbols::addType(ClassOrNamespace *b, NameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! b) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |     unsigned startToken = ast->firstToken(); | 
					
						
							|  |  |  |     if (DestructorNameAST *dtor = ast->asDestructorName()) | 
					
						
							|  |  |  |         startToken = dtor->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Token &tok = tokenAt(startToken); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     if (tok.generated()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned line, column; | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |     getTokenStartPosition(startToken, &line, &column); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     const unsigned length = tok.length(); | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |     const Use use(line, column, length, SemanticInfo::TypeUse); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     addUse(use); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     //qDebug() << "added use" << oo(ast->name) << line << column << length;
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 15:39:00 +02:00
										 |  |  | bool CheckSymbols::isTemplateClass(Symbol *symbol) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (symbol) { | 
					
						
							|  |  |  |         if (Template *templ = symbol->asTemplate()) { | 
					
						
							|  |  |  |             if (Symbol *declaration = templ->declaration()) { | 
					
						
							| 
									
										
										
										
											2010-08-12 11:03:17 +02:00
										 |  |  |                 if (declaration->isClass() || declaration->isForwardClassDeclaration()) | 
					
						
							| 
									
										
										
										
											2010-08-11 15:39:00 +02:00
										 |  |  |                     return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | void CheckSymbols::addTypeOrStatic(const QList<LookupItem> &candidates, NameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |     unsigned startToken = ast->firstToken(); | 
					
						
							|  |  |  |     if (DestructorNameAST *dtor = ast->asDestructorName()) | 
					
						
							|  |  |  |         startToken = dtor->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Token &tok = tokenAt(startToken); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     if (tok.generated()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |     foreach (const LookupItem &r, candidates) { | 
					
						
							|  |  |  |         Symbol *c = r.declaration(); | 
					
						
							| 
									
										
										
										
											2010-05-25 17:49:29 +02:00
										 |  |  |         if (c->isUsingDeclaration()) // skip using declarations...
 | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         else if (c->isUsingNamespaceDirective()) // ... and using namespace directives.
 | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         else if (c->isTypedef() || c->isNamespace() || | 
					
						
							| 
									
										
										
										
											2010-08-11 15:39:00 +02:00
										 |  |  |                  c->isClass() || c->isEnum() || isTemplateClass(c) || | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                  c->isForwardClassDeclaration() || c->isTypenameArgument() || c->enclosingEnum() != 0) { | 
					
						
							| 
									
										
										
										
											2010-08-09 15:40:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             unsigned line, column; | 
					
						
							|  |  |  |             getTokenStartPosition(startToken, &line, &column); | 
					
						
							|  |  |  |             const unsigned length = tok.length(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |             UseKind kind = SemanticInfo::TypeUse; | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |             if (c->enclosingEnum() != 0) | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |                 kind = SemanticInfo::StaticUse; | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             const Use use(line, column, length, kind); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |             addUse(use); | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |             //qDebug() << "added use" << oo(ast->name) << line << column << length;
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | void CheckSymbols::addClassMember(const QList<LookupItem> &candidates, NameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned startToken = ast->firstToken(); | 
					
						
							|  |  |  |     if (DestructorNameAST *dtor = ast->asDestructorName()) | 
					
						
							|  |  |  |         startToken = dtor->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Token &tok = tokenAt(startToken); | 
					
						
							|  |  |  |     if (tok.generated()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |     foreach (const LookupItem &r, candidates) { | 
					
						
							|  |  |  |         Symbol *c = r.declaration(); | 
					
						
							|  |  |  |         if (! c) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         else if (! c->isDeclaration()) | 
					
						
							| 
									
										
										
										
											2010-08-03 18:37:04 +02:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |         else if (! (c->enclosingScope() && c->enclosingScope()->isClass())) | 
					
						
							| 
									
										
										
										
											2010-08-03 18:37:04 +02:00
										 |  |  |             return; // shadowed
 | 
					
						
							|  |  |  |         else if (c->isTypedef() || c->type()->isFunctionType()) | 
					
						
							|  |  |  |             return; // shadowed
 | 
					
						
							| 
									
										
										
										
											2010-07-15 16:03:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 15:40:14 +02:00
										 |  |  |         unsigned line, column; | 
					
						
							|  |  |  |         getTokenStartPosition(startToken, &line, &column); | 
					
						
							|  |  |  |         const unsigned length = tok.length(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |         const Use use(line, column, length, SemanticInfo::FieldUse); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         addUse(use); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | void CheckSymbols::addStatic(const QList<LookupItem> &candidates, NameAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->asDestructorName() != 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned startToken = ast->firstToken(); | 
					
						
							|  |  |  |     const Token &tok = tokenAt(startToken); | 
					
						
							|  |  |  |     if (tok.generated()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     foreach (const LookupItem &r, candidates) { | 
					
						
							|  |  |  |         Symbol *c = r.declaration(); | 
					
						
							|  |  |  |         if (! c) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |         if (c->enclosingScope()->isEnum()) { | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |             unsigned line, column; | 
					
						
							|  |  |  |             getTokenStartPosition(startToken, &line, &column); | 
					
						
							|  |  |  |             const unsigned length = tok.length(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |             const Use use(line, column, length, SemanticInfo::StaticUse); | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  |             addUse(use); | 
					
						
							|  |  |  |             //qDebug() << "added use" << oo(ast->name) << line << column << length;
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | void CheckSymbols::addVirtualMethod(const QList<LookupItem> &candidates, NameAST *ast, unsigned argumentCount) | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned startToken = ast->firstToken(); | 
					
						
							|  |  |  |     if (DestructorNameAST *dtor = ast->asDestructorName()) | 
					
						
							|  |  |  |         startToken = dtor->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Token &tok = tokenAt(startToken); | 
					
						
							|  |  |  |     if (tok.generated()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     foreach (const LookupItem &r, candidates) { | 
					
						
							|  |  |  |         Symbol *c = r.declaration(); | 
					
						
							|  |  |  |         if (! c) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Function *funTy = r.type()->asFunctionType(); | 
					
						
							|  |  |  |         if (! funTy) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         if (! funTy->isVirtual()) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         else if (argumentCount < funTy->minimumArgumentCount()) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2010-08-09 15:40:14 +02:00
										 |  |  |         else if (argumentCount > funTy->argumentCount()) { | 
					
						
							|  |  |  |             if (! funTy->isVariadic()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         unsigned line, column; | 
					
						
							|  |  |  |         getTokenStartPosition(startToken, &line, &column); | 
					
						
							|  |  |  |         const unsigned length = tok.length(); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 09:47:54 +02:00
										 |  |  |         const Use use(line, column, length, SemanticInfo::VirtualMethodUse); | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |         addUse(use); | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2010-05-25 14:53:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | NameAST *CheckSymbols::declaratorId(DeclaratorAST *ast) const | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (ast && ast->core_declarator) { | 
					
						
							|  |  |  |         if (NestedDeclaratorAST *nested = ast->core_declarator->asNestedDeclarator()) | 
					
						
							|  |  |  |             return declaratorId(nested->declarator); | 
					
						
							|  |  |  |         else if (DeclaratorIdAST *declId = ast->core_declarator->asDeclaratorId()) { | 
					
						
							|  |  |  |             return declId->name; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | bool CheckSymbols::maybeType(const Name *name) const | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (name) { | 
					
						
							|  |  |  |         if (const Identifier *ident = name->identifier()) { | 
					
						
							|  |  |  |             const QByteArray id = QByteArray::fromRawData(ident->chars(), ident->size()); | 
					
						
							|  |  |  |             if (_potentialTypes.contains(id)) | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2010-06-01 11:57:25 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-06 12:42:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  | bool CheckSymbols::maybeMember(const Name *name) const | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (name) { | 
					
						
							|  |  |  |         if (const Identifier *ident = name->identifier()) { | 
					
						
							|  |  |  |             const QByteArray id = QByteArray::fromRawData(ident->chars(), ident->size()); | 
					
						
							|  |  |  |             if (_potentialMembers.contains(id)) | 
					
						
							|  |  |  |                 return true; | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 18:07:09 +02:00
										 |  |  | bool CheckSymbols::maybeStatic(const Name *name) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (name) { | 
					
						
							|  |  |  |         if (const Identifier *ident = name->identifier()) { | 
					
						
							|  |  |  |             const QByteArray id = QByteArray::fromRawData(ident->chars(), ident->size()); | 
					
						
							|  |  |  |             if (_potentialStatics.contains(id)) | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  | bool CheckSymbols::maybeVirtualMethod(const Name *name) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-05 12:19:07 +02:00
										 |  |  |     if (name) { | 
					
						
							|  |  |  |         if (const Identifier *ident = name->identifier()) { | 
					
						
							|  |  |  |             const QByteArray id = QByteArray::fromRawData(ident->chars(), ident->size()); | 
					
						
							|  |  |  |             if (_potentialVirtualMethods.contains(id)) | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-08-03 17:34:51 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  | static bool sortByLinePredicate(const CheckSymbols::Use &lhs, const CheckSymbols::Use &rhs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return lhs.line < rhs.line; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-13 15:25:05 +02:00
										 |  |  | void CheckSymbols::flush() | 
					
						
							| 
									
										
										
										
											2010-07-06 12:42:55 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  |     _lineOfLastUsage = 0; | 
					
						
							| 
									
										
										
										
											2010-07-15 13:41:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 18:00:54 +02:00
										 |  |  |     if (_usages.isEmpty()) | 
					
						
							| 
									
										
										
										
											2010-07-06 12:42:55 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  |     qSort(_usages.begin(), _usages.end(), sortByLinePredicate); | 
					
						
							| 
									
										
										
										
											2010-07-16 18:00:54 +02:00
										 |  |  |     reportResults(_usages); | 
					
						
							|  |  |  |     _usages.clear(); | 
					
						
							| 
									
										
										
										
											2011-08-30 10:52:41 +02:00
										 |  |  |     _usages.reserve(chunkSize); | 
					
						
							| 
									
										
										
										
											2010-07-06 12:42:55 +02:00
										 |  |  | } |