forked from qt-creator/qt-creator
Removed the CppBindings.
This commit is contained in:
@@ -1,866 +0,0 @@
|
|||||||
/**************************************************************************
|
|
||||||
**
|
|
||||||
** This file is part of Qt Creator
|
|
||||||
**
|
|
||||||
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
|
||||||
**
|
|
||||||
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
||||||
**
|
|
||||||
** Commercial Usage
|
|
||||||
**
|
|
||||||
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
||||||
** accordance with the Qt Commercial License Agreement provided with the
|
|
||||||
** Software or, alternatively, in accordance with the terms contained in
|
|
||||||
** a written agreement between you and Nokia.
|
|
||||||
**
|
|
||||||
** GNU Lesser General Public License Usage
|
|
||||||
**
|
|
||||||
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
||||||
** General Public License version 2.1 as published by the Free Software
|
|
||||||
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
||||||
** packaging of this file. Please review the following information to
|
|
||||||
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
||||||
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
||||||
**
|
|
||||||
** If you are unsure which license is appropriate for your use, please
|
|
||||||
** contact the sales department at http://qt.nokia.com/contact.
|
|
||||||
**
|
|
||||||
**************************************************************************/
|
|
||||||
|
|
||||||
#include "CppBindings.h"
|
|
||||||
#include "CppDocument.h"
|
|
||||||
#include "Overview.h"
|
|
||||||
|
|
||||||
#include <CoreTypes.h>
|
|
||||||
#include <Symbols.h>
|
|
||||||
#include <Literals.h>
|
|
||||||
#include <Names.h>
|
|
||||||
#include <Scope.h>
|
|
||||||
#include <Control.h>
|
|
||||||
#include <SymbolVisitor.h>
|
|
||||||
|
|
||||||
#include <QtDebug>
|
|
||||||
|
|
||||||
using namespace CPlusPlus;
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Location
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
Location::Location()
|
|
||||||
: _fileId(0),
|
|
||||||
_sourceLocation(0)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
Location::Location(Symbol *symbol)
|
|
||||||
: _fileId(symbol->fileId()),
|
|
||||||
_sourceLocation(symbol->sourceLocation())
|
|
||||||
{ }
|
|
||||||
|
|
||||||
Location::Location(const StringLiteral *fileId, unsigned sourceLocation)
|
|
||||||
: _fileId(fileId), _sourceLocation(sourceLocation)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// NamespaceBinding
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
NamespaceBinding::NamespaceBinding(NamespaceBinding *parent)
|
|
||||||
: parent(parent),
|
|
||||||
anonymousNamespaceBinding(0)
|
|
||||||
{
|
|
||||||
if (parent)
|
|
||||||
parent->children.append(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding::~NamespaceBinding()
|
|
||||||
{
|
|
||||||
qDeleteAll(QList<NamespaceBinding *>(children));
|
|
||||||
qDeleteAll(QList<ClassBinding *>(classBindings));
|
|
||||||
|
|
||||||
if (parent)
|
|
||||||
parent->asNamespaceBinding()->children.removeAll(this);
|
|
||||||
parent = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const NameId *NamespaceBinding::name() const
|
|
||||||
{
|
|
||||||
if (symbols.size()) {
|
|
||||||
if (const Name *name = symbols.first()->name()) {
|
|
||||||
const NameId *nameId = name->asNameId();
|
|
||||||
Q_ASSERT(nameId != 0);
|
|
||||||
|
|
||||||
return nameId;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const Identifier *NamespaceBinding::identifier() const
|
|
||||||
{
|
|
||||||
if (const NameId *nameId = name())
|
|
||||||
return nameId->identifier();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::globalNamespaceBinding()
|
|
||||||
{
|
|
||||||
NamespaceBinding *it = this;
|
|
||||||
|
|
||||||
for (; it; it = it->parent) {
|
|
||||||
if (! it->parent)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return it;
|
|
||||||
}
|
|
||||||
|
|
||||||
Binding *NamespaceBinding::findClassOrNamespaceBinding(const Identifier *id, QSet<Binding *> *processed)
|
|
||||||
{
|
|
||||||
if (processed->contains(this))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
processed->insert(this);
|
|
||||||
|
|
||||||
if (id->isEqualTo(identifier()))
|
|
||||||
return const_cast<NamespaceBinding *>(this);
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *nestedNamespaceBinding, children) {
|
|
||||||
if (id->isEqualTo(nestedNamespaceBinding->identifier()))
|
|
||||||
return nestedNamespaceBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (ClassBinding *classBinding, classBindings) {
|
|
||||||
if (id->isEqualTo(classBinding->identifier()))
|
|
||||||
return classBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *u, usings) {
|
|
||||||
if (Binding *b = u->findClassOrNamespaceBinding(id, processed))
|
|
||||||
return b;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parent)
|
|
||||||
return parent->findClassOrNamespaceBinding(id, processed);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding *NamespaceBinding::findClassBinding(const Name *name, QSet<Binding *> *processed)
|
|
||||||
{
|
|
||||||
if (! name)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (processed->contains(this))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (const QualifiedNameId *q = name->asQualifiedNameId()) {
|
|
||||||
Binding *current = this;
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < q->nameCount(); ++i) {
|
|
||||||
const Identifier *nameId = q->nameAt(i)->identifier();
|
|
||||||
if (! nameId)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
QSet<Binding *> visited;
|
|
||||||
Binding *binding = current->findClassOrNamespaceBinding(nameId, &visited); // ### TODO: check recursion.
|
|
||||||
if (! binding)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
current = binding;
|
|
||||||
}
|
|
||||||
|
|
||||||
return current->asClassBinding();
|
|
||||||
}
|
|
||||||
|
|
||||||
processed->insert(this);
|
|
||||||
|
|
||||||
const Identifier *id = name->identifier();
|
|
||||||
|
|
||||||
foreach (ClassBinding *classBinding, classBindings) {
|
|
||||||
if (id->isEqualTo(classBinding->identifier()))
|
|
||||||
return classBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parent)
|
|
||||||
return parent->findClassBinding(name, processed);
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *u, usings) {
|
|
||||||
if (ClassBinding *classBinding = u->findClassBinding(name, processed))
|
|
||||||
return classBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::findNamespaceBinding(const Name *name)
|
|
||||||
{
|
|
||||||
if (! name)
|
|
||||||
return anonymousNamespaceBinding;
|
|
||||||
|
|
||||||
else if (const NameId *nameId = name->asNameId())
|
|
||||||
return findNamespaceBindingForNameId(nameId, /*lookAtParent = */ true);
|
|
||||||
|
|
||||||
else if (const QualifiedNameId *q = name->asQualifiedNameId()) {
|
|
||||||
NamespaceBinding *current = this;
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < q->nameCount(); ++i) {
|
|
||||||
const NameId *namespaceName = q->nameAt(i)->asNameId();
|
|
||||||
if (! namespaceName)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
bool lookAtParent = false;
|
|
||||||
if (i == 0)
|
|
||||||
lookAtParent = true;
|
|
||||||
|
|
||||||
NamespaceBinding *binding = current->findNamespaceBindingForNameId(namespaceName, lookAtParent);
|
|
||||||
if (! binding)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
current = binding;
|
|
||||||
}
|
|
||||||
|
|
||||||
return current;
|
|
||||||
}
|
|
||||||
|
|
||||||
// invalid binding
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::findNamespaceBindingForNameId(const NameId *name,
|
|
||||||
bool lookAtParentNamespace)
|
|
||||||
{
|
|
||||||
QSet<NamespaceBinding *> processed;
|
|
||||||
return findNamespaceBindingForNameId_helper(name, lookAtParentNamespace, &processed);
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::findNamespaceBindingForNameId_helper(const NameId *name,
|
|
||||||
bool lookAtParentNamespace,
|
|
||||||
QSet<NamespaceBinding *> *processed)
|
|
||||||
{
|
|
||||||
if (processed->contains(this))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
processed->insert(this);
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *binding, children) {
|
|
||||||
const Name *bindingName = binding->name();
|
|
||||||
|
|
||||||
if (! bindingName)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (const NameId *bindingNameId = bindingName->asNameId()) {
|
|
||||||
if (name->isEqualTo(bindingNameId))
|
|
||||||
return binding;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *u, usings) {
|
|
||||||
if (NamespaceBinding *b = u->findNamespaceBindingForNameId_helper(name, lookAtParentNamespace, processed)) {
|
|
||||||
return b;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (lookAtParentNamespace && parent)
|
|
||||||
return parent->findNamespaceBindingForNameId_helper(name, lookAtParentNamespace, processed);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::findOrCreateNamespaceBinding(Namespace *symbol)
|
|
||||||
{
|
|
||||||
if (NamespaceBinding *binding = findNamespaceBinding(symbol->name())) {
|
|
||||||
int index = 0;
|
|
||||||
|
|
||||||
for (; index < binding->symbols.size(); ++index) {
|
|
||||||
Namespace *ns = binding->symbols.at(index);
|
|
||||||
|
|
||||||
if (ns == symbol)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (index == binding->symbols.size())
|
|
||||||
binding->symbols.append(symbol);
|
|
||||||
|
|
||||||
return binding;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *binding = new NamespaceBinding(this);
|
|
||||||
binding->symbols.append(symbol);
|
|
||||||
|
|
||||||
if (! symbol->name()) {
|
|
||||||
Q_ASSERT(! anonymousNamespaceBinding);
|
|
||||||
|
|
||||||
anonymousNamespaceBinding = binding;
|
|
||||||
}
|
|
||||||
|
|
||||||
return binding;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void closure(const Location &loc,
|
|
||||||
NamespaceBinding *binding, const Name *name,
|
|
||||||
QList<NamespaceBinding *> *bindings)
|
|
||||||
{
|
|
||||||
if (bindings->contains(binding))
|
|
||||||
return;
|
|
||||||
|
|
||||||
bindings->append(binding);
|
|
||||||
|
|
||||||
Q_ASSERT(name->isNameId());
|
|
||||||
|
|
||||||
const Identifier *id = name->asNameId()->identifier();
|
|
||||||
bool ignoreUsingDirectives = false;
|
|
||||||
|
|
||||||
foreach (Namespace *symbol, binding->symbols) {
|
|
||||||
Scope *scope = symbol->members();
|
|
||||||
|
|
||||||
for (Symbol *symbol = scope->lookat(id); symbol; symbol = symbol->next()) {
|
|
||||||
if (symbol->name() != name || ! symbol->isNamespace())
|
|
||||||
continue;
|
|
||||||
|
|
||||||
const Location l(symbol);
|
|
||||||
|
|
||||||
if (l.fileId() == loc.fileId() && l.sourceLocation() < loc.sourceLocation()) {
|
|
||||||
ignoreUsingDirectives = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ignoreUsingDirectives)
|
|
||||||
return;
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *u, binding->usings)
|
|
||||||
closure(loc, u, name, bindings);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::resolveNamespace(const Location &loc,
|
|
||||||
const Name *name,
|
|
||||||
bool lookAtParent)
|
|
||||||
{
|
|
||||||
if (! name)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
else if (const NameId *nameId = name->asNameId()) {
|
|
||||||
QList<NamespaceBinding *> bindings;
|
|
||||||
closure(loc, this, nameId, &bindings);
|
|
||||||
|
|
||||||
QList<NamespaceBinding *> results;
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *binding, bindings) {
|
|
||||||
if (NamespaceBinding *b = binding->findNamespaceBinding(nameId))
|
|
||||||
results.append(b);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (results.size() == 1)
|
|
||||||
return results.at(0);
|
|
||||||
|
|
||||||
else if (results.size() > 1) {
|
|
||||||
// ### FIXME: return 0;
|
|
||||||
return results.at(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (parent && lookAtParent)
|
|
||||||
return parent->resolveNamespace(loc, name);
|
|
||||||
|
|
||||||
} else if (const QualifiedNameId *q = name->asQualifiedNameId()) {
|
|
||||||
if (q->nameCount() == 1) {
|
|
||||||
Q_ASSERT(q->isGlobal());
|
|
||||||
|
|
||||||
return globalNamespaceBinding()->resolveNamespace(loc, q->nameAt(0));
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *current = this;
|
|
||||||
if (q->isGlobal())
|
|
||||||
current = globalNamespaceBinding();
|
|
||||||
|
|
||||||
current = current->resolveNamespace(loc, q->nameAt(0));
|
|
||||||
for (unsigned i = 1; current && i < q->nameCount(); ++i)
|
|
||||||
current = current->resolveNamespace(loc, q->nameAt(i), false);
|
|
||||||
|
|
||||||
return current;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ### rewrite me
|
|
||||||
QByteArray NamespaceBinding::qualifiedId() const
|
|
||||||
{
|
|
||||||
if (! parent)
|
|
||||||
return "<root>";
|
|
||||||
|
|
||||||
QByteArray s;
|
|
||||||
|
|
||||||
s.append(parent->qualifiedId());
|
|
||||||
s.append("::");
|
|
||||||
|
|
||||||
if (const Identifier *id = identifier())
|
|
||||||
s.append(id->chars(), id->size());
|
|
||||||
|
|
||||||
else
|
|
||||||
s.append("<anonymous>");
|
|
||||||
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ### rewrite me
|
|
||||||
QByteArray ClassBinding::qualifiedId() const
|
|
||||||
{
|
|
||||||
QByteArray s = parent->qualifiedId();
|
|
||||||
s += "::";
|
|
||||||
|
|
||||||
if (const Identifier *id = identifier())
|
|
||||||
s.append(id->chars(), id->size());
|
|
||||||
|
|
||||||
else
|
|
||||||
s.append("<anonymous>");
|
|
||||||
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
Binding *ClassBinding::findClassOrNamespaceBinding(const Identifier *id, QSet<Binding *> *processed)
|
|
||||||
{
|
|
||||||
if (id->isEqualTo(identifier()))
|
|
||||||
return this;
|
|
||||||
|
|
||||||
if (processed->contains(this))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
processed->insert(this);
|
|
||||||
|
|
||||||
foreach (ClassBinding *nestedClassBinding, children) {
|
|
||||||
if (id->isEqualTo(nestedClassBinding->identifier()))
|
|
||||||
return nestedClassBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (ClassBinding *baseClassBinding, baseClassBindings) {
|
|
||||||
if (! baseClassBinding)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
else if (Binding *b = baseClassBinding->findClassOrNamespaceBinding(id, processed))
|
|
||||||
return b;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parent)
|
|
||||||
return parent->findClassOrNamespaceBinding(id, processed);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding *ClassBinding::findClassBinding(const Name *name, QSet<Binding *> *processed)
|
|
||||||
{
|
|
||||||
if (! name)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (processed->contains(this))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
processed->insert(this);
|
|
||||||
|
|
||||||
if (const QualifiedNameId *q = name->asQualifiedNameId()) {
|
|
||||||
Binding *currentBinding = this;
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < q->nameCount() - 1; ++i) {
|
|
||||||
const Identifier *id = q->nameAt(i)->identifier();
|
|
||||||
if (! id)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
Binding *classOrNamespaceBinding = currentBinding->findClassOrNamespaceBinding(id, processed);
|
|
||||||
|
|
||||||
if (! classOrNamespaceBinding)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
currentBinding = classOrNamespaceBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentBinding)
|
|
||||||
return currentBinding->findClassBinding(q->unqualifiedNameId(), processed);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (const Identifier *id = name->identifier()) {
|
|
||||||
if (id->isEqualTo(identifier()))
|
|
||||||
return this;
|
|
||||||
|
|
||||||
foreach (ClassBinding *nestedClassBinding, children) {
|
|
||||||
if (const Identifier *nestedClassId = nestedClassBinding->identifier()) {
|
|
||||||
if (nestedClassId->isEqualTo(id))
|
|
||||||
return nestedClassBinding;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parent)
|
|
||||||
return parent->findClassBinding(name, processed);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int depth;
|
|
||||||
|
|
||||||
void NamespaceBinding::dump()
|
|
||||||
{
|
|
||||||
qDebug() << QByteArray(depth, ' ').constData() << "namespace" << qualifiedId().constData()
|
|
||||||
<< " # " << symbols.size();
|
|
||||||
|
|
||||||
++depth;
|
|
||||||
|
|
||||||
foreach (ClassBinding *classBinding, classBindings) {
|
|
||||||
classBinding->dump();
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *child, children) {
|
|
||||||
child->dump();
|
|
||||||
}
|
|
||||||
|
|
||||||
--depth;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ClassBinding::dump()
|
|
||||||
{
|
|
||||||
qDebug() << QByteArray(depth, ' ').constData() << "class" << qualifiedId().constData()
|
|
||||||
<< " # " << symbols.size();
|
|
||||||
|
|
||||||
++depth;
|
|
||||||
|
|
||||||
foreach (ClassBinding *classBinding, children) {
|
|
||||||
classBinding->dump();
|
|
||||||
}
|
|
||||||
|
|
||||||
--depth;
|
|
||||||
}
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// ClassBinding
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
ClassBinding::ClassBinding(NamespaceBinding *parent)
|
|
||||||
: parent(parent)
|
|
||||||
{
|
|
||||||
parent->classBindings.append(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding::ClassBinding(ClassBinding *parentClass)
|
|
||||||
: parent(parentClass)
|
|
||||||
{
|
|
||||||
parentClass->children.append(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding::~ClassBinding()
|
|
||||||
{
|
|
||||||
qDeleteAll(QList<ClassBinding *>(children));
|
|
||||||
if (NamespaceBinding *nsBinding = parent->asNamespaceBinding())
|
|
||||||
nsBinding->classBindings.removeAll(this);
|
|
||||||
if (ClassBinding *classBinding = parent->asClassBinding())
|
|
||||||
classBinding->children.removeAll(this);
|
|
||||||
parent = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const Name *ClassBinding::name() const
|
|
||||||
{
|
|
||||||
if (symbols.isEmpty())
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return symbols.first()->name();
|
|
||||||
}
|
|
||||||
|
|
||||||
const Identifier *ClassBinding::identifier() const
|
|
||||||
{
|
|
||||||
if (const Name *n = name())
|
|
||||||
return n->identifier();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Binder
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
class Binder: protected SymbolVisitor
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
Binder(NamespaceBinding *globals);
|
|
||||||
virtual ~Binder();
|
|
||||||
|
|
||||||
NamespaceBinding *operator()(Document::Ptr doc, const Snapshot &snapshot)
|
|
||||||
{
|
|
||||||
namespaceBinding = _globals;
|
|
||||||
const Snapshot previousSnapshot = _snapshot;
|
|
||||||
|
|
||||||
_snapshot = snapshot;
|
|
||||||
(void) bind(doc);
|
|
||||||
_snapshot = previousSnapshot;
|
|
||||||
|
|
||||||
return _globals;
|
|
||||||
}
|
|
||||||
|
|
||||||
Snapshot _snapshot;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
NamespaceBinding *bind(Document::Ptr doc)
|
|
||||||
{
|
|
||||||
QSet<QString> processed;
|
|
||||||
return bind(doc, &processed);
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *bind(Document::Ptr doc, QSet<QString> *processed)
|
|
||||||
{
|
|
||||||
if (processed->contains(doc->fileName()))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
processed->insert(doc->fileName());
|
|
||||||
|
|
||||||
foreach (const Document::Include &i, doc->includes()) {
|
|
||||||
if (Document::Ptr includedDoc = _snapshot.document(i.fileName())) {
|
|
||||||
/*NamepaceBinding *binding = */ bind(includedDoc, processed);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Namespace *ns = doc->globalNamespace();
|
|
||||||
_globals->symbols.append(ns);
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < ns->memberCount(); ++i) {
|
|
||||||
(void) bind(ns->memberAt(i), _globals);
|
|
||||||
}
|
|
||||||
|
|
||||||
return _globals;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *bind(Symbol *symbol, NamespaceBinding *binding);
|
|
||||||
NamespaceBinding *findOrCreateNamespaceBinding(Namespace *symbol);
|
|
||||||
NamespaceBinding *resolveNamespace(const Location &loc, const Name *name);
|
|
||||||
|
|
||||||
NamespaceBinding *switchNamespaceBinding(NamespaceBinding *binding);
|
|
||||||
|
|
||||||
ClassBinding *findOrCreateClassBinding(Class *classSymbol);
|
|
||||||
ClassBinding *findClassBinding(const Name *name);
|
|
||||||
|
|
||||||
ClassBinding *switchClassBinding(ClassBinding *binding);
|
|
||||||
|
|
||||||
using SymbolVisitor::visit;
|
|
||||||
|
|
||||||
virtual bool visit(Namespace *);
|
|
||||||
virtual bool visit(UsingNamespaceDirective *);
|
|
||||||
virtual bool visit(Class *);
|
|
||||||
virtual bool visit(Function *);
|
|
||||||
virtual bool visit(Block *);
|
|
||||||
|
|
||||||
private:
|
|
||||||
NamespaceBinding *_globals;
|
|
||||||
NamespaceBinding *namespaceBinding;
|
|
||||||
ClassBinding *classBinding;
|
|
||||||
};
|
|
||||||
|
|
||||||
Binder::Binder(NamespaceBinding *globals)
|
|
||||||
: _globals(globals),
|
|
||||||
namespaceBinding(0),
|
|
||||||
classBinding(0)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
Binder::~Binder()
|
|
||||||
{ }
|
|
||||||
|
|
||||||
NamespaceBinding *Binder::bind(Symbol *symbol, NamespaceBinding *binding)
|
|
||||||
{
|
|
||||||
NamespaceBinding *previousBinding = switchNamespaceBinding(binding);
|
|
||||||
accept(symbol);
|
|
||||||
return switchNamespaceBinding(previousBinding);
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *Binder::findOrCreateNamespaceBinding(Namespace *symbol)
|
|
||||||
{ return namespaceBinding->findOrCreateNamespaceBinding(symbol); }
|
|
||||||
|
|
||||||
NamespaceBinding *Binder::resolveNamespace(const Location &loc, const Name *name)
|
|
||||||
{
|
|
||||||
if (! namespaceBinding)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return namespaceBinding->resolveNamespace(loc, name);
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *Binder::switchNamespaceBinding(NamespaceBinding *binding)
|
|
||||||
{
|
|
||||||
NamespaceBinding *previousBinding = namespaceBinding;
|
|
||||||
namespaceBinding = binding;
|
|
||||||
return previousBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding *Binder::findOrCreateClassBinding(Class *classSymbol)
|
|
||||||
{
|
|
||||||
// ### FINISH ME
|
|
||||||
ClassBinding *binding = 0;
|
|
||||||
|
|
||||||
if (classBinding)
|
|
||||||
binding = new ClassBinding(classBinding);
|
|
||||||
else
|
|
||||||
binding = new ClassBinding(namespaceBinding);
|
|
||||||
|
|
||||||
binding->symbols.append(classSymbol);
|
|
||||||
return binding;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding *Binder::findClassBinding(const Name *name)
|
|
||||||
{
|
|
||||||
QSet<Binding *> processed;
|
|
||||||
|
|
||||||
if (classBinding) {
|
|
||||||
if (ClassBinding *k = classBinding->findClassBinding(name, &processed))
|
|
||||||
return k;
|
|
||||||
|
|
||||||
processed.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (namespaceBinding)
|
|
||||||
return namespaceBinding->findClassBinding(name, &processed);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding *Binder::switchClassBinding(ClassBinding *binding)
|
|
||||||
{
|
|
||||||
ClassBinding *previousClassBinding = classBinding;
|
|
||||||
classBinding = binding;
|
|
||||||
return previousClassBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Binder::visit(Namespace *symbol)
|
|
||||||
{
|
|
||||||
NamespaceBinding *binding = findOrCreateNamespaceBinding(symbol);
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < symbol->memberCount(); ++i) {
|
|
||||||
Symbol *member = symbol->memberAt(i);
|
|
||||||
|
|
||||||
bind(member, binding);
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Binder::visit(UsingNamespaceDirective *u)
|
|
||||||
{
|
|
||||||
NamespaceBinding *resolved = resolveNamespace(Location(u), u->name());
|
|
||||||
|
|
||||||
if (! resolved)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
namespaceBinding->usings.append(resolved);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Binder::visit(Class *classSymbol)
|
|
||||||
{
|
|
||||||
ClassBinding *binding = findOrCreateClassBinding(classSymbol);
|
|
||||||
ClassBinding *previousClassBinding = switchClassBinding(binding);
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < classSymbol->baseClassCount(); ++i) {
|
|
||||||
BaseClass *baseClass = classSymbol->baseClassAt(i);
|
|
||||||
ClassBinding *baseClassBinding = findClassBinding(baseClass->name());
|
|
||||||
binding->baseClassBindings.append(baseClassBinding);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (unsigned i = 0; i < classSymbol->memberCount(); ++i)
|
|
||||||
accept(classSymbol->memberAt(i));
|
|
||||||
|
|
||||||
(void) switchClassBinding(previousClassBinding);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Binder::visit(Function *)
|
|
||||||
{ return false; }
|
|
||||||
|
|
||||||
bool Binder::visit(Block *)
|
|
||||||
{ return false; }
|
|
||||||
|
|
||||||
} // end of anonymous namespace
|
|
||||||
|
|
||||||
static NamespaceBinding *find_helper(Namespace *symbol, NamespaceBinding *binding,
|
|
||||||
QSet<NamespaceBinding *> *processed)
|
|
||||||
{
|
|
||||||
if (binding && ! processed->contains(binding)) {
|
|
||||||
processed->insert(binding);
|
|
||||||
|
|
||||||
if (binding->symbols.contains(symbol))
|
|
||||||
return binding;
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *nestedBinding, binding->children) {
|
|
||||||
if (NamespaceBinding *ns = find_helper(symbol, nestedBinding, processed))
|
|
||||||
return ns;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (NamespaceBinding *a = find_helper(symbol, binding->anonymousNamespaceBinding, processed))
|
|
||||||
return a;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static ClassBinding *find_helper(Class *symbol, Binding *binding,
|
|
||||||
QSet<Binding *> *processed)
|
|
||||||
{
|
|
||||||
if (binding && ! processed->contains(binding)) {
|
|
||||||
processed->insert(binding);
|
|
||||||
|
|
||||||
if (NamespaceBinding *namespaceBinding = binding->asNamespaceBinding()) {
|
|
||||||
foreach (ClassBinding *classBinding, namespaceBinding->classBindings) {
|
|
||||||
if (ClassBinding *c = find_helper(symbol, classBinding, processed))
|
|
||||||
return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (NamespaceBinding *nestedBinding, namespaceBinding->children) {
|
|
||||||
if (ClassBinding *c = find_helper(symbol, nestedBinding, processed))
|
|
||||||
return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ClassBinding *a = find_helper(symbol, namespaceBinding->anonymousNamespaceBinding, processed))
|
|
||||||
return a;
|
|
||||||
|
|
||||||
} else if (ClassBinding *classBinding = binding->asClassBinding()) {
|
|
||||||
foreach (Class *klass, classBinding->symbols) {
|
|
||||||
if (klass == symbol)
|
|
||||||
return classBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (ClassBinding *nestedClassBinding, classBinding->children) {
|
|
||||||
if (ClassBinding *c = find_helper(symbol, nestedClassBinding, processed))
|
|
||||||
return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if 0 // ### FIXME
|
|
||||||
if (ClassBinding *a = find_helper(symbol, classBinding->anonymousClassBinding, processed))
|
|
||||||
return a;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBinding *NamespaceBinding::find(Namespace *symbol, NamespaceBinding *binding)
|
|
||||||
{
|
|
||||||
QSet<NamespaceBinding *> processed;
|
|
||||||
return find_helper(symbol, binding, &processed);
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassBinding *NamespaceBinding::find(Class *symbol, NamespaceBinding *binding)
|
|
||||||
{
|
|
||||||
QSet<Binding *> processed;
|
|
||||||
return find_helper(symbol, binding, &processed);
|
|
||||||
}
|
|
||||||
|
|
||||||
NamespaceBindingPtr CPlusPlus::bind(Document::Ptr doc, Snapshot snapshot)
|
|
||||||
{
|
|
||||||
NamespaceBindingPtr global(new NamespaceBinding());
|
|
||||||
|
|
||||||
Binder bind(global.data());
|
|
||||||
bind(doc, snapshot);
|
|
||||||
return global;
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,195 +0,0 @@
|
|||||||
/**************************************************************************
|
|
||||||
**
|
|
||||||
** This file is part of Qt Creator
|
|
||||||
**
|
|
||||||
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
|
||||||
**
|
|
||||||
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
||||||
**
|
|
||||||
** Commercial Usage
|
|
||||||
**
|
|
||||||
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
||||||
** accordance with the Qt Commercial License Agreement provided with the
|
|
||||||
** Software or, alternatively, in accordance with the terms contained in
|
|
||||||
** a written agreement between you and Nokia.
|
|
||||||
**
|
|
||||||
** GNU Lesser General Public License Usage
|
|
||||||
**
|
|
||||||
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
||||||
** General Public License version 2.1 as published by the Free Software
|
|
||||||
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
||||||
** packaging of this file. Please review the following information to
|
|
||||||
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
||||||
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
||||||
**
|
|
||||||
** If you are unsure which license is appropriate for your use, please
|
|
||||||
** contact the sales department at http://qt.nokia.com/contact.
|
|
||||||
**
|
|
||||||
**************************************************************************/
|
|
||||||
|
|
||||||
#ifndef CPPBINDINGS_H
|
|
||||||
#define CPPBINDINGS_H
|
|
||||||
|
|
||||||
#include "CppDocument.h"
|
|
||||||
|
|
||||||
#include <QtCore/QList>
|
|
||||||
#include <QtCore/QSharedPointer>
|
|
||||||
#include <QtCore/QString>
|
|
||||||
#include <QtCore/QByteArray>
|
|
||||||
|
|
||||||
namespace CPlusPlus {
|
|
||||||
|
|
||||||
class Location;
|
|
||||||
class Binding;
|
|
||||||
class NamespaceBinding;
|
|
||||||
class ClassBinding;
|
|
||||||
|
|
||||||
typedef QSharedPointer<Binding> BindingPtr;
|
|
||||||
typedef QSharedPointer<ClassBinding> ClassBindingPtr;
|
|
||||||
typedef QSharedPointer<NamespaceBinding> NamespaceBindingPtr;
|
|
||||||
|
|
||||||
class CPLUSPLUS_EXPORT Location
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
Location();
|
|
||||||
Location(Symbol *symbol);
|
|
||||||
Location(const StringLiteral *fileId, unsigned sourceLocation);
|
|
||||||
|
|
||||||
inline bool isValid() const
|
|
||||||
{ return _fileId != 0; }
|
|
||||||
|
|
||||||
inline operator bool() const
|
|
||||||
{ return _fileId != 0; }
|
|
||||||
|
|
||||||
inline const StringLiteral *fileId() const
|
|
||||||
{ return _fileId; }
|
|
||||||
|
|
||||||
inline unsigned sourceLocation() const
|
|
||||||
{ return _sourceLocation; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
const StringLiteral *_fileId;
|
|
||||||
unsigned _sourceLocation;
|
|
||||||
};
|
|
||||||
|
|
||||||
class CPLUSPLUS_EXPORT Binding
|
|
||||||
{
|
|
||||||
Q_DISABLE_COPY(Binding)
|
|
||||||
|
|
||||||
public:
|
|
||||||
Binding() {}
|
|
||||||
virtual ~Binding() {}
|
|
||||||
|
|
||||||
virtual QByteArray qualifiedId() const = 0;
|
|
||||||
virtual NamespaceBinding *asNamespaceBinding() { return 0; }
|
|
||||||
virtual ClassBinding *asClassBinding() { return 0; }
|
|
||||||
|
|
||||||
virtual ClassBinding *findClassBinding(const Name *name, QSet<Binding *> *processed) = 0;
|
|
||||||
virtual Binding *findClassOrNamespaceBinding(const Identifier *id, QSet<Binding *> *processed) = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
class CPLUSPLUS_EXPORT NamespaceBinding: public Binding
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
/// Constructs a binding with the given parent.
|
|
||||||
NamespaceBinding(NamespaceBinding *parent = 0);
|
|
||||||
|
|
||||||
/// Destroys the binding.
|
|
||||||
virtual ~NamespaceBinding();
|
|
||||||
|
|
||||||
/// Returns this binding's name.
|
|
||||||
const NameId *name() const;
|
|
||||||
|
|
||||||
/// Returns this binding's identifier.
|
|
||||||
const Identifier *identifier() const;
|
|
||||||
|
|
||||||
/// Returns the binding for the global namespace (aka ::).
|
|
||||||
NamespaceBinding *globalNamespaceBinding();
|
|
||||||
|
|
||||||
/// Returns the binding for the given namespace symbol.
|
|
||||||
NamespaceBinding *findNamespaceBinding(const Name *name);
|
|
||||||
|
|
||||||
/// Returns the binding associated with the given symbol.
|
|
||||||
NamespaceBinding *findOrCreateNamespaceBinding(Namespace *symbol);
|
|
||||||
|
|
||||||
NamespaceBinding *resolveNamespace(const Location &loc,
|
|
||||||
const Name *name,
|
|
||||||
bool lookAtParent = true);
|
|
||||||
|
|
||||||
virtual ClassBinding *findClassBinding(const Name *name, QSet<Binding *> *processed);
|
|
||||||
virtual Binding *findClassOrNamespaceBinding(const Identifier *id, QSet<Binding *> *processed);
|
|
||||||
|
|
||||||
/// Helpers.
|
|
||||||
virtual QByteArray qualifiedId() const;
|
|
||||||
void dump();
|
|
||||||
|
|
||||||
virtual NamespaceBinding *asNamespaceBinding() { return this; }
|
|
||||||
|
|
||||||
static NamespaceBinding *find(Namespace *symbol, NamespaceBinding *binding);
|
|
||||||
static ClassBinding *find(Class *symbol, NamespaceBinding *binding);
|
|
||||||
|
|
||||||
private:
|
|
||||||
NamespaceBinding *findNamespaceBindingForNameId(const NameId *name,
|
|
||||||
bool lookAtParentNamespace);
|
|
||||||
|
|
||||||
NamespaceBinding *findNamespaceBindingForNameId_helper(const NameId *name,
|
|
||||||
bool lookAtParentNamespace,
|
|
||||||
QSet<NamespaceBinding *> *processed);
|
|
||||||
|
|
||||||
public: // attributes
|
|
||||||
/// This binding's parent.
|
|
||||||
NamespaceBinding *parent;
|
|
||||||
|
|
||||||
/// Binding for anonymous namespace symbols.
|
|
||||||
NamespaceBinding *anonymousNamespaceBinding;
|
|
||||||
|
|
||||||
/// This binding's connections.
|
|
||||||
QList<NamespaceBinding *> children;
|
|
||||||
|
|
||||||
/// This binding's list of using namespaces.
|
|
||||||
QList<NamespaceBinding *> usings;
|
|
||||||
|
|
||||||
/// This binding's namespace symbols.
|
|
||||||
QList<Namespace *> symbols;
|
|
||||||
|
|
||||||
QList<ClassBinding *> classBindings;
|
|
||||||
};
|
|
||||||
|
|
||||||
class CPLUSPLUS_EXPORT ClassBinding: public Binding
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
ClassBinding(NamespaceBinding *parent);
|
|
||||||
ClassBinding(ClassBinding *parentClass);
|
|
||||||
virtual ~ClassBinding();
|
|
||||||
|
|
||||||
virtual ClassBinding *asClassBinding() { return this; }
|
|
||||||
|
|
||||||
/// Returns this binding's name.
|
|
||||||
const Name *name() const;
|
|
||||||
|
|
||||||
/// Returns this binding's identifier.
|
|
||||||
const Identifier *identifier() const;
|
|
||||||
virtual QByteArray qualifiedId() const;
|
|
||||||
|
|
||||||
virtual ClassBinding *findClassBinding(const Name *name, QSet<Binding *> *processed);
|
|
||||||
virtual Binding *findClassOrNamespaceBinding(const Identifier *id, QSet<Binding *> *processed);
|
|
||||||
|
|
||||||
void dump();
|
|
||||||
|
|
||||||
public: // attributes
|
|
||||||
Binding *parent;
|
|
||||||
|
|
||||||
QList<ClassBinding *> children;
|
|
||||||
|
|
||||||
/// This binding's class symbols.
|
|
||||||
QList<Class *> symbols;
|
|
||||||
|
|
||||||
/// Bindings for the base classes.
|
|
||||||
QList<ClassBinding *> baseClassBindings;
|
|
||||||
};
|
|
||||||
|
|
||||||
CPLUSPLUS_EXPORT NamespaceBindingPtr bind(Document::Ptr doc, Snapshot snapshot);
|
|
||||||
|
|
||||||
} // end of namespace CPlusPlus
|
|
||||||
|
|
||||||
#endif // CPPBINDINGS_H
|
|
||||||
@@ -30,7 +30,6 @@
|
|||||||
#include "DeprecatedLookupContext.h"
|
#include "DeprecatedLookupContext.h"
|
||||||
#include "ResolveExpression.h"
|
#include "ResolveExpression.h"
|
||||||
#include "Overview.h"
|
#include "Overview.h"
|
||||||
#include "CppBindings.h"
|
|
||||||
|
|
||||||
#include <CoreTypes.h>
|
#include <CoreTypes.h>
|
||||||
#include <Symbols.h>
|
#include <Symbols.h>
|
||||||
@@ -623,142 +622,3 @@ void DeprecatedLookupContext::expand(Scope *scope,
|
|||||||
expandObjCClass(objcKlass, visibleScopes, expandedScopes);
|
expandObjCClass(objcKlass, visibleScopes, expandedScopes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void visibleClassBindings_helper(ClassBinding *classBinding,
|
|
||||||
QList<ClassBinding *> *allClassBindings,
|
|
||||||
QSet<ClassBinding *> *processed)
|
|
||||||
{
|
|
||||||
if (! classBinding)
|
|
||||||
return;
|
|
||||||
|
|
||||||
else if (processed->contains(classBinding))
|
|
||||||
return;
|
|
||||||
|
|
||||||
processed->insert(classBinding);
|
|
||||||
|
|
||||||
foreach (ClassBinding *baseClassBinding, classBinding->baseClassBindings)
|
|
||||||
visibleClassBindings_helper(baseClassBinding, allClassBindings, processed);
|
|
||||||
|
|
||||||
allClassBindings->append(classBinding);
|
|
||||||
}
|
|
||||||
|
|
||||||
static QList<ClassBinding *> visibleClassBindings(Symbol *symbol, NamespaceBinding *globalNamespace)
|
|
||||||
{
|
|
||||||
QList<ClassBinding *> classBindings;
|
|
||||||
|
|
||||||
if (! symbol)
|
|
||||||
return classBindings;
|
|
||||||
|
|
||||||
else if (Class *klass = symbol->asClass()) {
|
|
||||||
QSet<ClassBinding *> processed;
|
|
||||||
|
|
||||||
visibleClassBindings_helper(NamespaceBinding::find(klass, globalNamespace),
|
|
||||||
&classBindings, &processed);
|
|
||||||
}
|
|
||||||
|
|
||||||
return classBindings;
|
|
||||||
}
|
|
||||||
|
|
||||||
Symbol *DeprecatedLookupContext::canonicalSymbol(Symbol *symbol,
|
|
||||||
NamespaceBinding *globalNamespace)
|
|
||||||
{
|
|
||||||
Symbol *canonicalSymbol = DeprecatedLookupContext::canonicalSymbol(symbol);
|
|
||||||
if (! canonicalSymbol)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (const Identifier *symbolId = canonicalSymbol->identifier()) {
|
|
||||||
if (symbolId && canonicalSymbol->type()->isFunctionType()) {
|
|
||||||
Class *enclosingClass = canonicalSymbol->scope()->owner()->asClass();
|
|
||||||
const QList<ClassBinding *> classBindings = visibleClassBindings(enclosingClass, globalNamespace);
|
|
||||||
|
|
||||||
foreach (ClassBinding *baseClassBinding, classBindings) {
|
|
||||||
if (! baseClassBinding)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
foreach (Class *baseClass, baseClassBinding->symbols) {
|
|
||||||
if (! baseClass)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
for (Symbol *c = baseClass->members()->lookat(symbolId); c; c = c->next()) {
|
|
||||||
if (! symbolId->isEqualTo(c->identifier()))
|
|
||||||
continue;
|
|
||||||
else if (Function *f = c->type()->asFunctionType()) {
|
|
||||||
if (f->isVirtual())
|
|
||||||
return DeprecatedLookupContext::canonicalSymbol(f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return canonicalSymbol;
|
|
||||||
}
|
|
||||||
|
|
||||||
Symbol *DeprecatedLookupContext::canonicalSymbol(const QList<Symbol *> &candidates,
|
|
||||||
NamespaceBinding *globalNamespaceBinding)
|
|
||||||
{
|
|
||||||
if (candidates.isEmpty())
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return canonicalSymbol(candidates.first(), globalNamespaceBinding);
|
|
||||||
}
|
|
||||||
|
|
||||||
Symbol *DeprecatedLookupContext::canonicalSymbol(const QList<LookupItem> &results,
|
|
||||||
NamespaceBinding *globalNamespaceBinding)
|
|
||||||
{
|
|
||||||
QList<Symbol *> candidates;
|
|
||||||
|
|
||||||
foreach (const LookupItem &result, results)
|
|
||||||
candidates.append(result.declaration()); // ### not exactly.
|
|
||||||
|
|
||||||
return canonicalSymbol(candidates, globalNamespaceBinding);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Symbol *DeprecatedLookupContext::canonicalSymbol(Symbol *symbol)
|
|
||||||
{
|
|
||||||
Symbol *canonical = symbol;
|
|
||||||
Class *canonicalClass = 0;
|
|
||||||
ObjCClass *canonicalObjCClass = 0;
|
|
||||||
ObjCProtocol *canonicalObjCProto = 0;
|
|
||||||
|
|
||||||
for (; symbol; symbol = symbol->next()) {
|
|
||||||
if (symbol->identifier() == canonical->identifier()) {
|
|
||||||
canonical = symbol;
|
|
||||||
|
|
||||||
if (Class *klass = symbol->asClass())
|
|
||||||
canonicalClass = klass;
|
|
||||||
else if (ObjCClass *clazz = symbol->asObjCClass())
|
|
||||||
canonicalObjCClass = clazz;
|
|
||||||
else if (ObjCProtocol *proto = symbol->asObjCProtocol())
|
|
||||||
canonicalObjCProto = proto;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (canonicalClass) {
|
|
||||||
Q_ASSERT(canonical != 0);
|
|
||||||
|
|
||||||
if (canonical->isForwardClassDeclaration())
|
|
||||||
return canonicalClass; // prefer class declarations when available.
|
|
||||||
} else if (canonicalObjCClass) {
|
|
||||||
Q_ASSERT(canonical != 0);
|
|
||||||
|
|
||||||
if (canonical->isObjCForwardClassDeclaration())
|
|
||||||
return canonicalObjCClass;
|
|
||||||
} else if (canonicalObjCProto) {
|
|
||||||
Q_ASSERT(canonical != 0);
|
|
||||||
|
|
||||||
if (canonical->isObjCForwardProtocolDeclaration())
|
|
||||||
return canonicalObjCProto;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (canonical && canonical->scope()->isClassScope()) {
|
|
||||||
Class *enclosingClass = canonical->scope()->owner()->asClass();
|
|
||||||
|
|
||||||
if (enclosingClass->identifier() == canonical->identifier())
|
|
||||||
return enclosingClass;
|
|
||||||
}
|
|
||||||
|
|
||||||
return canonical;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -57,15 +57,6 @@ public:
|
|||||||
Document::Ptr document(const QString &fileName) const;
|
Document::Ptr document(const QString &fileName) const;
|
||||||
Snapshot snapshot() const;
|
Snapshot snapshot() const;
|
||||||
|
|
||||||
static Symbol *canonicalSymbol(const QList<Symbol *> &candidates,
|
|
||||||
NamespaceBinding *globalNamespaceBinding);
|
|
||||||
|
|
||||||
static Symbol *canonicalSymbol(Symbol *symbol,
|
|
||||||
NamespaceBinding *globalNamespaceBinding);
|
|
||||||
|
|
||||||
static Symbol *canonicalSymbol(const QList<LookupItem> &candidates,
|
|
||||||
NamespaceBinding *globalNamespaceBinding);
|
|
||||||
|
|
||||||
QList<Symbol *> resolve(const Name *name) const
|
QList<Symbol *> resolve(const Name *name) const
|
||||||
{ return resolve(name, visibleScopes()); }
|
{ return resolve(name, visibleScopes()); }
|
||||||
|
|
||||||
@@ -157,7 +148,6 @@ public:
|
|||||||
QList<Scope *> *expandedScopes) const;
|
QList<Scope *> *expandedScopes) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static Symbol *canonicalSymbol(Symbol *symbol);
|
|
||||||
|
|
||||||
QList<Symbol *> resolveQualifiedNameId(const QualifiedNameId *q,
|
QList<Symbol *> resolveQualifiedNameId(const QualifiedNameId *q,
|
||||||
const QList<Scope *> &visibleScopes,
|
const QList<Scope *> &visibleScopes,
|
||||||
|
|||||||
@@ -39,6 +39,7 @@
|
|||||||
#include <TranslationUnit.h>
|
#include <TranslationUnit.h>
|
||||||
|
|
||||||
#include <QtCore/QDir>
|
#include <QtCore/QDir>
|
||||||
|
#include <QtCore/QDebug>
|
||||||
|
|
||||||
using namespace CPlusPlus;
|
using namespace CPlusPlus;
|
||||||
|
|
||||||
@@ -56,11 +57,6 @@ FindUsages::FindUsages(Document::Ptr doc, const Snapshot &snapshot)
|
|||||||
typeofExpression.init(_doc, _snapshot, _context.bindings());
|
typeofExpression.init(_doc, _snapshot, _context.bindings());
|
||||||
}
|
}
|
||||||
|
|
||||||
void FindUsages::setGlobalNamespaceBinding(NamespaceBindingPtr globalNamespaceBinding)
|
|
||||||
{
|
|
||||||
_globalNamespaceBinding = globalNamespaceBinding;
|
|
||||||
}
|
|
||||||
|
|
||||||
QList<Usage> FindUsages::usages() const
|
QList<Usage> FindUsages::usages() const
|
||||||
{ return _usages; }
|
{ return _usages; }
|
||||||
|
|
||||||
@@ -153,61 +149,12 @@ void FindUsages::reportResult(unsigned tokenIndex)
|
|||||||
|
|
||||||
bool FindUsages::checkCandidates(const QList<Symbol *> &candidates) const
|
bool FindUsages::checkCandidates(const QList<Symbol *> &candidates) const
|
||||||
{
|
{
|
||||||
if (Symbol *canonicalSymbol = DeprecatedLookupContext::canonicalSymbol(candidates, _globalNamespaceBinding.data())) {
|
if (ClassOrNamespace *c = _context.lookupType(_declSymbol)) {
|
||||||
|
for (int i = candidates.size() - 1; i != -1; --i) {
|
||||||
#if 0
|
Symbol *s = candidates.at(i);
|
||||||
Symbol *c = candidates.first();
|
if (_context.lookupType(s) == c)
|
||||||
qDebug() << "*** canonical symbol:" << canonicalSymbol->fileName()
|
|
||||||
<< canonicalSymbol->line() << canonicalSymbol->column()
|
|
||||||
<< "candidates:" << candidates.size()
|
|
||||||
<< c->fileName() << c->line() << c->column();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return checkSymbol(canonicalSymbol);
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool FindUsages::checkScope(Symbol *symbol, Symbol *otherSymbol) const
|
|
||||||
{
|
|
||||||
if (! (symbol && otherSymbol))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
else if (symbol->scope() == otherSymbol->scope())
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
else if (symbol->name() && otherSymbol->name()) {
|
|
||||||
|
|
||||||
if (! symbol->name()->isEqualTo(otherSymbol->name()))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
} else if (symbol->name() != otherSymbol->name()) {
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return checkScope(symbol->enclosingSymbol(), otherSymbol->enclosingSymbol());
|
|
||||||
}
|
|
||||||
|
|
||||||
bool FindUsages::checkSymbol(Symbol *symbol) const
|
|
||||||
{
|
|
||||||
if (! symbol) {
|
|
||||||
return false;
|
|
||||||
|
|
||||||
} else if (symbol == _declSymbol) {
|
|
||||||
return true;
|
|
||||||
|
|
||||||
} else if (symbol->line() == _declSymbol->line() && symbol->column() == _declSymbol->column()) {
|
|
||||||
if (! qstrcmp(symbol->fileName(), _declSymbol->fileName()))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
} else if (symbol->isForwardClassDeclaration() && (_declSymbol->isClass() ||
|
|
||||||
_declSymbol->isForwardClassDeclaration())) {
|
|
||||||
return checkScope(symbol, _declSymbol);
|
|
||||||
|
|
||||||
} else if (_declSymbol->isForwardClassDeclaration() && (symbol->isClass() ||
|
|
||||||
symbol->isForwardClassDeclaration())) {
|
|
||||||
return checkScope(symbol, _declSymbol);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@@ -32,7 +32,6 @@
|
|||||||
|
|
||||||
#include "LookupContext.h"
|
#include "LookupContext.h"
|
||||||
#include "CppDocument.h"
|
#include "CppDocument.h"
|
||||||
#include "CppBindings.h"
|
|
||||||
#include "Semantic.h"
|
#include "Semantic.h"
|
||||||
#include "TypeOfExpression.h"
|
#include "TypeOfExpression.h"
|
||||||
#include <ASTVisitor.h>
|
#include <ASTVisitor.h>
|
||||||
@@ -62,8 +61,6 @@ class CPLUSPLUS_EXPORT FindUsages: protected ASTVisitor
|
|||||||
public:
|
public:
|
||||||
FindUsages(Document::Ptr doc, const Snapshot &snapshot);
|
FindUsages(Document::Ptr doc, const Snapshot &snapshot);
|
||||||
|
|
||||||
void setGlobalNamespaceBinding(NamespaceBindingPtr globalNamespaceBinding);
|
|
||||||
|
|
||||||
void operator()(Symbol *symbol);
|
void operator()(Symbol *symbol);
|
||||||
|
|
||||||
QList<Usage> usages() const;
|
QList<Usage> usages() const;
|
||||||
@@ -79,9 +76,7 @@ protected:
|
|||||||
void reportResult(unsigned tokenIndex, const QList<Symbol *> &candidates);
|
void reportResult(unsigned tokenIndex, const QList<Symbol *> &candidates);
|
||||||
void reportResult(unsigned tokenIndex);
|
void reportResult(unsigned tokenIndex);
|
||||||
|
|
||||||
bool checkSymbol(Symbol *symbol) const;
|
|
||||||
bool checkCandidates(const QList<Symbol *> &candidates) const;
|
bool checkCandidates(const QList<Symbol *> &candidates) const;
|
||||||
bool checkScope(Symbol *symbol, Symbol *otherSymbol) const;
|
|
||||||
void checkExpression(unsigned startToken, unsigned endToken);
|
void checkExpression(unsigned startToken, unsigned endToken);
|
||||||
|
|
||||||
void ensureNameIsValid(NameAST *ast);
|
void ensureNameIsValid(NameAST *ast);
|
||||||
@@ -112,7 +107,6 @@ private:
|
|||||||
QByteArray _source;
|
QByteArray _source;
|
||||||
Document::Ptr _exprDoc;
|
Document::Ptr _exprDoc;
|
||||||
Semantic _sem;
|
Semantic _sem;
|
||||||
NamespaceBindingPtr _globalNamespaceBinding;
|
|
||||||
QList<PostfixExpressionAST *> _postfixExpressionStack;
|
QList<PostfixExpressionAST *> _postfixExpressionStack;
|
||||||
QList<QualifiedNameAST *> _qualifiedNameStack;
|
QList<QualifiedNameAST *> _qualifiedNameStack;
|
||||||
QList<int> _references;
|
QList<int> _references;
|
||||||
|
|||||||
@@ -30,7 +30,6 @@
|
|||||||
#include "LookupContext.h"
|
#include "LookupContext.h"
|
||||||
#include "ResolveExpression.h"
|
#include "ResolveExpression.h"
|
||||||
#include "Overview.h"
|
#include "Overview.h"
|
||||||
#include "CppBindings.h"
|
|
||||||
#include "DeprecatedGenTemplateInstance.h"
|
#include "DeprecatedGenTemplateInstance.h"
|
||||||
|
|
||||||
#include <CoreTypes.h>
|
#include <CoreTypes.h>
|
||||||
@@ -186,10 +185,8 @@ QList<Symbol *> LookupContext::lookup(const Name *name, Scope *scope) const
|
|||||||
if (! name)
|
if (! name)
|
||||||
return candidates;
|
return candidates;
|
||||||
|
|
||||||
const Identifier *id = name->identifier();
|
|
||||||
|
|
||||||
for (; scope; scope = scope->enclosingScope()) {
|
for (; scope; scope = scope->enclosingScope()) {
|
||||||
if (id && scope->isBlockScope()) {
|
if ((name->isNameId() || name->isTemplateNameId()) && scope->isBlockScope()) {
|
||||||
bindings()->lookupInScope(name, scope, &candidates, /*templateId = */ 0);
|
bindings()->lookupInScope(name, scope, &candidates, /*templateId = */ 0);
|
||||||
|
|
||||||
if (! candidates.isEmpty())
|
if (! candidates.isEmpty())
|
||||||
|
|||||||
@@ -37,7 +37,6 @@ HEADERS += \
|
|||||||
$$PWD/LookupItem.h \
|
$$PWD/LookupItem.h \
|
||||||
$$PWD/LookupContext.h \
|
$$PWD/LookupContext.h \
|
||||||
$$PWD/DeprecatedLookupContext.h \
|
$$PWD/DeprecatedLookupContext.h \
|
||||||
$$PWD/CppBindings.h \
|
|
||||||
$$PWD/ASTParent.h \
|
$$PWD/ASTParent.h \
|
||||||
$$PWD/DeprecatedGenTemplateInstance.h \
|
$$PWD/DeprecatedGenTemplateInstance.h \
|
||||||
$$PWD/FindUsages.h \
|
$$PWD/FindUsages.h \
|
||||||
@@ -64,7 +63,6 @@ SOURCES += \
|
|||||||
$$PWD/LookupItem.cpp \
|
$$PWD/LookupItem.cpp \
|
||||||
$$PWD/LookupContext.cpp \
|
$$PWD/LookupContext.cpp \
|
||||||
$$PWD/DeprecatedLookupContext.cpp \
|
$$PWD/DeprecatedLookupContext.cpp \
|
||||||
$$PWD/CppBindings.cpp \
|
|
||||||
$$PWD/ASTParent.cpp \
|
$$PWD/ASTParent.cpp \
|
||||||
$$PWD/DeprecatedGenTemplateInstance.cpp \
|
$$PWD/DeprecatedGenTemplateInstance.cpp \
|
||||||
$$PWD/FindUsages.cpp \
|
$$PWD/FindUsages.cpp \
|
||||||
|
|||||||
@@ -48,7 +48,6 @@
|
|||||||
#include <TranslationUnit.h>
|
#include <TranslationUnit.h>
|
||||||
#include <cplusplus/ExpressionUnderCursor.h>
|
#include <cplusplus/ExpressionUnderCursor.h>
|
||||||
#include <cplusplus/TypeOfExpression.h>
|
#include <cplusplus/TypeOfExpression.h>
|
||||||
#include <cplusplus/DeprecatedLookupContext.h>
|
|
||||||
#include <cplusplus/Overview.h>
|
#include <cplusplus/Overview.h>
|
||||||
#include <cplusplus/OverviewModel.h>
|
#include <cplusplus/OverviewModel.h>
|
||||||
#include <cplusplus/SimpleLexer.h>
|
#include <cplusplus/SimpleLexer.h>
|
||||||
@@ -56,7 +55,6 @@
|
|||||||
#include <cplusplus/MatchingText.h>
|
#include <cplusplus/MatchingText.h>
|
||||||
#include <cplusplus/BackwardsScanner.h>
|
#include <cplusplus/BackwardsScanner.h>
|
||||||
#include <cplusplus/FastPreprocessor.h>
|
#include <cplusplus/FastPreprocessor.h>
|
||||||
#include <cplusplus/CppBindings.h>
|
|
||||||
#include <cplusplus/CheckUndefinedSymbols.h>
|
#include <cplusplus/CheckUndefinedSymbols.h>
|
||||||
|
|
||||||
#include <cpptools/cppmodelmanagerinterface.h>
|
#include <cpptools/cppmodelmanagerinterface.h>
|
||||||
@@ -846,13 +844,17 @@ CPlusPlus::Symbol *CPPEditor::findCanonicalSymbol(const QTextCursor &cursor,
|
|||||||
TypeOfExpression typeOfExpression;
|
TypeOfExpression typeOfExpression;
|
||||||
typeOfExpression.init(doc, snapshot);
|
typeOfExpression.init(doc, snapshot);
|
||||||
|
|
||||||
const QList<LookupItem> results = typeOfExpression(code, doc->scopeAt(line, col),
|
Scope *scope = doc->scopeAt(line, col);
|
||||||
TypeOfExpression::Preprocess);
|
|
||||||
|
|
||||||
NamespaceBindingPtr glo = bind(doc, snapshot);
|
const QList<LookupItem> results = typeOfExpression(code, scope, TypeOfExpression::Preprocess);
|
||||||
Symbol *canonicalSymbol = DeprecatedLookupContext::canonicalSymbol(results, glo.data());
|
for (int i = results.size() - 1; i != -1; --i) { // ### TODO virtual methods and classes.
|
||||||
|
const LookupItem &r = results.at(i);
|
||||||
|
|
||||||
return canonicalSymbol;
|
if (r.declaration())
|
||||||
|
return r.declaration();
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Macro *CPPEditor::findCanonicalMacro(const QTextCursor &cursor,
|
const Macro *CPPEditor::findCanonicalMacro(const QTextCursor &cursor,
|
||||||
@@ -934,13 +936,13 @@ void CPPEditor::renameUsagesNow()
|
|||||||
|
|
||||||
Symbol *CPPEditor::markSymbols()
|
Symbol *CPPEditor::markSymbols()
|
||||||
{
|
{
|
||||||
updateSemanticInfo(m_semanticHighlighter->semanticInfo(currentSource()));
|
//updateSemanticInfo(m_semanticHighlighter->semanticInfo(currentSource()));
|
||||||
|
|
||||||
abortRename();
|
abortRename();
|
||||||
|
|
||||||
QList<QTextEdit::ExtraSelection> selections;
|
QList<QTextEdit::ExtraSelection> selections;
|
||||||
|
|
||||||
SemanticInfo info = m_lastSemanticInfo;
|
const SemanticInfo info = m_lastSemanticInfo;
|
||||||
|
|
||||||
Symbol *canonicalSymbol = findCanonicalSymbol(textCursor(), info.doc, info.snapshot);
|
Symbol *canonicalSymbol = findCanonicalSymbol(textCursor(), info.doc, info.snapshot);
|
||||||
if (canonicalSymbol) {
|
if (canonicalSymbol) {
|
||||||
@@ -1916,7 +1918,11 @@ void CPPEditor::updateSemanticInfo(const SemanticInfo &semanticInfo)
|
|||||||
}
|
}
|
||||||
|
|
||||||
setExtraSelections(UnusedSymbolSelection, unusedSelections);
|
setExtraSelections(UnusedSymbolSelection, unusedSelections);
|
||||||
|
|
||||||
|
if (! m_renameSelections.isEmpty())
|
||||||
setExtraSelections(CodeSemanticsSelection, m_renameSelections); // ###
|
setExtraSelections(CodeSemanticsSelection, m_renameSelections); // ###
|
||||||
|
else
|
||||||
|
markSymbols();
|
||||||
|
|
||||||
m_lastSemanticInfo.forced = false; // clear the forced flag
|
m_lastSemanticInfo.forced = false; // clear the forced flag
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,7 +51,6 @@
|
|||||||
#include <Scope.h>
|
#include <Scope.h>
|
||||||
|
|
||||||
#include <cplusplus/CppDocument.h>
|
#include <cplusplus/CppDocument.h>
|
||||||
#include <cplusplus/CppBindings.h>
|
|
||||||
#include <cplusplus/Overview.h>
|
#include <cplusplus/Overview.h>
|
||||||
|
|
||||||
#include <QtCore/QTime>
|
#include <QtCore/QTime>
|
||||||
@@ -120,8 +119,6 @@ public:
|
|||||||
doc->check();
|
doc->check();
|
||||||
|
|
||||||
FindUsages process(doc, snapshot);
|
FindUsages process(doc, snapshot);
|
||||||
process.setGlobalNamespaceBinding(bind(doc, snapshot));
|
|
||||||
|
|
||||||
process(symbol);
|
process(symbol);
|
||||||
usages = process.usages();
|
usages = process.usages();
|
||||||
}
|
}
|
||||||
@@ -177,7 +174,6 @@ QList<int> CppFindReferences::references(Symbol *symbol,
|
|||||||
QList<int> references;
|
QList<int> references;
|
||||||
|
|
||||||
FindUsages findUsages(doc, snapshot);
|
FindUsages findUsages(doc, snapshot);
|
||||||
findUsages.setGlobalNamespaceBinding(bind(doc, snapshot));
|
|
||||||
findUsages(symbol);
|
findUsages(symbol);
|
||||||
references = findUsages.references();
|
references = findUsages.references();
|
||||||
|
|
||||||
|
|||||||
@@ -791,8 +791,7 @@ QList<int> CppModelManager::references(CPlusPlus::Symbol *symbol,
|
|||||||
CPlusPlus::Document::Ptr doc,
|
CPlusPlus::Document::Ptr doc,
|
||||||
const CPlusPlus::Snapshot &snapshot)
|
const CPlusPlus::Snapshot &snapshot)
|
||||||
{
|
{
|
||||||
NamespaceBindingPtr glo = bind(doc, snapshot);
|
return m_findReferences->references(symbol, doc, snapshot);
|
||||||
return m_findReferences->references(DeprecatedLookupContext::canonicalSymbol(symbol, glo.data()), doc, snapshot);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CppModelManager::findUsages(CPlusPlus::Document::Ptr symbolDocument, CPlusPlus::Symbol *symbol)
|
void CppModelManager::findUsages(CPlusPlus::Document::Ptr symbolDocument, CPlusPlus::Symbol *symbol)
|
||||||
|
|||||||
Reference in New Issue
Block a user