2010-08-28 05:57:20 +08:00
|
|
|
//===- CIndexCXX.cpp - Clang-C Source Indexing Library --------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the libclang support for C++ cursors.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "CIndexer.h"
|
|
|
|
#include "CXCursor.h"
|
|
|
|
#include "CXType.h"
|
|
|
|
#include "clang/AST/DeclCXX.h"
|
2010-09-01 06:12:17 +08:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
2010-08-28 05:57:20 +08:00
|
|
|
|
|
|
|
using namespace clang;
|
2010-09-02 08:07:54 +08:00
|
|
|
using namespace clang::cxcursor;
|
2010-08-28 05:57:20 +08:00
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
unsigned clang_isVirtualBase(CXCursor C) {
|
|
|
|
if (C.kind != CXCursor_CXXBaseSpecifier)
|
|
|
|
return 0;
|
|
|
|
|
2013-01-12 05:01:49 +08:00
|
|
|
const CXXBaseSpecifier *B = getCursorCXXBaseSpecifier(C);
|
2010-08-28 05:57:20 +08:00
|
|
|
return B->isVirtual();
|
|
|
|
}
|
|
|
|
|
|
|
|
enum CX_CXXAccessSpecifier clang_getCXXAccessSpecifier(CXCursor C) {
|
2011-10-01 01:58:23 +08:00
|
|
|
AccessSpecifier spec = AS_none;
|
|
|
|
|
2013-04-12 01:02:10 +08:00
|
|
|
if (C.kind == CXCursor_CXXAccessSpecifier || clang_isDeclaration(C.kind))
|
2011-10-01 01:58:23 +08:00
|
|
|
spec = getCursorDecl(C)->getAccess();
|
|
|
|
else if (C.kind == CXCursor_CXXBaseSpecifier)
|
|
|
|
spec = getCursorCXXBaseSpecifier(C)->getAccessSpecifier();
|
|
|
|
else
|
2010-08-28 05:57:20 +08:00
|
|
|
return CX_CXXInvalidAccessSpecifier;
|
|
|
|
|
2011-10-01 01:58:23 +08:00
|
|
|
switch (spec) {
|
2010-08-28 05:57:20 +08:00
|
|
|
case AS_public: return CX_CXXPublic;
|
|
|
|
case AS_protected: return CX_CXXProtected;
|
|
|
|
case AS_private: return CX_CXXPrivate;
|
|
|
|
case AS_none: return CX_CXXInvalidAccessSpecifier;
|
|
|
|
}
|
2012-01-17 14:56:22 +08:00
|
|
|
|
|
|
|
llvm_unreachable("Invalid AccessSpecifier!");
|
2010-08-28 05:57:20 +08:00
|
|
|
}
|
|
|
|
|
2010-09-01 06:12:17 +08:00
|
|
|
enum CXCursorKind clang_getTemplateCursorKind(CXCursor C) {
|
|
|
|
using namespace clang::cxcursor;
|
|
|
|
|
|
|
|
switch (C.kind) {
|
|
|
|
case CXCursor_ClassTemplate:
|
|
|
|
case CXCursor_FunctionTemplate:
|
2013-01-24 01:25:27 +08:00
|
|
|
if (const TemplateDecl *Template
|
2010-09-01 06:12:17 +08:00
|
|
|
= dyn_cast_or_null<TemplateDecl>(getCursorDecl(C)))
|
2013-01-12 03:28:44 +08:00
|
|
|
return MakeCXCursor(Template->getTemplatedDecl(), getCursorTU(C)).kind;
|
2010-09-01 06:12:17 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CXCursor_ClassTemplatePartialSpecialization:
|
2013-01-24 01:25:27 +08:00
|
|
|
if (const ClassTemplateSpecializationDecl *PartialSpec
|
2012-09-01 06:18:20 +08:00
|
|
|
= dyn_cast_or_null<ClassTemplatePartialSpecializationDecl>(
|
|
|
|
getCursorDecl(C))) {
|
|
|
|
switch (PartialSpec->getTagKind()) {
|
|
|
|
case TTK_Interface:
|
|
|
|
case TTK_Struct: return CXCursor_StructDecl;
|
|
|
|
case TTK_Class: return CXCursor_ClassDecl;
|
|
|
|
case TTK_Union: return CXCursor_UnionDecl;
|
|
|
|
case TTK_Enum: return CXCursor_NoDeclFound;
|
|
|
|
}
|
2010-09-01 06:12:17 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CXCursor_NoDeclFound;
|
|
|
|
}
|
|
|
|
|
2010-09-02 08:07:54 +08:00
|
|
|
CXCursor clang_getSpecializedCursorTemplate(CXCursor C) {
|
|
|
|
if (!clang_isDeclaration(C.kind))
|
|
|
|
return clang_getNullCursor();
|
|
|
|
|
2013-01-24 01:25:27 +08:00
|
|
|
const Decl *D = getCursorDecl(C);
|
2010-09-02 08:07:54 +08:00
|
|
|
if (!D)
|
|
|
|
return clang_getNullCursor();
|
|
|
|
|
|
|
|
Decl *Template = 0;
|
2013-01-24 01:25:27 +08:00
|
|
|
if (const CXXRecordDecl *CXXRecord = dyn_cast<CXXRecordDecl>(D)) {
|
|
|
|
if (const ClassTemplatePartialSpecializationDecl *PartialSpec
|
2010-09-02 08:07:54 +08:00
|
|
|
= dyn_cast<ClassTemplatePartialSpecializationDecl>(CXXRecord))
|
|
|
|
Template = PartialSpec->getSpecializedTemplate();
|
2013-01-24 01:25:27 +08:00
|
|
|
else if (const ClassTemplateSpecializationDecl *ClassSpec
|
2010-09-02 08:07:54 +08:00
|
|
|
= dyn_cast<ClassTemplateSpecializationDecl>(CXXRecord)) {
|
|
|
|
llvm::PointerUnion<ClassTemplateDecl *,
|
|
|
|
ClassTemplatePartialSpecializationDecl *> Result
|
|
|
|
= ClassSpec->getSpecializedTemplateOrPartial();
|
|
|
|
if (Result.is<ClassTemplateDecl *>())
|
|
|
|
Template = Result.get<ClassTemplateDecl *>();
|
|
|
|
else
|
|
|
|
Template = Result.get<ClassTemplatePartialSpecializationDecl *>();
|
|
|
|
|
|
|
|
} else
|
|
|
|
Template = CXXRecord->getInstantiatedFromMemberClass();
|
2013-01-24 01:25:27 +08:00
|
|
|
} else if (const FunctionDecl *Function = dyn_cast<FunctionDecl>(D)) {
|
2010-09-02 08:07:54 +08:00
|
|
|
Template = Function->getPrimaryTemplate();
|
|
|
|
if (!Template)
|
|
|
|
Template = Function->getInstantiatedFromMemberFunction();
|
2013-01-24 01:25:27 +08:00
|
|
|
} else if (const VarDecl *Var = dyn_cast<VarDecl>(D)) {
|
2010-09-02 08:07:54 +08:00
|
|
|
if (Var->isStaticDataMember())
|
|
|
|
Template = Var->getInstantiatedFromStaticDataMember();
|
2013-01-24 01:25:27 +08:00
|
|
|
} else if (const RedeclarableTemplateDecl *Tmpl
|
2010-09-02 08:07:54 +08:00
|
|
|
= dyn_cast<RedeclarableTemplateDecl>(D))
|
|
|
|
Template = Tmpl->getInstantiatedFromMemberTemplate();
|
|
|
|
|
|
|
|
if (!Template)
|
|
|
|
return clang_getNullCursor();
|
|
|
|
|
2013-01-12 03:28:44 +08:00
|
|
|
return MakeCXCursor(Template, getCursorTU(C));
|
2010-09-02 08:07:54 +08:00
|
|
|
}
|
|
|
|
|
2010-08-28 05:57:20 +08:00
|
|
|
} // end extern "C"
|