| 
									
										
										
										
											2010-05-05 10:28:32 +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). | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-11-02 15:59:12 +01:00
										 |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +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.
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +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. | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "LookupContext.h"
 | 
					
						
							|  |  |  | #include "ResolveExpression.h"
 | 
					
						
							|  |  |  | #include "Overview.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-20 13:44:38 +02:00
										 |  |  | #include "DeprecatedGenTemplateInstance.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <CoreTypes.h>
 | 
					
						
							|  |  |  | #include <Symbols.h>
 | 
					
						
							|  |  |  | #include <Literals.h>
 | 
					
						
							|  |  |  | #include <Names.h>
 | 
					
						
							|  |  |  | #include <Scope.h>
 | 
					
						
							|  |  |  | #include <Control.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-15 10:42:41 +01:00
										 |  |  | #include <QStack>
 | 
					
						
							|  |  |  | #include <QHash>
 | 
					
						
							|  |  |  | #include <QVarLengthArray>
 | 
					
						
							|  |  |  | #include <QtDebug>
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | using namespace CPlusPlus; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 14:00:56 +02:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | const bool debug = ! qgetenv("CPLUSPLUS_LOOKUPCONTEXT_DEBUG").isEmpty(); | 
					
						
							|  |  |  | } // end of anonymous namespace
 | 
					
						
							| 
									
										
										
										
											2010-05-14 14:00:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  | static void addNames(const Name *name, QList<const Name *> *names, bool addAllNames = false) | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! name) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     else if (const QualifiedNameId *q = name->asQualifiedNameId()) { | 
					
						
							|  |  |  |         addNames(q->base(), names); | 
					
						
							| 
									
										
										
										
											2010-07-21 12:48:05 +02:00
										 |  |  |         addNames(q->name(), names, addAllNames); | 
					
						
							|  |  |  |     } else if (addAllNames || name->isNameId() || name->isTemplateNameId()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |         names->append(name); | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  | static void path_helper(Symbol *symbol, QList<const Name *> *names) | 
					
						
							| 
									
										
										
										
											2010-05-05 16:25:16 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! symbol) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |     path_helper(symbol->enclosingScope(), names); | 
					
						
							| 
									
										
										
										
											2010-05-05 16:25:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 15:10:28 +02:00
										 |  |  |     if (symbol->name()) { | 
					
						
							|  |  |  |         if (symbol->isClass() || symbol->isNamespace()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |             addNames(symbol->name(), names); | 
					
						
							| 
									
										
										
										
											2010-05-14 15:10:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-20 12:13:44 +02:00
										 |  |  |         } else if (symbol->isObjCClass() || symbol->isObjCBaseClass() || symbol->isObjCProtocol() | 
					
						
							| 
									
										
										
										
											2010-06-16 15:35:34 +02:00
										 |  |  |                 || symbol->isObjCForwardClassDeclaration() || symbol->isObjCForwardProtocolDeclaration() | 
					
						
							|  |  |  |                 || symbol->isForwardClassDeclaration()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |             addNames(symbol->name(), names); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 15:10:28 +02:00
										 |  |  |         } else if (symbol->isFunction()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |             if (const QualifiedNameId *q = symbol->name()->asQualifiedNameId()) | 
					
						
							|  |  |  |                 addNames(q->base(), names); | 
					
						
							| 
									
										
										
										
											2010-05-05 16:25:16 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  | bool ClassOrNamespace::CompareName::operator()(const Name *name, const Name *other) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Q_ASSERT(name != 0); | 
					
						
							|  |  |  |     Q_ASSERT(other != 0); | 
					
						
							| 
									
										
										
										
											2010-05-10 12:19:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     const Identifier *id = name->identifier(); | 
					
						
							|  |  |  |     const Identifier *otherId = other->identifier(); | 
					
						
							| 
									
										
										
										
											2010-05-10 12:19:13 +02:00
										 |  |  |     return strcmp(id->chars(), otherId->chars()) < 0; | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | /////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // LookupContext
 | 
					
						
							|  |  |  | /////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | LookupContext::LookupContext() | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  |     : _control(new Control()) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LookupContext::LookupContext(Document::Ptr thisDocument, | 
					
						
							|  |  |  |                              const Snapshot &snapshot) | 
					
						
							|  |  |  |     : _expressionDocument(Document::create("<LookupContext>")), | 
					
						
							|  |  |  |       _thisDocument(thisDocument), | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  |       _snapshot(snapshot), | 
					
						
							|  |  |  |       _control(new Control()) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LookupContext::LookupContext(Document::Ptr expressionDocument, | 
					
						
							|  |  |  |                              Document::Ptr thisDocument, | 
					
						
							|  |  |  |                              const Snapshot &snapshot) | 
					
						
							|  |  |  |     : _expressionDocument(expressionDocument), | 
					
						
							|  |  |  |       _thisDocument(thisDocument), | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  |       _snapshot(snapshot), | 
					
						
							|  |  |  |       _control(new Control()) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LookupContext::LookupContext(const LookupContext &other) | 
					
						
							| 
									
										
										
										
											2010-05-17 16:56:22 +02:00
										 |  |  |     : _expressionDocument(other._expressionDocument), | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |       _thisDocument(other._thisDocument), | 
					
						
							|  |  |  |       _snapshot(other._snapshot), | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  |       _bindings(other._bindings), | 
					
						
							|  |  |  |       _control(other._control) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LookupContext &LookupContext::operator = (const LookupContext &other) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _expressionDocument = other._expressionDocument; | 
					
						
							|  |  |  |     _thisDocument = other._thisDocument; | 
					
						
							|  |  |  |     _snapshot = other._snapshot; | 
					
						
							|  |  |  |     _bindings = other._bindings; | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  |     _control = other._control; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     return *this; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 12:47:20 +02:00
										 |  |  | QList<const Name *> LookupContext::fullyQualifiedName(Symbol *symbol) | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |     QList<const Name *> qualifiedName = path(symbol->enclosingScope()); | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |     addNames(symbol->name(), &qualifiedName, /*add all names*/ true); | 
					
						
							|  |  |  |     return qualifiedName; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QList<const Name *> LookupContext::path(Symbol *symbol) | 
					
						
							| 
									
										
										
										
											2010-05-11 12:47:20 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     QList<const Name *> names; | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |     path_helper(symbol, &names); | 
					
						
							| 
									
										
										
										
											2010-05-11 12:47:20 +02:00
										 |  |  |     return names; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-31 12:09:27 +01:00
										 |  |  | static bool symbolIdentical(Symbol *s1, Symbol *s2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!s1 || !s2) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     if (s1->line() != s2->line()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     if (s1->column() != s2->column()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return QByteArray(s1->fileName()) == QByteArray(s2->fileName()); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-31 12:09:27 +01:00
										 |  |  | const Name *LookupContext::minimalName(Symbol *symbol, ClassOrNamespace *target, Control *control) | 
					
						
							| 
									
										
										
										
											2010-07-08 16:40:46 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-31 12:09:27 +01:00
										 |  |  |     const Name *n = 0; | 
					
						
							|  |  |  |     QList<const Name *> names = LookupContext::fullyQualifiedName(symbol); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = names.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |         if (! n) | 
					
						
							|  |  |  |             n = names.at(i); | 
					
						
							| 
									
										
										
										
											2010-07-08 16:40:46 +02:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2011-10-31 12:09:27 +01:00
										 |  |  |             n = control->qualifiedNameId(names.at(i), n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // once we're qualified enough to get the same symbol, break
 | 
					
						
							|  |  |  |         if (target) { | 
					
						
							|  |  |  |             const QList<LookupItem> tresults = target->lookup(n); | 
					
						
							|  |  |  |             foreach (const LookupItem &tr, tresults) { | 
					
						
							|  |  |  |                 if (symbolIdentical(tr.declaration(), symbol)) | 
					
						
							|  |  |  |                     return n; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-07-08 16:40:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-31 12:09:27 +01:00
										 |  |  |     return n; | 
					
						
							| 
									
										
										
										
											2010-07-08 16:40:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | QSharedPointer<CreateBindings> LookupContext::bindings() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! _bindings) | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  |         _bindings = QSharedPointer<CreateBindings>(new CreateBindings(_thisDocument, _snapshot, control())); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return _bindings; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LookupContext::setBindings(QSharedPointer<CreateBindings> bindings) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _bindings = bindings; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  | QSharedPointer<Control> LookupContext::control() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _control; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | Document::Ptr LookupContext::expressionDocument() const | 
					
						
							|  |  |  | { return _expressionDocument; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Document::Ptr LookupContext::thisDocument() const | 
					
						
							|  |  |  | { return _thisDocument; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Document::Ptr LookupContext::document(const QString &fileName) const | 
					
						
							|  |  |  | { return _snapshot.document(fileName); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Snapshot LookupContext::snapshot() const | 
					
						
							|  |  |  | { return _snapshot; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ClassOrNamespace *LookupContext::globalNamespace() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return bindings()->globalNamespace(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  | ClassOrNamespace *LookupContext::lookupType(const Name *name, Scope *scope) const | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-10-04 14:42:37 +02:00
										 |  |  |     if (! scope) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } else if (Block *block = scope->asBlock()) { | 
					
						
							|  |  |  |         for (unsigned i = 0; i < block->memberCount(); ++i) { | 
					
						
							|  |  |  |             if (UsingNamespaceDirective *u = block->memberAt(i)->asUsingNamespaceDirective()) { | 
					
						
							|  |  |  |                 if (ClassOrNamespace *uu = lookupType(u->name(), scope->enclosingNamespace())) { | 
					
						
							|  |  |  |                     if (ClassOrNamespace *r = uu->lookupType(name)) | 
					
						
							|  |  |  |                         return r; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return lookupType(name, scope->enclosingScope()); | 
					
						
							|  |  |  |     } else if (ClassOrNamespace *b = bindings()->lookupType(scope)) { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |         return b->lookupType(name); | 
					
						
							| 
									
										
										
										
											2010-10-04 14:42:37 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  | ClassOrNamespace *LookupContext::lookupType(Symbol *symbol) const | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 14:00:56 +02:00
										 |  |  |     return bindings()->lookupType(symbol); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | QList<LookupItem> LookupContext::lookup(const Name *name, Scope *scope) const | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |     QList<LookupItem> candidates; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (! name) | 
					
						
							|  |  |  |         return candidates; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-26 16:16:22 +02:00
										 |  |  |     for (; scope; scope = scope->enclosingScope()) { | 
					
						
							| 
									
										
										
										
											2010-10-04 14:42:37 +02:00
										 |  |  |         if (name->identifier() != 0 && scope->isBlock()) { | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |             bindings()->lookupInScope(name, scope, &candidates, /*templateId = */ 0, /*binding=*/ 0); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                 break; // it's a local.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-04 14:42:37 +02:00
										 |  |  |             for (unsigned i = 0; i < scope->memberCount(); ++i) { | 
					
						
							|  |  |  |                 if (UsingNamespaceDirective *u = scope->memberAt(i)->asUsingNamespaceDirective()) { | 
					
						
							|  |  |  |                     if (ClassOrNamespace *uu = lookupType(u->name(), scope->enclosingNamespace())) { | 
					
						
							|  |  |  |                         candidates = uu->find(name); | 
					
						
							| 
									
										
										
										
											2010-05-05 12:06:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-04 14:42:37 +02:00
										 |  |  |                         if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                             return candidates; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         } else if (Function *fun = scope->asFunction()) { | 
					
						
							|  |  |  |             bindings()->lookupInScope(name, fun, &candidates, /*templateId = */ 0, /*binding=*/ 0); | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             if (! candidates.isEmpty()) | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  |                 break; // it's an argument or a template parameter.
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (fun->name() && fun->name()->isQualifiedNameId()) { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  |                 if (ClassOrNamespace *binding = bindings()->lookupType(fun)) { | 
					
						
							| 
									
										
										
										
											2010-06-01 12:22:07 +02:00
										 |  |  |                     candidates = binding->find(name); | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                         return candidates; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  |             // contunue, and look at the enclosing scope.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         } else if (ObjCMethod *method = scope->asObjCMethod()) { | 
					
						
							|  |  |  |             bindings()->lookupInScope(name, method, &candidates, /*templateId = */ 0, /*binding=*/ 0); | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                 break; // it's a formal argument.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-12 12:34:23 +02:00
										 |  |  |         } else if (Template *templ = scope->asTemplate()) { | 
					
						
							|  |  |  |             bindings()->lookupInScope(name, templ, &candidates, /*templateId = */ 0, /*binding=*/ 0); | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (! candidates.isEmpty()) | 
					
						
							| 
									
										
										
										
											2010-08-12 12:34:23 +02:00
										 |  |  |                 return candidates;  // it's a template parameter.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } else if (Class *klass = scope->asClass()) { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (ClassOrNamespace *binding = bindings()->lookupType(klass)) { | 
					
						
							| 
									
										
										
										
											2010-06-01 12:22:07 +02:00
										 |  |  |                 candidates = binding->find(name); | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                     return candidates; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         } else if (Namespace *ns = scope->asNamespace()) { | 
					
						
							|  |  |  |             if (ClassOrNamespace *binding = bindings()->lookupType(ns)) | 
					
						
							| 
									
										
										
										
											2010-06-01 12:22:07 +02:00
										 |  |  |                 candidates = binding->find(name); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-01 12:22:07 +02:00
										 |  |  |                 if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                     return candidates; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         } else if (scope->isObjCClass() || scope->isObjCProtocol()) { | 
					
						
							|  |  |  |             if (ClassOrNamespace *binding = bindings()->lookupType(scope)) | 
					
						
							| 
									
										
										
										
											2010-06-01 12:22:07 +02:00
										 |  |  |                 candidates = binding->find(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (! candidates.isEmpty()) | 
					
						
							|  |  |  |                     return candidates; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return candidates; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-08 16:40:46 +02:00
										 |  |  | ClassOrNamespace *LookupContext::lookupParent(Symbol *symbol) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |     QList<const Name *> fqName = path(symbol); | 
					
						
							| 
									
										
										
										
											2010-07-08 16:40:46 +02:00
										 |  |  |     ClassOrNamespace *binding = globalNamespace(); | 
					
						
							|  |  |  |     foreach (const Name *name, fqName) { | 
					
						
							|  |  |  |         binding = binding->findType(name); | 
					
						
							|  |  |  |         if (!binding) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return binding; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | ClassOrNamespace::ClassOrNamespace(CreateBindings *factory, ClassOrNamespace *parent) | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |     : _factory(factory), _parent(parent), _templateId(0), _instantiationOrigin(0) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 16:04:43 +02:00
										 |  |  | const TemplateNameId *ClassOrNamespace::templateId() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _templateId; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::instantiationOrigin() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _instantiationOrigin; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-10 12:19:13 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::parent() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _parent; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | QList<ClassOrNamespace *> ClassOrNamespace::usings() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const_cast<ClassOrNamespace *>(this)->flush(); | 
					
						
							|  |  |  |     return _usings; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QList<Enum *> ClassOrNamespace::enums() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const_cast<ClassOrNamespace *>(this)->flush(); | 
					
						
							|  |  |  |     return _enums; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QList<Symbol *> ClassOrNamespace::symbols() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-12 14:33:21 +02:00
										 |  |  |     if (_templateId && ! _usings.isEmpty()) | 
					
						
							|  |  |  |         return _usings.first()->symbols(); // ask to the base implementation
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     const_cast<ClassOrNamespace *>(this)->flush(); | 
					
						
							|  |  |  |     return _symbols; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ClassOrNamespace *ClassOrNamespace::globalNamespace() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ClassOrNamespace *e = const_cast<ClassOrNamespace *>(this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         if (! e->_parent) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         e = e->_parent; | 
					
						
							|  |  |  |     } while (e); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return e; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | QList<LookupItem> ClassOrNamespace::find(const Name *name) | 
					
						
							| 
									
										
										
										
											2010-05-11 10:20:51 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return lookup_helper(name, false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | QList<LookupItem> ClassOrNamespace::lookup(const Name *name) | 
					
						
							| 
									
										
										
										
											2010-05-11 10:20:51 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return lookup_helper(name, true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | QList<LookupItem> ClassOrNamespace::lookup_helper(const Name *name, bool searchInEnclosingScope) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |     QList<LookupItem> result; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 16:04:43 +02:00
										 |  |  |     if (name) { | 
					
						
							|  |  |  |         if (const QualifiedNameId *q = name->asQualifiedNameId()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |             if (! q->base()) | 
					
						
							|  |  |  |                 result = globalNamespace()->find(q->name()); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |             else if (ClassOrNamespace *binding = lookupType(q->base())) | 
					
						
							|  |  |  |                 result = binding->find(q->name()); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 16:04:43 +02:00
										 |  |  |             return result; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 16:04:43 +02:00
										 |  |  |         QSet<ClassOrNamespace *> processed; | 
					
						
							|  |  |  |         ClassOrNamespace *binding = this; | 
					
						
							|  |  |  |         do { | 
					
						
							|  |  |  |             lookup_helper(name, binding, &result, &processed, /*templateId = */ 0); | 
					
						
							|  |  |  |             binding = binding->_parent; | 
					
						
							|  |  |  |         } while (searchInEnclosingScope && binding); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ClassOrNamespace::lookup_helper(const Name *name, ClassOrNamespace *binding, | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |                                           QList<LookupItem> *result, | 
					
						
							|  |  |  |                                           QSet<ClassOrNamespace *> *processed, | 
					
						
							|  |  |  |                                           const TemplateNameId *templateId) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 14:37:17 +02:00
										 |  |  |     if (binding && ! processed->contains(binding)) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         processed->insert(binding); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-01 14:00:02 +02:00
										 |  |  |         const Identifier *nameId = name->identifier(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         foreach (Symbol *s, binding->symbols()) { | 
					
						
							| 
									
										
										
										
											2010-07-16 15:23:26 +02:00
										 |  |  |             if (s->isFriend()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2010-12-03 11:26:31 +01:00
										 |  |  |             else if (s->isUsingNamespaceDirective()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2010-07-16 15:23:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |             if (Scope *scope = s->asScope()) { | 
					
						
							|  |  |  |                 if (Class *klass = scope->asClass()) { | 
					
						
							| 
									
										
										
										
											2010-06-01 14:00:02 +02:00
										 |  |  |                     if (const Identifier *id = klass->identifier()) { | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |                         if (nameId && nameId->isEqualTo(id)) { | 
					
						
							|  |  |  |                             LookupItem item; | 
					
						
							|  |  |  |                             item.setDeclaration(klass); | 
					
						
							|  |  |  |                             item.setBinding(binding); | 
					
						
							|  |  |  |                             result->append(item); | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2010-06-01 14:00:02 +02:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |                 _factory->lookupInScope(name, scope, result, templateId, binding); | 
					
						
							| 
									
										
										
										
											2010-06-01 14:00:02 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach (Enum *e, binding->enums()) | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |             _factory->lookupInScope(name, e, result, templateId, binding); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         foreach (ClassOrNamespace *u, binding->usings()) | 
					
						
							| 
									
										
										
										
											2010-05-06 14:01:38 +02:00
										 |  |  |             lookup_helper(name, u, result, processed, binding->_templateId); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 10:01:10 +02:00
										 |  |  | void CreateBindings::lookupInScope(const Name *name, Scope *scope, | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |                                    QList<LookupItem> *result, | 
					
						
							|  |  |  |                                    const TemplateNameId *templateId, | 
					
						
							|  |  |  |                                    ClassOrNamespace *binding) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! name) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else if (const OperatorNameId *op = name->asOperatorNameId()) { | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         for (Symbol *s = scope->find(op->kind()); s; s = s->next()) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             if (! s->name()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2010-07-16 15:23:26 +02:00
										 |  |  |             else if (s->isFriend()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             else if (! s->name()->isEqualTo(op)) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2010-05-06 14:01:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |             LookupItem item; | 
					
						
							|  |  |  |             item.setDeclaration(s); | 
					
						
							|  |  |  |             item.setBinding(binding); | 
					
						
							|  |  |  |             result->append(item); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else if (const Identifier *id = name->identifier()) { | 
					
						
							| 
									
										
										
										
											2010-08-11 12:26:02 +02:00
										 |  |  |         for (Symbol *s = scope->find(id); s; s = s->next()) { | 
					
						
							| 
									
										
										
										
											2010-07-16 15:23:26 +02:00
										 |  |  |             if (s->isFriend()) | 
					
						
							|  |  |  |                 continue; // skip friends
 | 
					
						
							| 
									
										
										
										
											2010-12-03 11:26:31 +01:00
										 |  |  |             else if (s->isUsingNamespaceDirective()) | 
					
						
							|  |  |  |                 continue; // skip using namespace directives
 | 
					
						
							| 
									
										
										
										
											2010-07-16 15:23:26 +02:00
										 |  |  |             else if (! id->isEqualTo(s->identifier())) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2010-05-12 11:16:54 +02:00
										 |  |  |             else if (s->name()->isQualifiedNameId()) | 
					
						
							|  |  |  |                 continue; // skip qualified ids.
 | 
					
						
							| 
									
										
										
										
											2010-05-06 14:01:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |             LookupItem item; | 
					
						
							|  |  |  |             item.setDeclaration(s); | 
					
						
							|  |  |  |             item.setBinding(binding); | 
					
						
							| 
									
										
										
										
											2010-05-10 09:44:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 10:51:54 +01:00
										 |  |  |             if (s->asNamespaceAlias() && binding) { | 
					
						
							|  |  |  |                 ClassOrNamespace *targetNamespaceBinding = binding->lookupType(name); | 
					
						
							|  |  |  |                 if (targetNamespaceBinding && targetNamespaceBinding->symbols().size() == 1) { | 
					
						
							|  |  |  |                     Symbol *resolvedSymbol = targetNamespaceBinding->symbols().first(); | 
					
						
							|  |  |  |                     item.setType(resolvedSymbol->type()); // override the type
 | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |             if (templateId && (s->isDeclaration() || s->isFunction())) { | 
					
						
							| 
									
										
										
										
											2010-05-31 13:57:32 +02:00
										 |  |  |                 FullySpecifiedType ty = DeprecatedGenTemplateInstance::instantiate(templateId, s, _control); | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |                 item.setType(ty); // override the type.
 | 
					
						
							| 
									
										
										
										
											2010-05-10 09:44:18 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  |             result->append(item); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::lookupType(const Name *name) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! name) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QSet<ClassOrNamespace *> processed; | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |     return lookupType_helper(name, &processed, /*searchInEnclosingScope =*/ true, this); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::findType(const Name *name) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     QSet<ClassOrNamespace *> processed; | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |     return lookupType_helper(name, &processed, /*searchInEnclosingScope =*/ false, this); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:49:16 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::lookupType_helper(const Name *name, | 
					
						
							| 
									
										
										
										
											2010-05-14 13:54:56 +02:00
										 |  |  |                                                       QSet<ClassOrNamespace *> *processed, | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |                                                       bool searchInEnclosingScope, | 
					
						
							|  |  |  |                                                       ClassOrNamespace *origin) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 15:32:40 +02:00
										 |  |  |     if (const QualifiedNameId *q = name->asQualifiedNameId()) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |         QSet<ClassOrNamespace *> innerProcessed; | 
					
						
							|  |  |  |         if (! q->base()) { | 
					
						
							|  |  |  |             return globalNamespace()->lookupType_helper(q->name(), &innerProcessed, true, origin); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |         if (ClassOrNamespace *binding = lookupType_helper(q->base(), processed, true, origin)) { | 
					
						
							|  |  |  |             return binding->lookupType_helper(q->name(), &innerProcessed, false, origin); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 15:32:40 +02:00
										 |  |  |     } else if (! processed->contains(this)) { | 
					
						
							|  |  |  |         processed->insert(this); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 15:32:40 +02:00
										 |  |  |         if (name->isNameId() || name->isTemplateNameId()) { | 
					
						
							|  |  |  |             flush(); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-01 14:00:02 +02:00
										 |  |  |             foreach (Symbol *s, symbols()) { | 
					
						
							|  |  |  |                 if (Class *klass = s->asClass()) { | 
					
						
							|  |  |  |                     if (klass->identifier() && klass->identifier()->isEqualTo(name->identifier())) | 
					
						
							|  |  |  |                         return this; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |             if (ClassOrNamespace *e = nestedType(name, origin)) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |                 return e; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  |             else if (_templateId) { | 
					
						
							| 
									
										
										
										
											2010-05-25 16:27:53 +02:00
										 |  |  |                 if (_usings.size() == 1) { | 
					
						
							|  |  |  |                     ClassOrNamespace *delegate = _usings.first(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |                     if (ClassOrNamespace *r = delegate->lookupType_helper(name, processed, /*searchInEnclosingScope = */ true, origin)) | 
					
						
							| 
									
										
										
										
											2010-05-25 16:27:53 +02:00
										 |  |  |                         return r; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     if (debug) | 
					
						
							|  |  |  |                         qWarning() << "expected one using declaration. Number of using declarations is:" << _usings.size(); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-05-21 15:44:04 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             foreach (ClassOrNamespace *u, usings()) { | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |                 if (ClassOrNamespace *r = u->lookupType_helper(name, processed, /*searchInEnclosingScope =*/ false, origin)) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |                     return r; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:54:56 +02:00
										 |  |  |         if (_parent && searchInEnclosingScope) | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |             return _parent->lookupType_helper(name, processed, searchInEnclosingScope, origin); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::nestedType(const Name *name, ClassOrNamespace *origin) const | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     Q_ASSERT(name != 0); | 
					
						
							|  |  |  |     Q_ASSERT(name->isNameId() || name->isTemplateNameId()); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     const_cast<ClassOrNamespace *>(this)->flush(); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     Table::const_iterator it = _classOrNamespaces.find(name); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     if (it == _classOrNamespaces.end()) | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 14:01:38 +02:00
										 |  |  |     ClassOrNamespace *c = it->second; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (const TemplateNameId *templId = name->asTemplateNameId()) { | 
					
						
							|  |  |  |         ClassOrNamespace *i = _factory->allocClassOrNamespace(c); | 
					
						
							|  |  |  |         i->_templateId = templId; | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |         i->_instantiationOrigin = origin; | 
					
						
							| 
									
										
										
										
											2010-05-06 14:01:38 +02:00
										 |  |  |         i->_usings.append(c); | 
					
						
							|  |  |  |         return i; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return c; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ClassOrNamespace::flush() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-05 14:07:10 +02:00
										 |  |  |     if (! _todo.isEmpty()) { | 
					
						
							|  |  |  |         const QList<Symbol *> todo = _todo; | 
					
						
							|  |  |  |         _todo.clear(); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 14:07:10 +02:00
										 |  |  |         foreach (Symbol *member, todo) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             _factory->process(member, this); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ClassOrNamespace::addSymbol(Symbol *symbol) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _symbols.append(symbol); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ClassOrNamespace::addTodo(Symbol *symbol) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _todo.append(symbol); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ClassOrNamespace::addEnum(Enum *e) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _enums.append(e); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ClassOrNamespace::addUsing(ClassOrNamespace *u) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _usings.append(u); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:49:16 +02:00
										 |  |  | void ClassOrNamespace::addNestedType(const Name *alias, ClassOrNamespace *e) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     _classOrNamespaces[alias] = e; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  | ClassOrNamespace *ClassOrNamespace::findOrCreateType(const Name *name, ClassOrNamespace *origin) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! name) | 
					
						
							|  |  |  |         return this; | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |     if (! origin) | 
					
						
							|  |  |  |         origin = this; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (const QualifiedNameId *q = name->asQualifiedNameId()) { | 
					
						
							| 
									
										
										
										
											2010-07-12 13:41:54 +02:00
										 |  |  |         if (! q->base()) | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |             return globalNamespace()->findOrCreateType(q->name(), origin); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |         return findOrCreateType(q->base(), origin)->findOrCreateType(q->name(), origin); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |     } else if (name->isNameId() || name->isTemplateNameId()) { | 
					
						
							| 
									
										
										
										
											2011-08-19 13:10:36 +02:00
										 |  |  |         ClassOrNamespace *e = nestedType(name, origin); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (! e) { | 
					
						
							|  |  |  |             e = _factory->allocClassOrNamespace(this); | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |             _classOrNamespaces[name] = e; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return e; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  | CreateBindings::CreateBindings(Document::Ptr thisDocument, const Snapshot &snapshot, QSharedPointer<Control> control) | 
					
						
							|  |  |  |     : _snapshot(snapshot), _control(control) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     _globalNamespace = allocClassOrNamespace(/*parent = */ 0); | 
					
						
							|  |  |  |     _currentClassOrNamespace = _globalNamespace; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     process(thisDocument); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CreateBindings::~CreateBindings() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     qDeleteAll(_entities); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 10:03:30 +02:00
										 |  |  | ClassOrNamespace *CreateBindings::switchCurrentClassOrNamespace(ClassOrNamespace *classOrNamespace) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     ClassOrNamespace *previous = _currentClassOrNamespace; | 
					
						
							|  |  |  |     _currentClassOrNamespace = classOrNamespace; | 
					
						
							|  |  |  |     return previous; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ClassOrNamespace *CreateBindings::globalNamespace() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _globalNamespace; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 14:00:56 +02:00
										 |  |  | ClassOrNamespace *CreateBindings::lookupType(Symbol *symbol) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |     const QList<const Name *> path = LookupContext::path(symbol); | 
					
						
							|  |  |  |     return lookupType(path); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  | ClassOrNamespace *CreateBindings::lookupType(const QList<const Name *> &path) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (path.isEmpty()) | 
					
						
							| 
									
										
										
										
											2010-05-05 16:25:16 +02:00
										 |  |  |         return _globalNamespace; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |     ClassOrNamespace *b = _globalNamespace->lookupType(path.at(0)); | 
					
						
							| 
									
										
										
										
											2010-05-05 16:25:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-12 15:09:36 +02:00
										 |  |  |     for (int i = 1; b && i < path.size(); ++i) | 
					
						
							|  |  |  |         b = b->findType(path.at(i)); | 
					
						
							| 
									
										
										
										
											2010-05-05 16:25:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return b; | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CreateBindings::process(Symbol *s, ClassOrNamespace *classOrNamespace) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:03:30 +02:00
										 |  |  |     ClassOrNamespace *previous = switchCurrentClassOrNamespace(classOrNamespace); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     accept(s); | 
					
						
							| 
									
										
										
										
											2010-05-11 10:03:30 +02:00
										 |  |  |     (void) switchCurrentClassOrNamespace(previous); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CreateBindings::process(Symbol *symbol) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _currentClassOrNamespace->addTodo(symbol); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-04 17:58:29 +02:00
										 |  |  | QSharedPointer<Control> CreateBindings::control() const | 
					
						
							| 
									
										
										
										
											2010-05-06 14:01:38 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return _control; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | ClassOrNamespace *CreateBindings::allocClassOrNamespace(ClassOrNamespace *parent) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ClassOrNamespace *e = new ClassOrNamespace(this, parent); | 
					
						
							|  |  |  |     _entities.append(e); | 
					
						
							|  |  |  |     return e; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CreateBindings::process(Document::Ptr doc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! doc) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     else if (Namespace *globalNamespace = doc->globalNamespace()) { | 
					
						
							|  |  |  |         if (! _processed.contains(globalNamespace)) { | 
					
						
							|  |  |  |             _processed.insert(globalNamespace); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             foreach (const Document::Include &i, doc->includes()) { | 
					
						
							|  |  |  |                 if (Document::Ptr incl = _snapshot.document(i.fileName())) | 
					
						
							|  |  |  |                     process(incl); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             accept(globalNamespace); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  | ClassOrNamespace *CreateBindings::enterClassOrNamespaceBinding(Symbol *symbol) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:54:56 +02:00
										 |  |  |     ClassOrNamespace *entity = _currentClassOrNamespace->findOrCreateType(symbol->name()); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     entity->addSymbol(symbol); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 10:03:30 +02:00
										 |  |  |     return switchCurrentClassOrNamespace(entity); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  | ClassOrNamespace *CreateBindings::enterGlobalClassOrNamespace(Symbol *symbol) | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:54:56 +02:00
										 |  |  |     ClassOrNamespace *entity = _globalNamespace->findOrCreateType(symbol->name()); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     entity->addSymbol(symbol); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-11 10:03:30 +02:00
										 |  |  |     return switchCurrentClassOrNamespace(entity); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 14:24:28 +02:00
										 |  |  | bool CreateBindings::visit(Template *templ) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-11 14:30:33 +02:00
										 |  |  |     if (Symbol *d = templ->declaration()) | 
					
						
							|  |  |  |         accept(d); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-11 14:24:28 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | bool CreateBindings::visit(Namespace *ns) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  |     ClassOrNamespace *previous = enterClassOrNamespaceBinding(ns); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < ns->memberCount(); ++i) | 
					
						
							|  |  |  |         process(ns->memberAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-03 13:50:04 +01:00
										 |  |  |     if (ns->isInline() && previous) | 
					
						
							|  |  |  |         previous->addUsing(_currentClassOrNamespace); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     _currentClassOrNamespace = previous; | 
					
						
							| 
									
										
										
										
											2012-02-03 13:50:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(Class *klass) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-10 12:30:42 +02:00
										 |  |  |     ClassOrNamespace *previous = _currentClassOrNamespace; | 
					
						
							|  |  |  |     ClassOrNamespace *binding = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (klass->name() && klass->name()->isQualifiedNameId()) | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |         binding = _currentClassOrNamespace->lookupType(klass->name()); | 
					
						
							| 
									
										
										
										
											2010-05-10 12:30:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (! binding) | 
					
						
							| 
									
										
										
										
											2010-05-14 13:54:56 +02:00
										 |  |  |         binding = _currentClassOrNamespace->findOrCreateType(klass->name()); | 
					
						
							| 
									
										
										
										
											2010-05-10 12:30:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     _currentClassOrNamespace = binding; | 
					
						
							|  |  |  |     _currentClassOrNamespace->addSymbol(klass); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < klass->baseClassCount(); ++i) | 
					
						
							|  |  |  |         process(klass->baseClassAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < klass->memberCount(); ++i) | 
					
						
							|  |  |  |         process(klass->memberAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _currentClassOrNamespace = previous; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ForwardClassDeclaration *klass) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-10 09:47:46 +02:00
										 |  |  |     if (! klass->isFriend()) { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  |         ClassOrNamespace *previous = enterClassOrNamespaceBinding(klass); | 
					
						
							| 
									
										
										
										
											2010-05-10 09:47:46 +02:00
										 |  |  |         _currentClassOrNamespace = previous; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(Enum *e) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _currentClassOrNamespace->addEnum(e); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(Declaration *decl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (decl->isTypedef()) { | 
					
						
							| 
									
										
										
										
											2010-05-26 10:08:29 +02:00
										 |  |  |         FullySpecifiedType ty = decl->type(); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         const Identifier *typedefId = decl->identifier(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (typedefId && ! (ty.isConst() || ty.isVolatile())) { | 
					
						
							|  |  |  |             if (const NamedType *namedTy = ty->asNamedType()) { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |                 if (ClassOrNamespace *e = _currentClassOrNamespace->lookupType(namedTy->name())) { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:49:16 +02:00
										 |  |  |                     _currentClassOrNamespace->addNestedType(decl->name(), e); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |                 } else if (false) { | 
					
						
							|  |  |  |                     Overview oo; | 
					
						
							|  |  |  |                     qDebug() << "found entity not found for" << oo(namedTy->name()); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-05-26 10:08:29 +02:00
										 |  |  |             } else if (Class *klass = ty->asClassType()) { | 
					
						
							| 
									
										
										
										
											2010-09-02 11:59:01 +02:00
										 |  |  |                 if (const Identifier *nameId = decl->name()->asNameId()) { | 
					
						
							| 
									
										
										
										
											2010-05-26 10:08:29 +02:00
										 |  |  |                     ClassOrNamespace *binding = _currentClassOrNamespace->findOrCreateType(nameId); | 
					
						
							|  |  |  |                     binding->addSymbol(klass); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(Function *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(BaseClass *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |     if (ClassOrNamespace *base = _currentClassOrNamespace->lookupType(b->name())) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         _currentClassOrNamespace->addUsing(base); | 
					
						
							|  |  |  |     } else if (false) { | 
					
						
							|  |  |  |         Overview oo; | 
					
						
							|  |  |  |         qDebug() << "no entity for:" << oo(b->name()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-27 16:33:18 +02:00
										 |  |  | bool CreateBindings::visit(UsingDeclaration *u) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (u->name()) { | 
					
						
							|  |  |  |         if (const QualifiedNameId *q = u->name()->asQualifiedNameId()) { | 
					
						
							| 
									
										
										
										
											2010-09-02 11:59:01 +02:00
										 |  |  |             if (const Identifier *unqualifiedId = q->name()->asNameId()) { | 
					
						
							| 
									
										
										
										
											2010-05-27 16:33:18 +02:00
										 |  |  |                 if (ClassOrNamespace *delegate = _currentClassOrNamespace->lookupType(q)) { | 
					
						
							|  |  |  |                     ClassOrNamespace *b = _currentClassOrNamespace->findOrCreateType(unqualifiedId); | 
					
						
							|  |  |  |                     b->addUsing(delegate); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | bool CreateBindings::visit(UsingNamespaceDirective *u) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |     if (ClassOrNamespace *e = _currentClassOrNamespace->lookupType(u->name())) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         _currentClassOrNamespace->addUsing(e); | 
					
						
							|  |  |  |     } else if (false) { | 
					
						
							|  |  |  |         Overview oo; | 
					
						
							|  |  |  |         qDebug() << "no entity for namespace:" << oo(u->name()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(NamespaceAlias *a) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! a->identifier()) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |     } else if (ClassOrNamespace *e = _currentClassOrNamespace->lookupType(a->namespaceName())) { | 
					
						
							| 
									
										
										
										
											2010-05-06 11:26:23 +02:00
										 |  |  |         if (a->name()->isNameId() || a->name()->isTemplateNameId()) | 
					
						
							| 
									
										
										
										
											2010-05-14 13:49:16 +02:00
										 |  |  |             _currentClassOrNamespace->addNestedType(a->name(), e); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } else if (false) { | 
					
						
							|  |  |  |         Overview oo; | 
					
						
							|  |  |  |         qDebug() << "no entity for namespace:" << oo(a->namespaceName()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCClass *klass) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  |     ClassOrNamespace *previous = enterGlobalClassOrNamespace(klass); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     process(klass->baseClass()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < klass->protocolCount(); ++i) | 
					
						
							|  |  |  |         process(klass->protocolAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < klass->memberCount(); ++i) | 
					
						
							|  |  |  |         process(klass->memberAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _currentClassOrNamespace = previous; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCBaseClass *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |     if (ClassOrNamespace *base = _globalNamespace->lookupType(b->name())) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         _currentClassOrNamespace->addUsing(base); | 
					
						
							|  |  |  |     } else if (false) { | 
					
						
							|  |  |  |         Overview oo; | 
					
						
							|  |  |  |         qDebug() << "no entity for:" << oo(b->name()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCForwardClassDeclaration *klass) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  |     ClassOrNamespace *previous = enterGlobalClassOrNamespace(klass); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     _currentClassOrNamespace = previous; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCProtocol *proto) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  |     ClassOrNamespace *previous = enterGlobalClassOrNamespace(proto); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < proto->protocolCount(); ++i) | 
					
						
							|  |  |  |         process(proto->protocolAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < proto->memberCount(); ++i) | 
					
						
							|  |  |  |         process(proto->memberAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _currentClassOrNamespace = previous; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCBaseProtocol *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 13:44:54 +02:00
										 |  |  |     if (ClassOrNamespace *base = _globalNamespace->lookupType(b->name())) { | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |         _currentClassOrNamespace->addUsing(base); | 
					
						
							|  |  |  |     } else if (false) { | 
					
						
							|  |  |  |         Overview oo; | 
					
						
							|  |  |  |         qDebug() << "no entity for:" << oo(b->name()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCForwardProtocolDeclaration *proto) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-11 10:12:49 +02:00
										 |  |  |     ClassOrNamespace *previous = enterGlobalClassOrNamespace(proto); | 
					
						
							| 
									
										
										
										
											2010-05-05 10:28:32 +02:00
										 |  |  |     _currentClassOrNamespace = previous; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CreateBindings::visit(ObjCMethod *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-16 11:03:39 +02:00
										 |  |  | 
 |