blob: f284b248cdf8b9724448d7297a8712ade1c65e1c [file] [log] [blame]
Ted Kremenek87553c42010-01-15 20:35:54 +00001//===- CXCursor.cpp - Routines for manipulating CXCursors -----------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
Douglas Gregor6c8959b2010-01-16 14:00:32 +000010// This file defines routines for manipulating CXCursors. It should be the
11// only file that has internal knowledge of the encoding of the data in
12// CXCursor.
Ted Kremenek87553c42010-01-15 20:35:54 +000013//
14//===----------------------------------------------------------------------===//
15
16#include "CXCursor.h"
17#include "clang/AST/Decl.h"
Douglas Gregorc58d05b2010-01-15 21:56:13 +000018#include "clang/AST/DeclObjC.h"
19#include "clang/AST/Expr.h"
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000020#include "llvm/Support/ErrorHandling.h"
Ted Kremenek87553c42010-01-15 20:35:54 +000021
22using namespace clang;
23
24CXCursor cxcursor::MakeCXCursor(CXCursorKind K, Decl *D) {
Douglas Gregorc58d05b2010-01-15 21:56:13 +000025 CXCursor C = { K, { D, 0, 0 } };
Ted Kremenek87553c42010-01-15 20:35:54 +000026 return C;
27}
28
Douglas Gregor7ecd0202010-01-18 23:41:10 +000029CXCursor cxcursor::MakeCXCursor(CXCursorKind K, Decl *D, Stmt *S,
30 ASTContext &Context) {
Ted Kremenek87553c42010-01-15 20:35:54 +000031 assert(clang_isReference(K));
Douglas Gregor7ecd0202010-01-18 23:41:10 +000032 CXCursor C = { K, { D, S, &Context } };
Ted Kremenek87553c42010-01-15 20:35:54 +000033 return C;
34}
35
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000036static CXCursorKind GetCursorKind(Decl *D) {
37 switch (D->getKind()) {
Ted Kremenek9cec0002010-01-16 01:44:12 +000038 case Decl::Enum: return CXCursor_EnumDecl;
39 case Decl::EnumConstant: return CXCursor_EnumConstantDecl;
40 case Decl::Field: return CXCursor_FieldDecl;
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000041 case Decl::Function:
Douglas Gregor6b8232f2010-01-19 19:34:47 +000042 return CXCursor_FunctionDecl;
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000043 case Decl::ObjCCategory: return CXCursor_ObjCCategoryDecl;
Douglas Gregor6b8232f2010-01-19 19:34:47 +000044 case Decl::ObjCCategoryImpl: return CXCursor_ObjCCategoryImplDecl;
Ted Kremenek9cec0002010-01-16 01:44:12 +000045 case Decl::ObjCClass:
46 // FIXME
Douglas Gregoraccb1832010-01-19 22:07:56 +000047 return CXCursor_UnexposedDecl;
Ted Kremenek0ea923c2010-01-18 22:07:45 +000048 case Decl::ObjCForwardProtocol:
49 // FIXME
Douglas Gregoraccb1832010-01-19 22:07:56 +000050 return CXCursor_UnexposedDecl;
Douglas Gregor6b8232f2010-01-19 19:34:47 +000051 case Decl::ObjCImplementation: return CXCursor_ObjCImplementationDecl;
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000052 case Decl::ObjCInterface: return CXCursor_ObjCInterfaceDecl;
Ted Kremenek9cec0002010-01-16 01:44:12 +000053 case Decl::ObjCIvar: return CXCursor_ObjCIvarDecl;
54 case Decl::ObjCMethod:
55 return cast<ObjCMethodDecl>(D)->isInstanceMethod()
56 ? CXCursor_ObjCInstanceMethodDecl : CXCursor_ObjCClassMethodDecl;
Ted Kremenek13751dc2010-01-16 02:08:29 +000057 case Decl::ObjCProperty: return CXCursor_ObjCPropertyDecl;
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000058 case Decl::ObjCProtocol: return CXCursor_ObjCProtocolDecl;
Ted Kremenek9cec0002010-01-16 01:44:12 +000059 case Decl::ParmVar: return CXCursor_ParmDecl;
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000060 case Decl::Typedef: return CXCursor_TypedefDecl;
61 case Decl::Var: return CXCursor_VarDecl;
62 default:
63 if (TagDecl *TD = dyn_cast<TagDecl>(D)) {
64 switch (TD->getTagKind()) {
65 case TagDecl::TK_struct: return CXCursor_StructDecl;
66 case TagDecl::TK_class: return CXCursor_ClassDecl;
67 case TagDecl::TK_union: return CXCursor_UnionDecl;
68 case TagDecl::TK_enum: return CXCursor_EnumDecl;
69 }
70 }
Douglas Gregoraccb1832010-01-19 22:07:56 +000071
72 return CXCursor_UnexposedDecl;
Ted Kremenekc2aa0f12010-01-16 00:36:30 +000073 }
74
75 llvm_unreachable("Invalid Decl");
76 return CXCursor_NotImplemented;
77}
78
79CXCursor cxcursor::MakeCXCursor(Decl *D) {
80 return MakeCXCursor(GetCursorKind(D), D);
81}
82
Douglas Gregor6c8959b2010-01-16 14:00:32 +000083CXCursor cxcursor::MakeCursorObjCSuperClassRef(ObjCInterfaceDecl *Super,
84 SourceLocation Loc) {
85 void *RawLoc = reinterpret_cast<void *>(Loc.getRawEncoding());
86 CXCursor C = { CXCursor_ObjCSuperClassRef, { Super, RawLoc, 0 } };
87 return C;
88}
89
90std::pair<ObjCInterfaceDecl *, SourceLocation>
91cxcursor::getCursorObjCSuperClassRef(CXCursor C) {
92 assert(C.kind == CXCursor_ObjCSuperClassRef);
93 return std::make_pair(static_cast<ObjCInterfaceDecl *>(C.data[0]),
94 SourceLocation::getFromRawEncoding(
95 reinterpret_cast<uintptr_t>(C.data[1])));
96}
97
Douglas Gregoref6eb842010-01-16 15:44:18 +000098CXCursor cxcursor::MakeCursorObjCProtocolRef(ObjCProtocolDecl *Super,
99 SourceLocation Loc) {
100 void *RawLoc = reinterpret_cast<void *>(Loc.getRawEncoding());
101 CXCursor C = { CXCursor_ObjCProtocolRef, { Super, RawLoc, 0 } };
102 return C;
103}
104
105std::pair<ObjCProtocolDecl *, SourceLocation>
106cxcursor::getCursorObjCProtocolRef(CXCursor C) {
107 assert(C.kind == CXCursor_ObjCProtocolRef);
108 return std::make_pair(static_cast<ObjCProtocolDecl *>(C.data[0]),
109 SourceLocation::getFromRawEncoding(
110 reinterpret_cast<uintptr_t>(C.data[1])));
111}
112
Douglas Gregor46d66142010-01-16 17:14:40 +0000113CXCursor cxcursor::MakeCursorObjCClassRef(ObjCInterfaceDecl *Class,
114 SourceLocation Loc) {
115 void *RawLoc = reinterpret_cast<void *>(Loc.getRawEncoding());
116 CXCursor C = { CXCursor_ObjCClassRef, { Class, RawLoc, 0 } };
117 return C;
118}
119
120std::pair<ObjCInterfaceDecl *, SourceLocation>
121cxcursor::getCursorObjCClassRef(CXCursor C) {
122 assert(C.kind == CXCursor_ObjCClassRef);
123 return std::make_pair(static_cast<ObjCInterfaceDecl *>(C.data[0]),
124 SourceLocation::getFromRawEncoding(
125 reinterpret_cast<uintptr_t>(C.data[1])));
126}
127
Douglas Gregorc58d05b2010-01-15 21:56:13 +0000128Decl *cxcursor::getCursorDecl(CXCursor Cursor) {
129 return (Decl *)Cursor.data[0];
130}
131
132Expr *cxcursor::getCursorExpr(CXCursor Cursor) {
133 return dyn_cast_or_null<Expr>(getCursorStmt(Cursor));
134}
135
136Stmt *cxcursor::getCursorStmt(CXCursor Cursor) {
Douglas Gregoref6eb842010-01-16 15:44:18 +0000137 if (Cursor.kind == CXCursor_ObjCSuperClassRef ||
Douglas Gregor46d66142010-01-16 17:14:40 +0000138 Cursor.kind == CXCursor_ObjCProtocolRef ||
139 Cursor.kind == CXCursor_ObjCClassRef)
Douglas Gregor6c8959b2010-01-16 14:00:32 +0000140 return 0;
141
Douglas Gregorc58d05b2010-01-15 21:56:13 +0000142 return (Stmt *)Cursor.data[1];
143}
144
Douglas Gregor7ecd0202010-01-18 23:41:10 +0000145ASTContext &cxcursor::getCursorContext(CXCursor Cursor) {
146 switch (Cursor.kind) {
147 case CXCursor_TypedefDecl:
148 case CXCursor_StructDecl:
149 case CXCursor_UnionDecl:
150 case CXCursor_ClassDecl:
151 case CXCursor_EnumDecl:
152 case CXCursor_FieldDecl:
153 case CXCursor_EnumConstantDecl:
154 case CXCursor_FunctionDecl:
155 case CXCursor_VarDecl:
156 case CXCursor_ParmDecl:
157 case CXCursor_ObjCInterfaceDecl:
158 case CXCursor_ObjCCategoryDecl:
159 case CXCursor_ObjCProtocolDecl:
160 case CXCursor_ObjCPropertyDecl:
161 case CXCursor_ObjCIvarDecl:
162 case CXCursor_ObjCInstanceMethodDecl:
163 case CXCursor_ObjCClassMethodDecl:
Douglas Gregor6b8232f2010-01-19 19:34:47 +0000164 case CXCursor_ObjCImplementationDecl:
165 case CXCursor_ObjCCategoryImplDecl:
Douglas Gregoraccb1832010-01-19 22:07:56 +0000166 case CXCursor_UnexposedDecl:
Douglas Gregor7ecd0202010-01-18 23:41:10 +0000167 return static_cast<Decl *>(Cursor.data[0])->getASTContext();
168
169 case CXCursor_ObjCSuperClassRef:
170 case CXCursor_ObjCProtocolRef:
171 case CXCursor_ObjCClassRef:
172 return static_cast<Decl *>(Cursor.data[0])->getASTContext();
173
174 case CXCursor_ObjCSelectorRef:
175 case CXCursor_ObjCIvarRef:
176 case CXCursor_VarRef:
177 case CXCursor_FunctionRef:
178 case CXCursor_EnumConstantRef:
179 case CXCursor_MemberRef:
180 return *static_cast<ASTContext *>(Cursor.data[2]);
181
182 case CXCursor_InvalidFile:
183 case CXCursor_NoDeclFound:
184 case CXCursor_NotImplemented:
185 llvm_unreachable("No context in an invalid cursor");
186 }
187
188 llvm_unreachable("No context available");
Douglas Gregorc58d05b2010-01-15 21:56:13 +0000189}
190
Douglas Gregorc58d05b2010-01-15 21:56:13 +0000191bool cxcursor::operator==(CXCursor X, CXCursor Y) {
192 return X.kind == Y.kind && X.data[0] == Y.data[0] && X.data[1] == Y.data[1] &&
193 X.data[2] == Y.data[2];
Douglas Gregor6c8959b2010-01-16 14:00:32 +0000194}