2009-07-30 07:38:21 +08:00
|
|
|
//===--- Indexer.cpp - IndexProvider implementation -------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// IndexProvider implementation.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/Index/Indexer.h"
|
|
|
|
#include "clang/Index/Program.h"
|
|
|
|
#include "clang/Index/Handlers.h"
|
|
|
|
#include "clang/Index/TranslationUnit.h"
|
2009-07-30 07:41:18 +08:00
|
|
|
#include "ASTVisitor.h"
|
2009-07-30 07:39:52 +08:00
|
|
|
#include "clang/AST/DeclBase.h"
|
2009-07-30 07:38:21 +08:00
|
|
|
using namespace clang;
|
|
|
|
using namespace idx;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class EntityIndexer : public EntityHandler {
|
|
|
|
TranslationUnit *TU;
|
|
|
|
Indexer::MapTy ⤅
|
2010-07-06 13:55:13 +08:00
|
|
|
Indexer::DefMapTy &DefMap;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:38:21 +08:00
|
|
|
public:
|
2010-07-06 13:55:13 +08:00
|
|
|
EntityIndexer(TranslationUnit *tu, Indexer::MapTy &map,
|
|
|
|
Indexer::DefMapTy &defmap)
|
|
|
|
: TU(tu), Map(map), DefMap(defmap) { }
|
2009-07-30 07:38:21 +08:00
|
|
|
|
2009-07-30 07:38:35 +08:00
|
|
|
virtual void Handle(Entity Ent) {
|
2009-07-30 07:38:21 +08:00
|
|
|
if (Ent.isInternalToTU())
|
|
|
|
return;
|
|
|
|
Map[Ent].insert(TU);
|
2010-07-06 13:55:13 +08:00
|
|
|
|
|
|
|
Decl *D = Ent.getDecl(TU->getASTContext());
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D))
|
2011-05-07 04:44:56 +08:00
|
|
|
if (FD->doesThisDeclarationHaveABody())
|
2010-07-06 13:55:13 +08:00
|
|
|
DefMap[Ent] = std::make_pair(FD, TU);
|
2009-07-30 07:38:21 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2009-07-30 07:41:18 +08:00
|
|
|
class SelectorIndexer : public ASTVisitor<SelectorIndexer> {
|
|
|
|
Program &Prog;
|
|
|
|
TranslationUnit *TU;
|
|
|
|
Indexer::SelMapTy ⤅
|
|
|
|
|
|
|
|
public:
|
|
|
|
SelectorIndexer(Program &prog, TranslationUnit *tu, Indexer::SelMapTy &map)
|
|
|
|
: Prog(prog), TU(tu), Map(map) { }
|
|
|
|
|
|
|
|
void VisitObjCMethodDecl(ObjCMethodDecl *D) {
|
|
|
|
Map[GlobalSelector::get(D->getSelector(), Prog)].insert(TU);
|
|
|
|
Base::VisitObjCMethodDecl(D);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VisitObjCMessageExpr(ObjCMessageExpr *Node) {
|
|
|
|
Map[GlobalSelector::get(Node->getSelector(), Prog)].insert(TU);
|
|
|
|
Base::VisitObjCMessageExpr(Node);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2009-07-30 07:38:21 +08:00
|
|
|
} // anonymous namespace
|
|
|
|
|
|
|
|
void Indexer::IndexAST(TranslationUnit *TU) {
|
2009-07-30 07:38:51 +08:00
|
|
|
assert(TU && "Passed null TranslationUnit");
|
2009-07-30 07:41:18 +08:00
|
|
|
ASTContext &Ctx = TU->getASTContext();
|
|
|
|
CtxTUMap[&Ctx] = TU;
|
2010-07-06 13:55:13 +08:00
|
|
|
EntityIndexer Idx(TU, Map, DefMap);
|
2009-07-30 07:41:18 +08:00
|
|
|
Prog.FindEntities(Ctx, Idx);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:41:18 +08:00
|
|
|
SelectorIndexer SelIdx(Prog, TU, SelMap);
|
|
|
|
SelIdx.Visit(Ctx.getTranslationUnitDecl());
|
2009-07-30 07:38:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Indexer::GetTranslationUnitsFor(Entity Ent,
|
2009-07-30 07:38:45 +08:00
|
|
|
TranslationUnitHandler &Handler) {
|
2009-07-30 07:38:21 +08:00
|
|
|
assert(Ent.isValid() && "Expected valid Entity");
|
2009-07-30 07:39:52 +08:00
|
|
|
|
|
|
|
if (Ent.isInternalToTU()) {
|
|
|
|
Decl *D = Ent.getInternalDecl();
|
|
|
|
CtxTUMapTy::iterator I = CtxTUMap.find(&D->getASTContext());
|
|
|
|
if (I != CtxTUMap.end())
|
|
|
|
Handler.Handle(I->second);
|
|
|
|
return;
|
|
|
|
}
|
2009-07-30 07:38:21 +08:00
|
|
|
|
|
|
|
MapTy::iterator I = Map.find(Ent);
|
|
|
|
if (I == Map.end())
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:38:21 +08:00
|
|
|
TUSetTy &Set = I->second;
|
|
|
|
for (TUSetTy::iterator I = Set.begin(), E = Set.end(); I != E; ++I)
|
2009-07-30 07:38:45 +08:00
|
|
|
Handler.Handle(*I);
|
2009-07-30 07:38:21 +08:00
|
|
|
}
|
2009-07-30 07:41:18 +08:00
|
|
|
|
|
|
|
void Indexer::GetTranslationUnitsFor(GlobalSelector Sel,
|
|
|
|
TranslationUnitHandler &Handler) {
|
|
|
|
assert(Sel.isValid() && "Expected valid GlobalSelector");
|
|
|
|
|
|
|
|
SelMapTy::iterator I = SelMap.find(Sel);
|
|
|
|
if (I == SelMap.end())
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-07-30 07:41:18 +08:00
|
|
|
TUSetTy &Set = I->second;
|
|
|
|
for (TUSetTy::iterator I = Set.begin(), E = Set.end(); I != E; ++I)
|
|
|
|
Handler.Handle(*I);
|
|
|
|
}
|
2010-07-06 13:55:13 +08:00
|
|
|
|
|
|
|
std::pair<FunctionDecl *, TranslationUnit *>
|
|
|
|
Indexer::getDefinitionFor(Entity Ent) {
|
|
|
|
DefMapTy::iterator I = DefMap.find(Ent);
|
|
|
|
if (I == DefMap.end())
|
|
|
|
return std::make_pair((FunctionDecl *)0, (TranslationUnit *)0);
|
|
|
|
else
|
|
|
|
return I->second;
|
|
|
|
}
|