kde-extraapps/kdevplatform/language/duchain/ducontext.cpp
2015-07-26 14:23:17 +03:00

1719 lines
55 KiB
C++

/* This is part of KDevelop
Copyright 2006 Hamish Rodda <rodda@kde.org>
Copyright 2007-2009 David Nolden <david.nolden.kdevelop@art-master.de>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License version 2 as published by the Free Software Foundation.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "ducontext.h"
#include <limits>
#include <QMutableLinkedListIterator>
#include <QSet>
#include <ktexteditor/document.h>
#include "ducontextdata.h"
#include "declaration.h"
#include "duchain.h"
#include "duchainlock.h"
#include "use.h"
#include "identifier.h"
#include "topducontext.h"
#include "persistentsymboltable.h"
#include "aliasdeclaration.h"
#include "namespacealiasdeclaration.h"
#include "abstractfunctiondeclaration.h"
#include "indexedstring.h"
#include "duchainregister.h"
#include "topducontextdynamicdata.h"
#include "importers.h"
#include "uses.h"
#include "navigation/abstractdeclarationnavigationcontext.h"
#include "navigation/abstractnavigationwidget.h"
#include "ducontextdynamicdata.h"
///It is fine to use one global static mutex here
const uint maxParentDepth = 20;
using namespace KTextEditor;
//Stored statically for performance-reasons
#ifndef NDEBUG
#define ENSURE_CAN_WRITE_(x) {if(x->inDUChain()) { ENSURE_CHAIN_WRITE_LOCKED }}
#define ENSURE_CAN_READ_(x) {if(x->inDUChain()) { ENSURE_CHAIN_READ_LOCKED }}
#else
#define ENSURE_CAN_WRITE_(x)
#define ENSURE_CAN_READ_(x)
#endif
namespace KDevelop
{
DEFINE_LIST_MEMBER_HASH(DUContextData, m_childContexts, LocalIndexedDUContext)
DEFINE_LIST_MEMBER_HASH(DUContextData, m_importers, IndexedDUContext)
DEFINE_LIST_MEMBER_HASH(DUContextData, m_importedContexts, DUContext::Import)
DEFINE_LIST_MEMBER_HASH(DUContextData, m_localDeclarations, LocalIndexedDeclaration)
DEFINE_LIST_MEMBER_HASH(DUContextData, m_uses, Use)
REGISTER_DUCHAIN_ITEM(DUContext);
DUChainVisitor::~DUChainVisitor()
{
}
/**
* We leak here, to prevent a possible crash during destruction, as the destructor
* of Identifier is not safe to be called after the duchain has been destroyed
*/
const Identifier& globalImportIdentifier() {
static const Identifier globalImportIdentifierObject(*new Identifier("{...import...}"));
return globalImportIdentifierObject;
}
const Identifier& globalAliasIdentifier() {
static const Identifier globalAliasIdentifierObject(*new Identifier("{...alias...}"));
return globalAliasIdentifierObject;
}
const IndexedIdentifier& globalIndexedImportIdentifier()
{
static const IndexedIdentifier id(globalImportIdentifier());
return id;
}
const IndexedIdentifier& globalIndexedAliasIdentifier()
{
static const IndexedIdentifier id(globalAliasIdentifier());
return id;
}
void DUContext::rebuildDynamicData(DUContext* parent, uint ownIndex) {
Q_ASSERT(!parent || ownIndex);
m_dynamicData->m_topContext = parent ? parent->topContext() : static_cast<TopDUContext*>(this);
m_dynamicData->m_indexInTopContext = ownIndex;
m_dynamicData->m_parentContext = DUContextPointer(parent);
m_dynamicData->m_context = this;
DUChainBase::rebuildDynamicData(parent, ownIndex);
}
DUContextData::DUContextData()
: m_inSymbolTable(false)
, m_anonymousInParent(false)
, m_propagateDeclarations(false)
{
initializeAppendedLists();
}
DUContextData::~DUContextData() {
freeAppendedLists();
}
DUContextData::DUContextData(const DUContextData& rhs)
: DUChainBaseData(rhs)
, m_inSymbolTable(rhs.m_inSymbolTable)
, m_anonymousInParent(rhs.m_anonymousInParent)
, m_propagateDeclarations(rhs.m_propagateDeclarations)
{
initializeAppendedLists();
copyListsFrom(rhs);
m_scopeIdentifier = rhs.m_scopeIdentifier;
m_contextType = rhs.m_contextType;
m_owner = rhs.m_owner;
}
DUContextDynamicData::DUContextDynamicData(DUContext* d)
: m_topContext(0)
, m_hasLocalDeclarationsHash(false)
, m_indexInTopContext(0)
, m_context(d)
, m_rangesChanged(true)
{
}
void DUContextDynamicData::scopeIdentifier(bool includeClasses, QualifiedIdentifier& target) const {
if (m_parentContext)
m_parentContext->m_dynamicData->scopeIdentifier(includeClasses, target);
if (includeClasses || m_context->d_func()->m_contextType != DUContext::Class)
target += m_context->d_func()->m_scopeIdentifier;
}
bool DUContextDynamicData::imports(const DUContext* context, const TopDUContext* source,
QSet<const DUContextDynamicData*>* recursionGuard) const
{
if( this == context->m_dynamicData )
return true;
if (recursionGuard->contains(this)) {
return false;
}
recursionGuard->insert(this);
FOREACH_FUNCTION( const DUContext::Import& ctx, m_context->d_func()->m_importedContexts ) {
DUContext* import = ctx.context(source);
if(import == context || (import && import->m_dynamicData->imports(context, source, recursionGuard)))
return true;
}
return false;
}
void DUContextDynamicData::enableLocalDeclarationsHash(DUContext* ctx, const Identifier& currentIdentifier,
Declaration* currentDecl)
{
m_hasLocalDeclarationsHash = true;
FOREACH_FUNCTION(const LocalIndexedDeclaration& indexedDecl, ctx->d_func()->m_localDeclarations) {
Declaration* decl = indexedDecl.data(m_topContext);
Q_ASSERT(decl);
if(currentDecl != decl)
m_localDeclarationsHash.insert( decl->identifier(), DeclarationPointer(decl) );
else
m_localDeclarationsHash.insert( currentIdentifier, DeclarationPointer(decl) );
}
FOREACH_FUNCTION(const LocalIndexedDUContext& child, ctx->d_func()->m_childContexts) {
DUContext* childCtx = child.data(m_topContext);
Q_ASSERT(childCtx);
if(childCtx->d_func()->m_propagateDeclarations)
enableLocalDeclarationsHash(childCtx, currentIdentifier, currentDecl);
}
}
void DUContextDynamicData::disableLocalDeclarationsHash()
{
m_hasLocalDeclarationsHash = false;
m_localDeclarationsHash.clear();
}
bool DUContextDynamicData::needsLocalDeclarationsHash()
{
///@todo Do this again, it brings a large performance boost
//For now disable the hash, until we make sure that all declarations needed for the hash are loaded first
//including those in propagating sub-contexts.
//Then, also make sure that we create the declaration hash after loading if needed
return false;
if(m_context->d_func()->m_localDeclarationsSize() > 15)
return true;
uint propagatingChildContexts = 0;
FOREACH_FUNCTION(const LocalIndexedDUContext& child, m_context->d_func()->m_childContexts) {
DUContext* childCtx = child.data(m_topContext);
Q_ASSERT(childCtx);
if(childCtx->d_func()->m_propagateDeclarations)
++propagatingChildContexts;
}
return propagatingChildContexts > 4;
}
void DUContextDynamicData::addDeclarationToHash(const Identifier& identifier, Declaration* declaration)
{
if(m_hasLocalDeclarationsHash)
m_localDeclarationsHash.insert( identifier, DeclarationPointer(declaration) );
if( m_context->d_func()->m_propagateDeclarations && m_parentContext )
m_parentContext->m_dynamicData->addDeclarationToHash(identifier, declaration);
if(!m_hasLocalDeclarationsHash && needsLocalDeclarationsHash())
enableLocalDeclarationsHash(m_context, identifier, declaration);
}
void DUContextDynamicData::removeDeclarationFromHash(const Identifier& identifier, Declaration* declaration)
{
if(m_hasLocalDeclarationsHash)
m_localDeclarationsHash.remove( identifier, DeclarationPointer(declaration) );
if( m_context->d_func()->m_propagateDeclarations && m_parentContext )
m_parentContext->m_dynamicData->removeDeclarationFromHash(identifier, declaration);
if(m_hasLocalDeclarationsHash && !needsLocalDeclarationsHash())
disableLocalDeclarationsHash();
}
inline bool isContextTemporary(uint index) {
return index > (0xffffffff/2);
}
void DUContextDynamicData::addDeclaration( Declaration * newDeclaration )
{
// The definition may not have its identifier set when it's assigned...
// allow dupes here, TODO catch the error elsewhere
//If this context is temporary, added declarations should be as well, and viceversa
Q_ASSERT(isContextTemporary(m_indexInTopContext) == isContextTemporary(newDeclaration->ownIndex()));
CursorInRevision start = newDeclaration->range().start;
///@todo Do binary search to find the position
bool inserted = false;
for (int i = m_context->d_func_dynamic()->m_localDeclarationsSize()-1; i >= 0; --i) {
Declaration* child = m_context->d_func_dynamic()->m_localDeclarations()[i].data(m_topContext);
if(!child) {
kWarning() << "child declaration number" << i << "of" << m_context->d_func_dynamic()->m_localDeclarationsSize() << "is invalid";
continue;
}
if(child == newDeclaration)
return;
//TODO: All declarations in a macro will have the same empty range, and just get appended
//that may not be Good Enough in complex cases.
if (start >= child->range().start) {
m_context->d_func_dynamic()->m_localDeclarationsList().insert(i+1, newDeclaration);
if(!m_context->d_func()->m_localDeclarations()[i+1].data(m_topContext))
kFatal() << "Inserted a not addressable declaration";
inserted = true;
break;
}
}
if( !inserted ) //We haven't found any child that is before this one, so prepend it
m_context->d_func_dynamic()->m_localDeclarationsList().insert(0, newDeclaration);
addDeclarationToHash(newDeclaration->identifier(), newDeclaration);
}
bool DUContextDynamicData::removeDeclaration(Declaration* declaration)
{
if(!m_topContext->deleting()) //We can save a lot of time by just not caring about the hash while deleting
removeDeclarationFromHash(declaration->identifier(), declaration);
if( m_context->d_func_dynamic()->m_localDeclarationsList().removeOne(LocalIndexedDeclaration(declaration)) ) {
return true;
}else {
return false;
}
}
void DUContext::changingIdentifier( Declaration* decl, const Identifier& from, const Identifier& to ) {
ENSURE_CAN_WRITE
m_dynamicData->removeDeclarationFromHash(from, decl);
m_dynamicData->addDeclarationToHash(to, decl);
}
void DUContextDynamicData::addChildContext( DUContext * context )
{
// Internal, don't need to assert a lock
Q_ASSERT(!context->m_dynamicData->m_parentContext
|| context->m_dynamicData->m_parentContext.data()->m_dynamicData == this );
LocalIndexedDUContext indexed(context->m_dynamicData->m_indexInTopContext);
//If this context is temporary, added declarations should be as well, and viceversa
Q_ASSERT(isContextTemporary(m_indexInTopContext) == isContextTemporary(indexed.localIndex()));
bool inserted = false;
int childCount = m_context->d_func()->m_childContextsSize();
for (int i = childCount-1; i >= 0; --i) {///@todo Do binary search to find the position
DUContext* child = m_context->d_func()->m_childContexts()[i].data(m_topContext);
if (context == child)
return;
if (context->range().start >= child->range().start) {
m_context->d_func_dynamic()->m_childContextsList().insert(i+1, indexed);
context->m_dynamicData->m_parentContext = m_context;
inserted = true;
break;
}
}
if( !inserted ) {
m_context->d_func_dynamic()->m_childContextsList().insert(0, indexed);
context->m_dynamicData->m_parentContext = m_context;
}
if(context->d_func()->m_propagateDeclarations) {
disableLocalDeclarationsHash();
if(needsLocalDeclarationsHash())
enableLocalDeclarationsHash(m_context);
}
}
bool DUContextDynamicData::removeChildContext( DUContext* context ) {
// ENSURE_CAN_WRITE
if( m_context->d_func_dynamic()->m_childContextsList().removeOne(LocalIndexedDUContext(context)) )
return true;
else
return false;
}
void DUContextDynamicData::addImportedChildContext( DUContext * context )
{
// ENSURE_CAN_WRITE
DUContext::Import import(m_context, context);
if(import.isDirect()) {
//Direct importers are registered directly within the data
if(m_context->d_func_dynamic()->m_importersList().contains(IndexedDUContext(context))) {
kDebug(9505) << m_context->scopeIdentifier(true).toString() << "importer added multiple times:" << context->scopeIdentifier(true).toString();
return;
}
m_context->d_func_dynamic()->m_importersList().append(context);
}else{
//Indirect importers are registered separately
Importers::self().addImporter(import.indirectDeclarationId(), IndexedDUContext(context));
}
}
//Can also be called with a context that is not in the list
void DUContextDynamicData::removeImportedChildContext( DUContext * context )
{
// ENSURE_CAN_WRITE
DUContext::Import import(m_context, context);
if(import.isDirect()) {
m_context->d_func_dynamic()->m_importersList().removeOne(IndexedDUContext(context));
}else{
//Indirect importers are registered separately
Importers::self().removeImporter(import.indirectDeclarationId(), IndexedDUContext(context));
}
}
int DUContext::depth() const
{
{ if (!parentContext()) return 0; return parentContext()->depth() + 1; }
}
DUContext::DUContext(DUContextData& data)
: DUChainBase(data)
, m_dynamicData(new DUContextDynamicData(this))
{
}
DUContext::DUContext(const RangeInRevision& range, DUContext* parent, bool anonymous)
: DUChainBase(*new DUContextData(), range)
, m_dynamicData(new DUContextDynamicData(this))
{
d_func_dynamic()->setClassId(this);
if(parent)
m_dynamicData->m_topContext = parent->topContext();
else
m_dynamicData->m_topContext = static_cast<TopDUContext*>(this);
d_func_dynamic()->setClassId(this);
DUCHAIN_D_DYNAMIC(DUContext);
d->m_contextType = Other;
m_dynamicData->m_parentContext = 0;
d->m_anonymousInParent = anonymous;
d->m_inSymbolTable = false;
if (parent) {
m_dynamicData->m_indexInTopContext = parent->topContext()->m_dynamicData->allocateContextIndex(this, parent->isAnonymous() || anonymous);
Q_ASSERT(m_dynamicData->m_indexInTopContext);
if( !anonymous )
parent->m_dynamicData->addChildContext(this);
else
m_dynamicData->m_parentContext = parent;
}
if(parent && !anonymous && parent->inSymbolTable())
setInSymbolTable(true);
}
bool DUContext::isAnonymous() const {
return d_func()->m_anonymousInParent || (m_dynamicData->m_parentContext && m_dynamicData->m_parentContext->isAnonymous());
}
DUContext::DUContext( DUContextData& dd, const RangeInRevision& range, DUContext * parent, bool anonymous )
: DUChainBase(dd, range)
, m_dynamicData(new DUContextDynamicData(this))
{
if(parent)
m_dynamicData->m_topContext = parent->topContext();
else
m_dynamicData->m_topContext = static_cast<TopDUContext*>(this);
DUCHAIN_D_DYNAMIC(DUContext);
d->m_contextType = Other;
m_dynamicData->m_parentContext = 0;
d->m_inSymbolTable = false;
d->m_anonymousInParent = anonymous;
if (parent) {
m_dynamicData->m_indexInTopContext = parent->topContext()->m_dynamicData->allocateContextIndex(this, parent->isAnonymous() || anonymous);
if( !anonymous )
parent->m_dynamicData->addChildContext(this);
else
m_dynamicData->m_parentContext = parent;
}
}
DUContext::DUContext(DUContext& useDataFrom)
: DUChainBase(useDataFrom)
, m_dynamicData(useDataFrom.m_dynamicData)
{
}
DUContext::~DUContext( )
{
TopDUContext* top = topContext();
if(!top->deleting() || !top->isOnDisk()) {
DUCHAIN_D_DYNAMIC(DUContext);
if(d->m_owner.declaration())
d->m_owner.declaration()->setInternalContext(0);
while( d->m_importersSize() != 0 ) {
if(d->m_importers()[0].data())
d->m_importers()[0].data()->removeImportedParentContext(this);
else {
kDebug() << "importer disappeared";
d->m_importersList().removeOne(d->m_importers()[0]);
}
}
clearImportedParentContexts();
}
deleteChildContextsRecursively();
if(!topContext()->deleting() || !topContext()->isOnDisk())
deleteUses();
deleteLocalDeclarations();
//If the top-context is being delete, we don't need to spend time rebuilding the inner structure.
//That's expensive, especially when the data is not dynamic.
if(!top->deleting() || !top->isOnDisk()) {
if (m_dynamicData->m_parentContext)
m_dynamicData->m_parentContext->m_dynamicData->removeChildContext(this);
}
top->m_dynamicData->clearContextIndex(this);
Q_ASSERT(d_func()->isDynamic() == (!top->deleting() || !top->isOnDisk() || top->m_dynamicData->isTemporaryContextIndex(m_dynamicData->m_indexInTopContext)));
delete m_dynamicData;
}
QVector< DUContext * > DUContext::childContexts( ) const
{
ENSURE_CAN_READ
QVector< DUContext * > ret;
FOREACH_FUNCTION(const LocalIndexedDUContext& ctx, d_func()->m_childContexts)
ret << ctx.data(topContext());
return ret;
}
Declaration* DUContext::owner() const {
ENSURE_CAN_READ
return d_func()->m_owner.declaration();
}
void DUContext::setOwner(Declaration* owner) {
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
if( owner == d->m_owner.declaration() )
return;
Declaration* oldOwner = d->m_owner.declaration();
d->m_owner = owner;
//Q_ASSERT(!oldOwner || oldOwner->internalContext() == this);
if( oldOwner && oldOwner->internalContext() == this )
oldOwner->setInternalContext(0);
//The context set as internal context should always be the last opened context
if( owner )
owner->setInternalContext(this);
}
DUContext* DUContext::parentContext( ) const
{
//ENSURE_CAN_READ Commented out for performance reasons
return m_dynamicData->m_parentContext.data();
}
void DUContext::setPropagateDeclarations(bool propagate)
{
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
if(propagate == d->m_propagateDeclarations)
return;
m_dynamicData->m_parentContext->m_dynamicData->disableLocalDeclarationsHash();
d->m_propagateDeclarations = propagate;
if(m_dynamicData->m_parentContext->m_dynamicData->needsLocalDeclarationsHash())
m_dynamicData->m_parentContext->m_dynamicData->enableLocalDeclarationsHash(m_dynamicData->m_parentContext.data());
}
bool DUContext::isPropagateDeclarations() const
{
return d_func()->m_propagateDeclarations;
}
QList<Declaration*> DUContext::findLocalDeclarations( const Identifier& identifier,
const CursorInRevision & position,
const TopDUContext* topContext,
const AbstractType::Ptr& dataType,
SearchFlags flags ) const
{
ENSURE_CAN_READ
DeclarationList ret;
findLocalDeclarationsInternal(identifier, position.isValid() ? position : range().end, dataType, ret, topContext ? topContext : this->topContext(), flags);
return ret.toList();
}
bool contextIsChildOrEqual(const DUContext* childContext, const DUContext* context) {
if(childContext == context)
return true;
if(childContext->parentContext())
return contextIsChildOrEqual(childContext->parentContext(), context);
else
return false;
}
void DUContext::findLocalDeclarationsInternal( const Identifier& identifier,
const CursorInRevision & position,
const AbstractType::Ptr& dataType,
DeclarationList& ret, const TopDUContext* /*source*/,
SearchFlags flags ) const
{
{
struct Checker {
Checker(SearchFlags flags, const AbstractType::Ptr& dataType, const CursorInRevision & position, DUContext::ContextType ownType) : m_flags(flags), m_dataType(dataType), m_position(position), m_ownType(ownType) {
}
Declaration* check(Declaration* declaration) {
///@todo This is C++-specific
if (m_ownType != Class && m_ownType != Template
&& m_position.isValid() && m_position <= declaration->range().start)
{
return 0;
}
if( declaration->kind() == Declaration::Alias && !(m_flags & DontResolveAliases) ) {
//Apply alias declarations
AliasDeclaration* alias = static_cast<AliasDeclaration*>(declaration);
if(alias->aliasedDeclaration().isValid()) {
declaration = alias->aliasedDeclaration().declaration();
} else {
#ifndef Q_CC_MSVC
kDebug() << "lost aliased declaration";
#endif
}
}
if( declaration->kind() == Declaration::NamespaceAlias && !(m_flags & NoFiltering) )
return 0;
if((m_flags & OnlyFunctions) && !declaration->isFunctionDeclaration())
return 0;
if (m_dataType && m_dataType->indexed() != declaration->indexedType()) {
return 0;
}
return declaration;
}
SearchFlags m_flags;
const AbstractType::Ptr m_dataType;
const CursorInRevision& m_position;
DUContext::ContextType m_ownType;
};
Checker checker(flags, dataType, position, type());
if(m_dynamicData->m_hasLocalDeclarationsHash) {
//Use a special hash that contains all declarations visible in this context
QHash<Identifier, DeclarationPointer>::const_iterator it = m_dynamicData->m_localDeclarationsHash.constFind(identifier);
QHash<Identifier, DeclarationPointer>::const_iterator end = m_dynamicData->m_localDeclarationsHash.constEnd();
for( ; it != end && it.key() == identifier; ++it ) {
Declaration* declaration = (*it).data();
if( !declaration ) {
//This should never happen, but let's see
kDebug(9505) << "DUContext::findLocalDeclarationsInternal: Invalid declaration in local-declaration-hash";
continue;
}
Declaration* checked = checker.check(declaration);
if(checked)
ret.append(checked);
}
}else if(d_func()->m_inSymbolTable && !indexedLocalScopeIdentifier().isEmpty() && !identifier.isEmpty()) {
//This context is in the symbol table, use the symbol-table to speed up the search
QualifiedIdentifier id(scopeIdentifier(true) + identifier);
TopDUContext* top = topContext();
uint count;
const IndexedDeclaration* declarations;
PersistentSymbolTable::self().declarations(id, count, declarations);
for(uint a = 0; a < count; ++a) {
///@todo Eventually do efficient iteration-free filtering
if(declarations[a].topContextIndex() == top->ownIndex()) {
Declaration* decl = LocalIndexedDeclaration(declarations[a].localIndex()).data(top);
if(decl && contextIsChildOrEqual(decl->context(), this)) {
Declaration* checked = checker.check(decl);
if(checked)
ret.append(checked);
}
}
}
}else {
//Iterate through all declarations
DUContextDynamicData::VisibleDeclarationIterator it(m_dynamicData);
IndexedIdentifier indexedIdentifier(identifier);
while(it) {
Declaration* declaration = *it;
if(declaration && declaration->indexedIdentifier() == indexedIdentifier) {
Declaration* checked = checker.check(declaration);
if(checked)
ret.append(checked);
}
++it;
}
}
}
}
bool DUContext::foundEnough( const DeclarationList& ret, SearchFlags flags ) const {
if( !ret.isEmpty() && !(flags & DUContext::NoFiltering))
return true;
else
return false;
}
bool DUContext::findDeclarationsInternal( const SearchItem::PtrList & baseIdentifiers,
const CursorInRevision & position,
const AbstractType::Ptr& dataType,
DeclarationList& ret, const TopDUContext* source,
SearchFlags flags, uint depth ) const
{
if(depth > maxParentDepth) {
kDebug() << "maximum depth reached in" << scopeIdentifier(true);
return false;
}
DUCHAIN_D(DUContext);
if( d_func()->m_contextType != Namespace ) { //If we're in a namespace, delay all the searching into the top-context, because only that has the overview to pick the correct declarations.
for(int a = 0; a < baseIdentifiers.size(); ++a)
if(!baseIdentifiers[a]->isExplicitlyGlobal && baseIdentifiers[a]->next.isEmpty()) //It makes no sense searching locally for qualified identifiers
findLocalDeclarationsInternal(baseIdentifiers[a]->identifier, position, dataType, ret, source, flags);
if( foundEnough(ret, flags) )
return true;
}
///Step 1: Apply namespace-aliases and -imports
SearchItem::PtrList aliasedIdentifiers;
//Because of namespace-imports and aliases, this identifier may need to be searched under multiple names
applyAliases(baseIdentifiers, aliasedIdentifiers, position, false, type() != DUContext::Namespace && type() != DUContext::Global);
if( d->m_importedContextsSize() != 0 ) {
///Step 2: Give identifiers that are not marked as explicitly-global to imported contexts(explicitly global ones are treatead in TopDUContext)
SearchItem::PtrList nonGlobalIdentifiers;
FOREACH_ARRAY( const SearchItem::Ptr& identifier, aliasedIdentifiers )
if( !identifier->isExplicitlyGlobal )
nonGlobalIdentifiers << identifier;
if( !nonGlobalIdentifiers.isEmpty() ) {
for(int import = d->m_importedContextsSize()-1; import >= 0; --import ) {
DUContext* context = d->m_importedContexts()[import].context(source);
while( !context && import > 0 ) {
--import;
context = d->m_importedContexts()[import].context(source);
}
if(context == this) {
kDebug() << "resolved self as import:" << scopeIdentifier(true);
continue;
}
if( !context )
break;
if( position.isValid() && d->m_importedContexts()[import].position.isValid() && position < d->m_importedContexts()[import].position )
continue;
if( !context->findDeclarationsInternal(nonGlobalIdentifiers, url() == context->url() ? position : context->range().end, dataType, ret, source, flags | InImportedParentContext, depth+1) )
return false;
}
}
}
if( foundEnough(ret, flags) )
return true;
///Step 3: Continue search in parent-context
if (!(flags & DontSearchInParent) && shouldSearchInParent(flags) && m_dynamicData->m_parentContext) {
applyUpwardsAliases(aliasedIdentifiers, source);
return m_dynamicData->m_parentContext->findDeclarationsInternal(aliasedIdentifiers, url() == m_dynamicData->m_parentContext->url() ? position : m_dynamicData->m_parentContext->range().end, dataType, ret, source, flags, depth);
}
return true;
}
QList< QualifiedIdentifier > DUContext::fullyApplyAliases(const QualifiedIdentifier& id,
const TopDUContext* source) const
{
ENSURE_CAN_READ
if(!source)
source = topContext();
SearchItem::PtrList identifiers;
identifiers << SearchItem::Ptr(new SearchItem(id));
const DUContext* current = this;
while(current) {
SearchItem::PtrList aliasedIdentifiers;
current->applyAliases(identifiers, aliasedIdentifiers, CursorInRevision::invalid(), true, false);
current->applyUpwardsAliases(identifiers, source);
current = current->parentContext();
}
QList<QualifiedIdentifier> ret;
FOREACH_ARRAY(const SearchItem::Ptr& item, identifiers)
ret += item->toList();
return ret;
}
QList<Declaration*> DUContext::findDeclarations( const QualifiedIdentifier & identifier,
const CursorInRevision & position,
const AbstractType::Ptr& dataType,
const TopDUContext* topContext, SearchFlags flags) const
{
ENSURE_CAN_READ
DeclarationList ret;
SearchItem::PtrList identifiers;
identifiers << SearchItem::Ptr(new SearchItem(identifier));
findDeclarationsInternal(identifiers, position.isValid() ? position : range().end, dataType, ret, topContext ? topContext : this->topContext(), flags, 0);
return ret.toList();
}
bool DUContext::imports(const DUContext* origin, const CursorInRevision& /*position*/ ) const
{
ENSURE_CAN_READ
QSet<const DUContextDynamicData*> recursionGuard;
recursionGuard.reserve(8);
return m_dynamicData->imports(origin, topContext(), &recursionGuard);
}
bool DUContext::addIndirectImport(const DUContext::Import& import) {
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
for(unsigned int a = 0; a < d->m_importedContextsSize(); ++a) {
if(d->m_importedContexts()[a] == import) {
d->m_importedContextsList()[a].position = import.position;
return true;
}
}
///Do not sort the imported contexts by their own line-number, it makes no sense.
///Contexts added first, aka template-contexts, should stay in first place, so they are searched first.
d->m_importedContextsList().append(import);
return false;
}
void DUContext::addImportedParentContext( DUContext * context, const CursorInRevision& position, bool anonymous, bool /*temporary*/ )
{
ENSURE_CAN_WRITE
if(context == this) {
kDebug() << "Tried to import self";
return;
}
if(!context) {
kDebug() << "Tried to import invalid context";
return;
}
Import import(context, this, position);
if(addIndirectImport(import))
return;
if( !anonymous ) {
ENSURE_CAN_WRITE_(context)
context->m_dynamicData->addImportedChildContext(this);
}
}
void DUContext::removeImportedParentContext( DUContext * context )
{
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
Import import(context, this, CursorInRevision::invalid());
for(unsigned int a = 0; a < d->m_importedContextsSize(); ++a) {
if(d->m_importedContexts()[a] == import) {
d->m_importedContextsList().remove(a);
break;
}
}
if( !context )
return;
context->m_dynamicData->removeImportedChildContext(this);
}
KDevVarLengthArray<IndexedDUContext> DUContext::indexedImporters() const
{
KDevVarLengthArray<IndexedDUContext> ret;
if(owner())
ret = Importers::self().importers(owner()->id()); //Add indirect importers to the list
FOREACH_FUNCTION(const IndexedDUContext& ctx, d_func()->m_importers)
ret.append(ctx);
return ret;
}
QVector<DUContext*> DUContext::importers() const
{
ENSURE_CAN_READ
QVector<DUContext*> ret;
FOREACH_FUNCTION(const IndexedDUContext& ctx, d_func()->m_importers)
ret << ctx.context();
if(owner()) {
//Add indirect importers to the list
KDevVarLengthArray<IndexedDUContext> indirect = Importers::self().importers(owner()->id());
FOREACH_ARRAY(const IndexedDUContext& ctx, indirect) {
ret << ctx.context();
}
}
return ret;
}
DUContext * DUContext::findContext( const CursorInRevision& position, DUContext* parent) const
{
ENSURE_CAN_READ
if (!parent)
parent = const_cast<DUContext*>(this);
FOREACH_FUNCTION(const LocalIndexedDUContext& context, parent->d_func()->m_childContexts)
if (context.data(topContext())->range().contains(position)) {
DUContext* ret = findContext(position, context.data(topContext()));
if (!ret)
ret = context.data(topContext());
return ret;
}
return 0;
}
bool DUContext::parentContextOf(DUContext* context) const
{
if (this == context)
return true;
FOREACH_FUNCTION(const LocalIndexedDUContext& child, d_func()->m_childContexts) {
if (child.data(topContext())->parentContextOf(context))
return true;
}
return false;
}
QList< QPair<Declaration*, int> > DUContext::allDeclarations(const CursorInRevision& position,
const TopDUContext* topContext,
bool searchInParents) const
{
ENSURE_CAN_READ
QList< QPair<Declaration*, int> > ret;
QHash<const DUContext*, bool> hadContexts;
// Iterate back up the chain
mergeDeclarationsInternal(ret, position, hadContexts, topContext ? topContext : this->topContext(), searchInParents);
return ret;
}
QVector<Declaration*> DUContext::localDeclarations(const TopDUContext* source) const
{
Q_UNUSED(source);
ENSURE_CAN_READ
QVector<Declaration*> ret;
FOREACH_FUNCTION(const LocalIndexedDeclaration& decl, d_func()->m_localDeclarations) {
ret << decl.data(topContext());
}
return ret;
}
void DUContext::mergeDeclarationsInternal(QList< QPair<Declaration*, int> >& definitions,
const CursorInRevision& position,
QHash<const DUContext*, bool>& hadContexts,
const TopDUContext* source,
bool searchInParents, int currentDepth) const
{
ENSURE_CAN_READ
if((currentDepth > 300 && currentDepth < 1000) || currentDepth > 1300) {
kDebug() << "too much depth";
return;
}
DUCHAIN_D(DUContext);
if(hadContexts.contains(this) && !searchInParents)
return;
if(!hadContexts.contains(this)) {
hadContexts[this] = true;
if( (type() == DUContext::Namespace || type() == DUContext::Global) && currentDepth < 1000 )
currentDepth += 1000;
{
DUContextDynamicData::VisibleDeclarationIterator it(m_dynamicData);
while(it) {
Declaration* decl = *it;
if ( decl && (!position.isValid() || decl->range().start <= position) )
definitions << qMakePair(decl, currentDepth);
++it;
}
}
for(int a = d->m_importedContextsSize()-1; a >= 0; --a) {
const Import* import(&d->m_importedContexts()[a]);
DUContext* context = import->context(source);
while( !context && a > 0 ) {
--a;
import = &d->m_importedContexts()[a];
context = import->context(source);
}
if( !context )
break;
if(context == this) {
kDebug() << "resolved self as import:" << scopeIdentifier(true);
continue;
}
if( position.isValid() && import->position.isValid() && position < import->position )
continue;
context->mergeDeclarationsInternal(definitions, CursorInRevision::invalid(), hadContexts, source, searchInParents && context->shouldSearchInParent(InImportedParentContext) && context->parentContext()->type() == DUContext::Helper, currentDepth+1);
}
}
///Only respect the position if the parent-context is not a class(@todo this is language-dependent)
if (parentContext() && searchInParents )
parentContext()->mergeDeclarationsInternal(definitions, parentContext()->type() == DUContext::Class ? parentContext()->range().end : position, hadContexts, source, searchInParents, currentDepth+1);
}
void DUContext::deleteLocalDeclarations()
{
ENSURE_CAN_WRITE
KDevVarLengthArray<LocalIndexedDeclaration> declarations;
{
FOREACH_FUNCTION(const LocalIndexedDeclaration& decl, d_func()->m_localDeclarations)
declarations.append(decl);
}
TopDUContext* top = topContext();
//If we are deleting something that is not stored to disk, we need to create + delete the declarations,
//so their destructor unregisters from the persistent symbol table and from TopDUContextDynamicData
FOREACH_ARRAY(const LocalIndexedDeclaration& decl, declarations)
if(decl.isLoaded(top) || !top->deleting() || !top->isOnDisk())
delete decl.data(top);
}
void DUContext::deleteChildContextsRecursively()
{
ENSURE_CAN_WRITE
TopDUContext* top = topContext();
QVector<LocalIndexedDUContext> children;
FOREACH_FUNCTION(const LocalIndexedDUContext& ctx, d_func()->m_childContexts)
children << ctx;
//If we are deleting a context that is already stored to disk, we don't need to load not yet loaded child-contexts.
//Else we need to, so the declarations are unregistered from symbol-table and from TopDUContextDynamicData in their destructor
foreach(const LocalIndexedDUContext &ctx, children)
if(ctx.isLoaded(top) || !top->deleting() || !top->isOnDisk())
delete ctx.data(top);
}
QVector< Declaration * > DUContext::clearLocalDeclarations( )
{
QVector< Declaration * > ret = localDeclarations();
foreach (Declaration* dec, ret)
dec->setContext(0);
return ret;
}
QualifiedIdentifier DUContext::scopeIdentifier(bool includeClasses) const
{
ENSURE_CAN_READ
QualifiedIdentifier ret;
m_dynamicData->scopeIdentifier(includeClasses, ret);
return ret;
}
bool DUContext::equalScopeIdentifier(const DUContext* rhs) const
{
ENSURE_CAN_READ
const DUContext* left = this;
const DUContext* right = rhs;
while(left || right) {
if(!left || !right)
return false;
if(!(left->d_func()->m_scopeIdentifier == right->d_func()->m_scopeIdentifier))
return false;
left = left->parentContext();
right = right->parentContext();
}
return true;
}
void DUContext::setLocalScopeIdentifier(const QualifiedIdentifier & identifier)
{
ENSURE_CAN_WRITE
//Q_ASSERT(d_func()->m_childContexts.isEmpty() && d_func()->m_localDeclarations.isEmpty());
bool wasInSymbolTable = inSymbolTable();
setInSymbolTable(false);
d_func_dynamic()->m_scopeIdentifier = identifier;
setInSymbolTable(wasInSymbolTable);
}
QualifiedIdentifier DUContext::localScopeIdentifier() const
{
//ENSURE_CAN_READ Commented out for performance reasons
return d_func()->m_scopeIdentifier;
}
IndexedQualifiedIdentifier DUContext::indexedLocalScopeIdentifier() const {
return d_func()->m_scopeIdentifier;
}
DUContext::ContextType DUContext::type() const
{
//ENSURE_CAN_READ This is disabled, because type() is called very often while searching, and it costs us performance
return d_func()->m_contextType;
}
void DUContext::setType(ContextType type)
{
ENSURE_CAN_WRITE
d_func_dynamic()->m_contextType = type;
}
QList<Declaration*> DUContext::findDeclarations(const Identifier& identifier, const CursorInRevision& position,
const TopDUContext* topContext, SearchFlags flags) const
{
ENSURE_CAN_READ
DeclarationList ret;
SearchItem::PtrList identifiers;
identifiers << SearchItem::Ptr(new SearchItem(QualifiedIdentifier(identifier)));
findDeclarationsInternal(identifiers, position.isValid() ? position : range().end, AbstractType::Ptr(), ret, topContext ? topContext : this->topContext(), flags, 0);
return ret.toList();
}
void DUContext::deleteUse(int index)
{
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
d->m_usesList().remove(index);
}
void DUContext::deleteUses()
{
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
d->m_usesList().clear();
}
void DUContext::deleteUsesRecursively()
{
deleteUses();
FOREACH_FUNCTION(const LocalIndexedDUContext& childContext, d_func()->m_childContexts)
childContext.data(topContext())->deleteUsesRecursively();
}
bool DUContext::inDUChain() const {
if( d_func()->m_anonymousInParent || !m_dynamicData->m_parentContext)
return false;
TopDUContext* top = topContext();
return top && top->inDUChain();
}
DUContext* DUContext::specialize(const IndexedInstantiationInformation& /*specialization*/,
const TopDUContext* topContext, int /*upDistance*/)
{
if(!topContext)
return 0;
return this;
}
CursorInRevision DUContext::importPosition(const DUContext* target) const
{
ENSURE_CAN_READ
DUCHAIN_D(DUContext);
Import import(const_cast<DUContext*>(target), this, CursorInRevision::invalid());
for(unsigned int a = 0; a < d->m_importedContextsSize(); ++a)
if(d->m_importedContexts()[a] == import)
return d->m_importedContexts()[a].position;
return CursorInRevision::invalid();
}
QVector<DUContext::Import> DUContext::importedParentContexts() const
{
ENSURE_CAN_READ
QVector<DUContext::Import> ret;
FOREACH_FUNCTION(const DUContext::Import& import, d_func()->m_importedContexts)
ret << import;
return ret;
}
void DUContext::applyAliases(const SearchItem::PtrList& baseIdentifiers, SearchItem::PtrList& identifiers,
const CursorInRevision& position, bool canBeNamespace, bool onlyImports) const {
DeclarationList imports;
findLocalDeclarationsInternal(globalImportIdentifier(), position, AbstractType::Ptr(), imports, topContext(), DUContext::NoFiltering);
if(imports.isEmpty() && onlyImports) {
identifiers = baseIdentifiers;
return;
}
FOREACH_ARRAY( const SearchItem::Ptr& identifier, baseIdentifiers ) {
bool addUnmodified = true;
if( !identifier->isExplicitlyGlobal ) {
if( !imports.isEmpty() )
{
//We have namespace-imports.
FOREACH_ARRAY( Declaration* importDecl, imports )
{
//Search for the identifier with the import-identifier prepended
if(dynamic_cast<NamespaceAliasDeclaration*>(importDecl))
{
NamespaceAliasDeclaration* alias = static_cast<NamespaceAliasDeclaration*>(importDecl);
identifiers.append( SearchItem::Ptr( new SearchItem( alias->importIdentifier(), identifier ) ) ) ;
}else{
kDebug() << "Declaration with namespace alias identifier has the wrong type" << importDecl->url().str() << importDecl->range().castToSimpleRange().textRange();
}
}
}
if( !identifier->isEmpty() && (identifier->hasNext() || canBeNamespace) ) {
DeclarationList aliases;
findLocalDeclarationsInternal(identifier->identifier, position, AbstractType::Ptr(), imports, 0, DUContext::NoFiltering);
if(!aliases.isEmpty()) {
//The first part of the identifier has been found as a namespace-alias.
//In c++, we only need the first alias. However, just to be correct, follow them all for now.
FOREACH_ARRAY( Declaration* aliasDecl, aliases )
{
if(!dynamic_cast<NamespaceAliasDeclaration*>(aliasDecl))
continue;
addUnmodified = false; //The un-modified identifier can be ignored, because it will be replaced with the resolved alias
NamespaceAliasDeclaration* alias = static_cast<NamespaceAliasDeclaration*>(aliasDecl);
//Create an identifier where namespace-alias part is replaced with the alias target
identifiers.append( SearchItem::Ptr( new SearchItem( alias->importIdentifier(), identifier->next ) ) ) ;
}
}
}
}
if( addUnmodified )
identifiers.append(identifier);
}
}
void DUContext::applyUpwardsAliases(SearchItem::PtrList& identifiers, const TopDUContext* /*source*/) const {
if(type() == Namespace) {
if(d_func()->m_scopeIdentifier.isEmpty())
return;
//Make sure we search for the items in all namespaces of the same name, by duplicating each one with the namespace-identifier prepended.
//We do this by prepending items to the current identifiers that equal the local scope identifier.
SearchItem::Ptr newItem( new SearchItem(d_func()->m_scopeIdentifier.identifier()) );
//This will exclude explictly global identifiers
newItem->addToEachNode( identifiers );
if(!newItem->next.isEmpty()) {
//Prepend the full scope before newItem
DUContext* parent = m_dynamicData->m_parentContext.data();
while(parent) {
newItem = SearchItem::Ptr( new SearchItem(parent->d_func()->m_scopeIdentifier, newItem) );
parent = parent->m_dynamicData->m_parentContext.data();
}
newItem->isExplicitlyGlobal = true;
identifiers.insert(0, newItem);
}
}
}
bool DUContext::shouldSearchInParent(SearchFlags flags) const
{
return (parentContext() && parentContext()->type() == DUContext::Helper && (flags & InImportedParentContext))
|| !(flags & InImportedParentContext);
}
const Use* DUContext::uses() const
{
ENSURE_CAN_READ
return d_func()->m_uses();
}
bool DUContext::declarationHasUses(Declaration* decl)
{
return DUChain::uses()->hasUses(decl->id());
}
int DUContext::usesCount() const
{
return d_func()->m_usesSize();
}
bool usesRangeLessThan(const Use& left, const Use& right)
{
return left.m_range.start < right.m_range.start;
}
int DUContext::createUse(int declarationIndex, const RangeInRevision& range, int insertBefore)
{
DUCHAIN_D_DYNAMIC(DUContext);
ENSURE_CAN_WRITE
Use use(range, declarationIndex);
if(insertBefore == -1) {
//Find position where to insert
const unsigned int size = d->m_usesSize();
const Use* uses = d->m_uses();
const Use* lowerBound = qLowerBound(uses, uses + size, use, usesRangeLessThan);
insertBefore = lowerBound - uses;
// comment out to test this:
/*
unsigned int a = 0;
for(; a < size && range.start > uses[a].m_range.start; ++a) {
}
Q_ASSERT(a == insertBefore);
*/
}
d->m_usesList().insert(insertBefore, use);
return insertBefore;
}
void DUContext::changeUseRange(int useIndex, const RangeInRevision& range)
{
ENSURE_CAN_WRITE
d_func_dynamic()->m_usesList()[useIndex].m_range = range;
}
void DUContext::setUseDeclaration(int useNumber, int declarationIndex)
{
ENSURE_CAN_WRITE
d_func_dynamic()->m_usesList()[useNumber].m_declarationIndex = declarationIndex;
}
DUContext * DUContext::findContextAt(const CursorInRevision & position, bool includeRightBorder) const
{
ENSURE_CAN_READ
// kDebug() << "searchign" << position.textCursor() << "in:" << scopeIdentifier(true).toString() << range().textRange() << includeRightBorder;
if (!range().contains(position) && (!includeRightBorder || range().end != position)) {
// kDebug() << "mismatch";
return 0;
}
for(int a = int(d_func()->m_childContextsSize())-1; a >= 0; --a)
if (DUContext* specific = d_func()->m_childContexts()[a].data(topContext())->findContextAt(position, includeRightBorder))
return specific;
return const_cast<DUContext*>(this);
}
Declaration * DUContext::findDeclarationAt(const CursorInRevision & position) const
{
ENSURE_CAN_READ
if (!range().contains(position))
return 0;
FOREACH_FUNCTION(const LocalIndexedDeclaration& child, d_func()->m_localDeclarations)
if (child.data(topContext())->range().contains(position))
return child.data(topContext());
return 0;
}
DUContext* DUContext::findContextIncluding(const RangeInRevision& range) const
{
ENSURE_CAN_READ
if (!this->range().contains(range))
return 0;
FOREACH_FUNCTION(const LocalIndexedDUContext& child, d_func()->m_childContexts)
if (DUContext* specific = child.data(topContext())->findContextIncluding(range))
return specific;
return const_cast<DUContext*>(this);
}
int DUContext::findUseAt(const CursorInRevision & position) const
{
ENSURE_CAN_READ
if (!range().contains(position))
return -1;
for(unsigned int a = 0; a < d_func()->m_usesSize(); ++a)
if (d_func()->m_uses()[a].m_range.contains(position))
return a;
return -1;
}
bool DUContext::inSymbolTable() const
{
return d_func()->m_inSymbolTable;
}
void DUContext::setInSymbolTable(bool inSymbolTable)
{
d_func_dynamic()->m_inSymbolTable = inSymbolTable;
}
// kate: indent-width 2;
void DUContext::clearImportedParentContexts()
{
ENSURE_CAN_WRITE
DUCHAIN_D_DYNAMIC(DUContext);
while( d->m_importedContextsSize() != 0 ) {
DUContext* ctx = d->m_importedContexts()[0].context(0, false);
if(ctx)
ctx->m_dynamicData->removeImportedChildContext(this);
d->m_importedContextsList().removeOne(d->m_importedContexts()[0]);
}
}
void DUContext::cleanIfNotEncountered(const QSet<DUChainBase*>& encountered)
{
ENSURE_CAN_WRITE
// It may happen that the deletion of one declaration triggers the deletion of another one
// Therefore we copy the list of indexed declarations and work on those. Indexed declarations
// will return zero for already deleted declarations.
KDevVarLengthArray<LocalIndexedDeclaration, 10> declarationsCopy;
{
declarationsCopy = d_func_dynamic()->m_localDeclarationsList();
}
FOREACH_ARRAY(const LocalIndexedDeclaration& indexedDec, declarationsCopy)
{
Declaration* dec = indexedDec.data(topContext());
if (dec && !encountered.contains(dec) && (!dec->isAutoDeclaration() || !dec->hasUses()))
delete dec;
}
//Copy since the array may change during the iteration
KDevVarLengthArray<LocalIndexedDUContext, 10> childrenCopy = d_func_dynamic()->m_childContextsList();
FOREACH_ARRAY(const LocalIndexedDUContext& childContext, childrenCopy)
if (!encountered.contains(childContext.data(topContext())))
delete childContext.data(topContext());
}
TopDUContext* DUContext::topContext() const
{
return m_dynamicData->m_topContext;
}
QWidget* DUContext::createNavigationWidget(Declaration* decl, TopDUContext* topContext,
const QString& htmlPrefix, const QString& htmlSuffix) const
{
if (decl) {
AbstractNavigationWidget* widget = new AbstractNavigationWidget;
AbstractDeclarationNavigationContext* context = new AbstractDeclarationNavigationContext(DeclarationPointer(decl),
TopDUContextPointer(topContext));
context->setPrefixSuffix(htmlPrefix, htmlSuffix);
widget->setContext(NavigationContextPointer(context));
return widget;
} else {
return 0;
}
}
void DUContext::squeeze()
{
FOREACH_FUNCTION(const LocalIndexedDUContext& child, d_func()->m_childContexts)
child.data(topContext())->squeeze();
}
QList<RangeInRevision> allUses(DUContext* context, int declarationIndex, bool noEmptyUses)
{
QList<RangeInRevision> ret;
for(int a = 0; a < context->usesCount(); ++a)
if(context->uses()[a].m_declarationIndex == declarationIndex)
if(!noEmptyUses || !context->uses()[a].m_range.isEmpty())
ret << context->uses()[a].m_range;
foreach(DUContext* child, context->childContexts())
ret += allUses(child, declarationIndex, noEmptyUses);
return ret;
}
DUContext::SearchItem::SearchItem(const QualifiedIdentifier& id, const Ptr& nextItem, int start)
: isExplicitlyGlobal(start == 0 ? id.explicitlyGlobal() : false)
{
if(!id.isEmpty()) {
if(id.count() > start)
identifier = id.at(start);
if(id.count() > start+1)
addNext(Ptr( new SearchItem(id, nextItem, start+1) ));
else if(nextItem)
next.append(nextItem);
}else if(nextItem) {
///If there is no prefix, just copy nextItem
isExplicitlyGlobal = nextItem->isExplicitlyGlobal;
identifier = nextItem->identifier;
next = nextItem->next;
}
}
DUContext::SearchItem::SearchItem(const QualifiedIdentifier& id, const PtrList& nextItems, int start)
: isExplicitlyGlobal(start == 0 ? id.explicitlyGlobal() : false)
{
if(id.count() > start)
identifier = id.at(start);
if(id.count() > start+1)
addNext(Ptr( new SearchItem(id, nextItems, start+1) ));
else
next = nextItems;
}
DUContext::SearchItem::SearchItem(bool explicitlyGlobal, const Identifier& id, const PtrList& nextItems)
: isExplicitlyGlobal(explicitlyGlobal)
, identifier(id)
, next(nextItems)
{
}
DUContext::SearchItem::SearchItem(bool explicitlyGlobal, const Identifier& id, const Ptr& nextItem)
: isExplicitlyGlobal(explicitlyGlobal)
, identifier(id)
{
next.append(nextItem);
}
bool DUContext::SearchItem::match(const QualifiedIdentifier& id, int offset) const {
if(id.isEmpty()) {
if(identifier.isEmpty() && next.isEmpty())
return true;
else
return false;
}
if(id.at(offset) != identifier) //The identifier is different
return false;
if(offset == id.count()-1) {
if(next.isEmpty())
return true; //match
else
return false; //id is too short
}
for(int a = 0; a < next.size(); ++a)
if(next[a]->match(id, offset+1))
return true;
return false;
}
bool DUContext::SearchItem::isEmpty() const {
return identifier.isEmpty();
}
bool DUContext::SearchItem::hasNext() const {
return !next.isEmpty();
}
QList<QualifiedIdentifier> DUContext::SearchItem::toList(const QualifiedIdentifier& prefix) const {
QList<QualifiedIdentifier> ret;
QualifiedIdentifier id = prefix;
if(id.isEmpty())
id.setExplicitlyGlobal(isExplicitlyGlobal);
if(!identifier.isEmpty())
id.push(identifier);
if(next.isEmpty()) {
ret << id;
} else {
for(int a = 0; a < next.size(); ++a)
ret += next[a]->toList(id);
}
return ret;
}
void DUContext::SearchItem::addNext(const SearchItem::Ptr& other) {
next.append(other);
}
void DUContext::SearchItem::addToEachNode(const SearchItem::Ptr& other) {
if(other->isExplicitlyGlobal)
return;
next.append(other);
for(int a = 0; a < next.size()-1; ++a)
next[a]->addToEachNode(other);
}
void DUContext::SearchItem::addToEachNode(const SearchItem::PtrList& other) {
int added = 0;
FOREACH_ARRAY(const SearchItem::Ptr& o, other) {
if(!o->isExplicitlyGlobal) {
next.append(o);
++added;
}
}
for(int a = 0; a < next.size()-added; ++a)
next[a]->addToEachNode(other);
}
DUContext::Import::Import(DUContext* _context, const DUContext* importer, const CursorInRevision& _position)
: position(_position)
{
if(_context && _context->owner() && (_context->owner()->specialization().index() || (importer && importer->topContext() != _context->topContext()))) {
m_declaration = _context->owner()->id();
}else{
m_context = _context;
}
}
DUContext::Import::Import(const DeclarationId& id, const CursorInRevision& _position)
: position(_position)
{
m_declaration = id;
}
DUContext* DUContext::Import::context(const TopDUContext* topContext, bool instantiateIfRequired) const {
if(m_declaration.isValid()) {
Declaration* decl = m_declaration.getDeclaration(topContext, instantiateIfRequired);
//This first case rests on the assumption that no context will ever import a function's expression context
//More accurately, that no specialized or cross-topContext imports will, but if the former assumption fails the latter will too
if (AbstractFunctionDeclaration *functionDecl = dynamic_cast<AbstractFunctionDeclaration*>(decl))
{
if (functionDecl->internalFunctionContext()) {
return functionDecl->internalFunctionContext();
} else {
kWarning() << "Import of function declaration without internal function context encountered!";
}
}
if(decl)
return decl->logicalInternalContext(topContext);
else
return 0;
}else{
return m_context.data();
}
}
bool DUContext::Import::isDirect() const {
return m_context.isValid();
}
void DUContext::visit(DUChainVisitor& visitor)
{
ENSURE_CAN_READ
visitor.visit(this);
TopDUContext* top = topContext();
{
FOREACH_FUNCTION(const LocalIndexedDeclaration& decl, d_func()->m_localDeclarations)
visitor.visit(decl.data(top));
}
FOREACH_FUNCTION(const LocalIndexedDUContext& ctx, d_func()->m_childContexts)
ctx.data(top)->visit(visitor);
}
}
// kate: space-indent on; indent-width 2; tab-width 4; replace-tabs on; auto-insert-doxygen on