#![allow(non_upper_case_globals, dead_code)]
use clang_sys::*;
use std::{mem, ptr, slice};
use std::ffi::{CStr, CString};
use std::fmt;
use std::hash::Hash;
use std::hash::Hasher;
use std::os::raw::{c_char, c_int, c_uint, c_ulong};
pub use c3_clang_extensions::*;
#[derive(Copy, Clone)]
pub struct Cursor {
pub(crate) x: CXCursor,
}
impl fmt::Debug for Cursor {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
if !self.is_valid() {
write!(fmt, "Invalid::")?;
}
match self.kind() {
CXCursor_NoDeclFound => write!(fmt, "NoDecl")?,
CXCursor_UnexposedExpr | CXCursor_UnexposedAttr => write!(fmt, "{:?}!", self.extended_kind())?,
_ => write!(fmt, "{}", kind_to_str(self.kind()))?,
}
write!(fmt, "({:?} {}", self.spelling(), self.location())?;
write!(fmt, ")")
}
}
impl Cursor {
pub fn usr(&self) -> Option<String> {
let s = unsafe { cxstring_into_string(clang_getCursorUSR(self.x)) };
if s.is_empty() { None } else { Some(s) }
}
pub fn is_declaration(&self) -> bool {
unsafe { clang_isDeclaration(self.kind()) != 0 }
}
pub fn null() -> Self {
Cursor {
x: unsafe { clang_getNullCursor() },
}
}
pub fn spelling(&self) -> String {
unsafe { cxstring_into_string(clang_getCursorSpelling(self.x)) }
}
pub fn display_name(&self) -> String {
unsafe { cxstring_into_string(clang_getCursorDisplayName(self.x)) }
}
pub fn mangling(&self) -> String {
if true {
unsafe { cxstring_into_string(clang_Cursor_getMangling(self.x)) }
} else {
self.spelling()
}
}
pub fn is_builtin(&self) -> bool {
let (file, _, _, _) = self.location().location();
file.name().is_none()
}
pub fn lexical_parent(&self) -> Cursor {
unsafe {
Cursor {
x: clang_getCursorLexicalParent(self.x),
}
}
}
pub fn fallible_semantic_parent(&self) -> Option<Cursor> {
let sp = unsafe {
Cursor {
x: clang_getCursorSemanticParent(self.x),
}
};
if sp == *self || !sp.is_valid() {
return None;
}
Some(sp)
}
pub fn semantic_parent(&self) -> Cursor {
self.fallible_semantic_parent().unwrap()
}
pub fn num_template_args(&self) -> Option<u32> {
self.cur_type()
.num_template_args()
.or_else(|| {
let n: c_int =
unsafe { clang_Cursor_getNumTemplateArguments(self.x) };
if n >= 0 {
Some(n as u32)
} else {
debug_assert_eq!(n, -1);
None
}
})
.or_else(|| {
let canonical = self.canonical();
if canonical.is_some() && canonical.unwrap() != *self {
canonical.unwrap().num_template_args()
} else {
None
}
})
}
pub fn translation_unit(&self) -> Cursor {
assert!(self.is_valid());
unsafe {
let tu = clang_Cursor_getTranslationUnit(self.x);
let cursor = Cursor {
x: clang_getTranslationUnitCursor(tu),
};
assert!(cursor.is_valid());
cursor
}
}
pub fn is_toplevel(&self) -> bool {
let mut semantic_parent = self.fallible_semantic_parent();
while semantic_parent.is_some() &&
(semantic_parent.unwrap().kind() == CXCursor_Namespace ||
semantic_parent.unwrap().kind() == CXCursor_NamespaceAlias ||
semantic_parent.unwrap().kind() == CXCursor_NamespaceRef) {
semantic_parent = semantic_parent.unwrap()
.fallible_semantic_parent();
}
let tu = self.translation_unit();
semantic_parent == tu.fallible_semantic_parent()
}
pub fn is_template_like(&self) -> bool {
match self.kind() {
CXCursor_ClassTemplate |
CXCursor_ClassTemplatePartialSpecialization |
CXCursor_TypeAliasTemplateDecl => true,
_ => false,
}
}
pub fn kind(&self) -> CXCursorKind {
self.x.kind
}
pub fn extended_kind(&self) -> CusorKindExt {
unsafe {
c3_Cursor_getKindExt(self.x)
}
}
pub fn is_definition(&self) -> bool {
unsafe { clang_isCursorDefinition(self.x) != 0 }
}
pub fn is_template_specialization(&self) -> bool {
self.specialized().is_some()
}
pub fn is_fully_specialized_template(&self) -> bool {
self.is_template_specialization() &&
self.kind() != CXCursor_ClassTemplatePartialSpecialization &&
self.num_template_args().unwrap_or(0) > 0
}
pub fn is_in_non_fully_specialized_template(&self) -> bool {
if self.is_toplevel() {
return false;
}
let parent = self.semantic_parent();
if parent.is_fully_specialized_template() {
return false;
}
if !parent.is_template_like() {
return parent.is_in_non_fully_specialized_template();
}
true
}
pub fn is_valid(&self) -> bool {
unsafe { clang_isInvalid(self.kind()) == 0 }
}
pub fn location(&self) -> SourceLocation {
unsafe {
SourceLocation {
x: clang_getCursorLocation(self.x),
}
}
}
pub fn extent(&self) -> CXSourceRange {
unsafe { clang_getCursorExtent(self.x) }
}
pub fn raw_comment(&self) -> Option<String> {
let s = unsafe {
cxstring_into_string(clang_Cursor_getRawCommentText(self.x))
};
if s.is_empty() { None } else { Some(s) }
}
pub fn comment(&self) -> Comment {
unsafe {
Comment {
x: clang_Cursor_getParsedComment(self.x),
}
}
}
pub fn cur_type(&self) -> Type {
unsafe {
Type {
x: clang_getCursorType(self.x),
}
}
}
pub fn definition(&self) -> Option<Cursor> {
unsafe {
let ret = Cursor {
x: clang_getCursorDefinition(self.x),
};
if ret.is_valid() && ret.kind() != CXCursor_NoDeclFound {
Some(ret)
} else {
None
}
}
}
pub fn referenced(&self) -> Option<Cursor> {
unsafe {
let ret = Cursor {
x: clang_getCursorReferenced(self.x),
};
if ret.is_valid() { Some(ret) } else { None }
}
}
pub fn canonical(&self) -> Option<Cursor> {
let cur = unsafe {
Cursor {
x: clang_getCanonicalCursor(self.x),
}
};
if cur.is_valid() && cur.kind() != CXCursor_NoDeclFound {
Some(cur)
} else {
None
}
}
pub fn specialized(&self) -> Option<Cursor> {
unsafe {
let ret = Cursor {
x: clang_getSpecializedCursorTemplate(self.x),
};
if ret.is_valid() { Some(ret) } else { None }
}
}
pub fn template_kind(&self) -> CXCursorKind {
unsafe { clang_getTemplateCursorKind(self.x) }
}
pub fn visit<Visitor>(&self, mut visitor: Visitor)
where Visitor: FnMut(Cursor) -> CXChildVisitResult,
{
unsafe {
clang_visitChildren(self.x,
visit_children::<Visitor>,
mem::transmute(&mut visitor));
}
}
pub fn collect_children(&self) -> Vec<Cursor> {
let mut children = vec![];
self.visit(|c| {
children.push(c);
CXChildVisit_Continue
});
children
}
pub fn has_children(&self) -> bool {
let mut has_children = false;
self.visit(|_| {
has_children = true;
CXChildVisit_Break
});
has_children
}
pub fn has_at_least_num_children(&self, n: usize) -> bool {
assert!(n > 0);
let mut num_left = n;
self.visit(|_| {
num_left -= 1;
if num_left == 0 {
CXChildVisit_Break
} else {
CXChildVisit_Continue
}
});
num_left == 0
}
pub fn contains_cursor(&self, kind: CXCursorKind) -> bool {
let mut found = false;
self.visit(|c| if c.kind() == kind {
found = true;
CXChildVisit_Break
} else {
CXChildVisit_Continue
});
found
}
pub fn is_inlined_function(&self) -> bool {
unsafe { clang_Cursor_isFunctionInlined(self.x) != 0 }
}
pub fn bit_width(&self) -> Option<u32> {
unsafe {
let w = clang_getFieldDeclBitWidth(self.x);
if w == -1 { None } else { Some(w as u32) }
}
}
pub fn enum_type(&self) -> Option<Type> {
unsafe {
let t = Type {
x: clang_getEnumDeclIntegerType(self.x),
};
if t.is_valid() { Some(t) } else { None }
}
}
pub fn enum_val_signed(&self) -> Option<i64> {
unsafe {
if self.kind() == CXCursor_EnumConstantDecl {
Some(clang_getEnumConstantDeclValue(self.x) as i64)
} else {
None
}
}
}
pub fn enum_val_unsigned(&self) -> Option<u64> {
unsafe {
if self.kind() == CXCursor_EnumConstantDecl {
Some(clang_getEnumConstantDeclUnsignedValue(self.x) as u64)
} else {
None
}
}
}
pub fn typedef_type(&self) -> Option<Type> {
let inner = Type {
x: unsafe { clang_getTypedefDeclUnderlyingType(self.x) },
};
if inner.is_valid() { Some(inner) } else { None }
}
pub fn linkage(&self) -> CXLinkageKind {
unsafe { clang_getCursorLinkage(self.x) }
}
pub fn visibility(&self) -> CXVisibilityKind {
if true {
unsafe { clang_getCursorVisibility(self.x) }
} else {
CXVisibility_Default
}
}
pub fn args(&self) -> Option<Vec<Cursor>> {
unsafe {
let w = clang_Cursor_getNumArguments(self.x);
if w == -1 {
None
} else {
let num = w as u32;
let mut args = vec![];
for i in 0..num {
args.push(Cursor {
x: clang_Cursor_getArgument(self.x, i as c_uint),
});
}
Some(args)
}
}
}
pub fn num_args(&self) -> Result<u32, ()> {
unsafe {
let w = clang_Cursor_getNumArguments(self.x);
if w == -1 { Err(()) } else { Ok(w as u32) }
}
}
pub fn access_specifier(&self) -> CX_CXXAccessSpecifier {
unsafe { clang_getCXXAccessSpecifier(self.x) }
}
pub fn is_mutable_field(&self) -> bool {
unsafe { clang_CXXField_isMutable(self.x) != 0 }
}
pub fn offset_of_field(&self) -> Result<usize, LayoutError> {
if false {
return Err(LayoutError::from(-1));
}
let offset = unsafe { clang_Cursor_getOffsetOfField(self.x) };
if offset < 0 {
Err(LayoutError::from(offset as i32))
} else {
Ok(offset as usize)
}
}
pub fn method_is_static(&self) -> bool {
unsafe { clang_CXXMethod_isStatic(self.x) != 0 }
}
pub fn method_is_const(&self) -> bool {
unsafe { clang_CXXMethod_isConst(self.x) != 0 }
}
pub fn method_is_virtual(&self) -> bool {
unsafe { clang_CXXMethod_isVirtual(self.x) != 0 }
}
pub fn is_virtual_base(&self) -> bool {
unsafe { clang_isVirtualBase(self.x) != 0 }
}
pub fn evaluate(&self) -> Option<EvalResult> {
EvalResult::new(*self)
}
pub fn ret_type(&self) -> Option<Type> {
let rt = Type {
x: unsafe { clang_getCursorResultType(self.x) },
};
if rt.is_valid() { Some(rt) } else { None }
}
}
extern "C" fn visit_children<Visitor>(cur: CXCursor,
_parent: CXCursor,
data: CXClientData)
-> CXChildVisitResult
where Visitor: FnMut(Cursor) -> CXChildVisitResult,
{
let func: &mut Visitor = unsafe { &mut *data.cast::<Visitor>() };
let child = Cursor {
x: cur,
};
(*func)(child)
}
impl PartialEq for Cursor {
fn eq(&self, other: &Cursor) -> bool {
unsafe { clang_equalCursors(self.x, other.x) == 1 }
}
}
impl Eq for Cursor {}
impl Hash for Cursor {
fn hash<H: Hasher>(&self, state: &mut H) {
unsafe { clang_hashCursor(self.x) }.hash(state);
}
}
#[derive(Clone, Copy)]
pub struct Type {
pub(crate) x: CXType,
}
impl PartialEq for Type {
fn eq(&self, other: &Self) -> bool {
unsafe { clang_equalTypes(self.x, other.x) != 0 }
}
}
impl Eq for Type {}
impl fmt::Debug for Type {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
match self.kind() {
CXType_Unexposed => write!(fmt, "{:?}!<{:?}",
self.extended_type(),
self.spelling())?,
_ => write!(fmt, "{}<{:?}",
type_to_str(self.kind()),
self.spelling())?,
}
if let Some(decl) = self.declaration() {
write!(fmt, " decl={decl:?}")?;
if let Some(canon) = decl.canonical() {
if decl != canon {
write!(fmt, " canon={canon:?}")?;
}
}
} else {
let canon = self.canonical_type();
if canon != *self {
write!(fmt, " canon={canon:?}")?;
}
}
if let Some(point) = self.pointee_type() {
write!(fmt, " ->{point:?}")?;
}
if let Some(point) = self.ret_type() {
write!(fmt, " ret={point:?}")?;
}
write!(fmt, ">")
}
}
#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
pub enum LayoutError {
Invalid,
Incomplete,
Dependent,
NotConstantSize,
InvalidFieldName,
Unknown,
}
impl ::std::convert::From<i32> for LayoutError {
fn from(val: i32) -> Self {
use self::LayoutError::*;
match val {
CXTypeLayoutError_Invalid => Invalid,
CXTypeLayoutError_Incomplete => Incomplete,
CXTypeLayoutError_Dependent => Dependent,
CXTypeLayoutError_NotConstantSize => NotConstantSize,
CXTypeLayoutError_InvalidFieldName => InvalidFieldName,
_ => Unknown,
}
}
}
impl Type {
pub fn kind(&self) -> CXTypeKind {
self.x.kind
}
pub fn extended_type(&self) -> ClangTypeExt {
unsafe {
c3_CursorType_getKindExt(self.x)
}
}
pub fn declaration(&self) -> Option<Cursor> {
let cur = unsafe {
Cursor {
x: clang_getTypeDeclaration(self.x),
}
};
if cur.is_valid() && cur.kind() != CXCursor_NoDeclFound {
Some(cur)
} else {
None
}
}
pub fn canonical_declaration(&self,
location: Option<&Cursor>)
-> Option<CanonicalTypeDeclaration> {
let mut declaration = self.declaration();
if declaration.is_none() {
if let Some(location) = location {
let mut location = *location;
if let Some(referenced) = location.referenced() {
location = referenced;
}
if location.is_template_like() {
declaration = Some(location);
}
}
}
let canonical = declaration.and_then(|d|d.canonical());
canonical.map(|canonical| CanonicalTypeDeclaration(*self, canonical))
}
pub fn spelling(&self) -> String {
unsafe { cxstring_into_string(clang_getTypeSpelling(self.x)) }
}
pub fn is_const(&self) -> bool {
unsafe { clang_isConstQualifiedType(self.x) != 0 }
}
pub fn size(&self) -> usize {
unsafe {
let val = clang_Type_getSizeOf(self.x);
if val < 0 { 0 } else { val as usize }
}
}
pub fn fallible_size(&self) -> Result<usize, LayoutError> {
let val = unsafe { clang_Type_getSizeOf(self.x) };
if val < 0 {
Err(LayoutError::from(val as i32))
} else {
Ok(val as usize)
}
}
pub fn align(&self) -> usize {
unsafe {
let val = clang_Type_getAlignOf(self.x);
if val < 0 { 0 } else { val as usize }
}
}
pub fn fallible_align(&self) -> Result<usize, LayoutError> {
unsafe {
let val = clang_Type_getAlignOf(self.x);
if val < 0 {
Err(LayoutError::from(val as i32))
} else {
Ok(val as usize)
}
}
}
pub fn num_template_args(&self) -> Option<u32> {
let n = unsafe { clang_Type_getNumTemplateArguments(self.x) };
if n >= 0 {
Some(n as u32)
} else {
debug_assert_eq!(n, -1);
None
}
}
pub fn template_args(&self) -> Option<TypeTemplateArgIterator> {
self.num_template_args().map(|n| {
TypeTemplateArgIterator {
x: self.x,
length: n,
index: 0,
}
})
}
pub fn pointee_type(&self) -> Option<Type> {
match self.kind() {
CXType_Pointer |
CXType_RValueReference |
CXType_LValueReference |
CXType_MemberPointer |
CXType_ObjCObjectPointer => {
let ret = Type {
x: unsafe { clang_getPointeeType(self.x) },
};
debug_assert!(ret.is_valid());
Some(ret)
}
_ => None,
}
}
pub fn elem_type(&self) -> Option<Type> {
let current_type = Type {
x: unsafe { clang_getElementType(self.x) },
};
if current_type.is_valid() {
Some(current_type)
} else {
None
}
}
pub fn num_elements(&self) -> Option<usize> {
let num_elements_returned = unsafe { clang_getNumElements(self.x) };
if num_elements_returned != -1 {
Some(num_elements_returned as usize)
} else {
None
}
}
pub fn canonical_type(&self) -> Type {
unsafe {
Type {
x: clang_getCanonicalType(self.x),
}
}
}
pub fn is_variadic(&self) -> bool {
unsafe { clang_isFunctionTypeVariadic(self.x) != 0 }
}
pub fn ret_type(&self) -> Option<Type> {
let rt = Type {
x: unsafe { clang_getResultType(self.x) },
};
if rt.is_valid() { Some(rt) } else { None }
}
pub fn call_conv(&self) -> CXCallingConv {
unsafe { clang_getFunctionTypeCallingConv(self.x) }
}
pub fn named(&self) -> Type {
unsafe {
Type {
x: if true {
clang_Type_getNamedType(self.x)
} else {
self.x
},
}
}
}
pub fn is_valid(&self) -> bool {
self.kind() != CXType_Invalid
}
pub fn is_valid_and_exposed(&self) -> bool {
self.is_valid() && self.kind() != CXType_Unexposed
}
pub fn is_fully_instantiated_template(&self) -> bool {
self.template_args().map_or(false, |args| args.len() > 0) &&
match self.declaration().map_or(CXCursor_NoDeclFound, |d|d.kind()) {
CXCursor_ClassTemplatePartialSpecialization |
CXCursor_TypeAliasTemplateDecl |
CXCursor_TemplateTemplateParameter => false,
_ => true,
}
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub struct CanonicalTypeDeclaration(Type, Cursor);
impl CanonicalTypeDeclaration {
pub fn ty(&self) -> &Type {
&self.0
}
pub fn cursor(&self) -> &Cursor {
&self.1
}
}
pub struct TypeTemplateArgIterator {
x: CXType,
length: u32,
index: u32,
}
impl Iterator for TypeTemplateArgIterator {
type Item = Type;
fn next(&mut self) -> Option<Type> {
if self.index < self.length {
let idx = self.index as c_uint;
self.index += 1;
Some(Type {
x: unsafe { clang_Type_getTemplateArgumentAsType(self.x, idx) },
})
} else {
None
}
}
}
impl ExactSizeIterator for TypeTemplateArgIterator {
fn len(&self) -> usize {
assert!(self.index <= self.length);
(self.length - self.index) as usize
}
}
pub struct SourceLocation {
pub(crate) x: CXSourceLocation,
}
impl fmt::Debug for SourceLocation {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let (_,_,_,off) = self.location();
off.fmt(f)
}
}
impl SourceLocation {
pub fn location(&self) -> (File, usize, usize, usize) {
unsafe {
let mut file = mem::zeroed();
let mut line = 0;
let mut col = 0;
let mut off = 0;
clang_getSpellingLocation(self.x,
&mut file,
&mut line,
&mut col,
&mut off);
(File {
x: file,
},
line as usize,
col as usize,
off as usize)
}
}
}
impl fmt::Display for SourceLocation {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let (file, line, col, _) = self.location();
if let Some(name) = file.name() {
write!(f, "{name}:{line}:{col}")
} else {
"builtin definitions".fmt(f)
}
}
}
pub struct Comment {
x: CXComment,
}
impl Comment {
pub fn kind(&self) -> CXCommentKind {
unsafe { clang_Comment_getKind(self.x) }
}
pub fn get_children(&self) -> CommentChildrenIterator {
CommentChildrenIterator {
parent: self.x,
length: unsafe { clang_Comment_getNumChildren(self.x) },
index: 0,
}
}
pub fn get_tag_name(&self) -> String {
unsafe { cxstring_into_string(clang_HTMLTagComment_getTagName(self.x)) }
}
pub fn get_tag_attrs(&self) -> CommentAttributesIterator {
CommentAttributesIterator {
x: self.x,
length: unsafe { clang_HTMLStartTag_getNumAttrs(self.x) },
index: 0,
}
}
}
pub struct CommentChildrenIterator {
parent: CXComment,
length: c_uint,
index: c_uint,
}
impl Iterator for CommentChildrenIterator {
type Item = Comment;
fn next(&mut self) -> Option<Comment> {
if self.index < self.length {
let idx = self.index;
self.index += 1;
Some(Comment {
x: unsafe { clang_Comment_getChild(self.parent, idx) },
})
} else {
None
}
}
}
pub struct CommentAttribute {
pub name: String,
pub value: String,
}
pub struct CommentAttributesIterator {
x: CXComment,
length: c_uint,
index: c_uint,
}
impl Iterator for CommentAttributesIterator {
type Item = CommentAttribute;
fn next(&mut self) -> Option<CommentAttribute> {
if self.index < self.length {
let idx = self.index;
self.index += 1;
Some(CommentAttribute {
name: unsafe {
cxstring_into_string(
clang_HTMLStartTag_getAttrName(self.x, idx))
},
value: unsafe {
cxstring_into_string(
clang_HTMLStartTag_getAttrValue(self.x, idx))
},
})
} else {
None
}
}
}
#[derive(Debug, PartialEq, Eq, Hash)]
pub struct File {
x: CXFile,
}
impl File {
pub fn name(&self) -> Option<String> {
if self.x.is_null() {
return None;
}
Some(unsafe { cxstring_into_string(clang_getFileName(self.x)) })
}
}
pub fn cxstring_into_string(s: CXString) -> String {
if s.data.is_null() {
return String::new();
}
unsafe {
let c_str = CStr::from_ptr(clang_getCString(s).cast());
let ret = c_str.to_string_lossy().into_owned();
clang_disposeString(s);
ret
}
}
pub struct Index {
x: CXIndex,
}
impl Index {
pub fn new(pch: bool, diag: bool) -> Index {
unsafe {
Index {
x: clang_createIndex(pch as c_int, diag as c_int),
}
}
}
}
impl fmt::Debug for Index {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
write!(fmt, "Index {{ }}")
}
}
impl Drop for Index {
fn drop(&mut self) {
unsafe {
clang_disposeIndex(self.x);
}
}
}
#[derive(Debug)]
pub struct Token {
pub kind: CXTokenKind,
pub extent: CXSourceRange,
pub spelling: String,
}
pub struct TranslationUnit {
pub(crate) x: CXTranslationUnit,
}
impl fmt::Debug for TranslationUnit {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
write!(fmt, "TranslationUnit {{ }}")
}
}
impl TranslationUnit {
pub fn parse(ix: &Index,
file: &str,
cmd_args: &[String],
unsaved: &[UnsavedFile],
opts: CXTranslationUnit_Flags)
-> Option<TranslationUnit> {
let fname = CString::new(file).unwrap();
let _c_args: Vec<CString> =
cmd_args.iter().map(|s| CString::new(s.clone()).unwrap()).collect();
let c_args: Vec<*const c_char> =
_c_args.iter().map(|s| s.as_ptr()).collect();
let mut c_unsaved: Vec<CXUnsavedFile> =
unsaved.iter().map(|f| f.x).collect();
let mut tu = ptr::null_mut();
let err = unsafe {
clang_parseTranslationUnit2(ix.x,
fname.as_ptr(),
c_args.as_ptr(),
c_args.len() as c_int,
c_unsaved.as_mut_ptr(),
c_unsaved.len() as c_uint,
opts,
&mut tu)
};
if tu.is_null() {
panic!("Translation unit parse failure, error {err:?}");
} else {
Some(TranslationUnit {
x: tu,
})
}
}
pub fn diags(&self) -> Vec<Diagnostic> {
unsafe {
let num = clang_getNumDiagnostics(self.x) as usize;
let mut diags = vec![];
for i in 0..num {
diags.push(Diagnostic {
x: clang_getDiagnostic(self.x, i as c_uint),
});
}
diags
}
}
pub fn cursor(&self) -> Cursor {
unsafe {
Cursor {
x: clang_getTranslationUnitCursor(self.x),
}
}
}
pub fn is_null(&self) -> bool {
self.x.is_null()
}
pub fn tokens(&self, cursor: &Cursor) -> Option<Vec<Token>> {
let range = cursor.extent();
let mut tokens = vec![];
unsafe {
let mut token_ptr = ptr::null_mut();
let mut num_tokens: c_uint = 0;
clang_tokenize(self.x, range, &mut token_ptr, &mut num_tokens);
if token_ptr.is_null() {
return None;
}
let token_array = slice::from_raw_parts(token_ptr,
num_tokens as usize);
for &token in token_array {
let kind = clang_getTokenKind(token);
let extent = clang_getTokenExtent(self.x, token);
let spelling =
cxstring_into_string(clang_getTokenSpelling(self.x, token));
tokens.push(Token { kind, extent, spelling });
}
clang_disposeTokens(self.x, token_ptr, num_tokens);
}
Some(tokens)
}
}
impl Drop for TranslationUnit {
fn drop(&mut self) {
unsafe {
clang_disposeTranslationUnit(self.x);
}
}
}
pub struct Diagnostic {
x: CXDiagnostic,
}
impl Diagnostic {
pub fn format(&self) -> String {
unsafe {
let opts = clang_defaultDiagnosticDisplayOptions();
cxstring_into_string(clang_formatDiagnostic(self.x, opts))
}
}
pub fn severity(&self) -> CXDiagnosticSeverity {
unsafe { clang_getDiagnosticSeverity(self.x) }
}
}
impl Drop for Diagnostic {
fn drop(&mut self) {
unsafe {
clang_disposeDiagnostic(self.x);
}
}
}
pub struct UnsavedFile {
x: CXUnsavedFile,
name: CString,
contents: CString,
}
impl UnsavedFile {
pub fn new(name: &str, contents: &str) -> UnsavedFile {
let name = CString::new(name).unwrap();
let contents = CString::new(contents).unwrap();
let x = CXUnsavedFile {
Filename: name.as_ptr(),
Contents: contents.as_ptr(),
Length: contents.as_bytes().len() as c_ulong,
};
UnsavedFile {
x,
name,
contents,
}
}
}
pub fn kind_to_str(x: CXCursorKind) -> String {
unsafe { cxstring_into_string(clang_getCursorKindSpelling(x)) }
}
pub fn type_to_str(x: CXTypeKind) -> String {
unsafe { cxstring_into_string(clang_getTypeKindSpelling(x)) }
}
pub fn ast_dump(c: &Cursor, depth: isize) -> CXChildVisitResult {
fn print_indent<S: AsRef<str>>(depth: isize, s: S) {
for _ in 0..depth {
print!(" ");
}
println!("{}", s.as_ref());
}
fn print_cursor<S: AsRef<str>>(depth: isize, prefix: S, c: &Cursor) {
if c.is_builtin() {
return;
}
let prefix = prefix.as_ref();
print_indent(depth,
format!(" {}kind = {}", prefix, kind_to_str(c.kind())));
print_indent(depth,
format!(" {}spelling = \"{}\"", prefix, c.spelling()));
print_indent(depth, format!(" {}location = {}", prefix, c.location()));
print_indent(depth,
format!(" {}is-definition? {}",
prefix,
c.is_definition()));
print_indent(depth,
format!(" {}is-declaration? {}",
prefix,
c.is_declaration()));
print_indent(depth,
format!(" {}is-inlined-function? {}",
prefix,
c.is_inlined_function()));
let templ_kind = c.template_kind();
if templ_kind != CXCursor_NoDeclFound {
print_indent(depth,
format!(" {}template-kind = {}",
prefix,
kind_to_str(templ_kind)));
}
if let Some(usr) = c.usr() {
print_indent(depth, format!(" {prefix}usr = \"{usr}\""));
}
if let Ok(num) = c.num_args() {
print_indent(depth, format!(" {prefix}number-of-args = {num}"));
}
if let Some(num) = c.num_template_args() {
print_indent(depth,
format!(" {prefix}number-of-template-args = {num}"));
}
if let Some(width) = c.bit_width() {
print_indent(depth, format!(" {prefix}bit-width = {width}"));
}
if let Some(ty) = c.enum_type() {
print_indent(depth,
format!(" {}enum-type = {}",
prefix,
type_to_str(ty.kind())));
}
if let Some(val) = c.enum_val_signed() {
print_indent(depth, format!(" {prefix}enum-val = {val}"));
}
if let Some(ty) = c.typedef_type() {
print_indent(depth,
format!(" {}typedef-type = {}",
prefix,
type_to_str(ty.kind())));
}
if let Some(ty) = c.ret_type() {
print_indent(depth,
format!(" {}ret-type = {}",
prefix,
type_to_str(ty.kind())));
}
if let Some(refd) = c.referenced() {
if refd != *c {
println!();
print_cursor(depth,
String::from(prefix) + "referenced.",
&refd);
}
}
let canonical = c.canonical();
if let Some(canonical) = canonical {
if canonical != *c {
println!();
print_cursor(depth,
String::from(prefix) + "canonical.",
&canonical);
}
}
if let Some(specialized) = c.specialized() {
if specialized != *c {
println!();
print_cursor(depth,
String::from(prefix) + "specialized.",
&specialized);
}
}
if let Some(parent) = c.fallible_semantic_parent() {
println!();
print_cursor(depth,
String::from(prefix) + "semantic-parent.",
&parent);
}
}
fn print_type<S: AsRef<str>>(depth: isize, prefix: S, ty: &Type) {
let prefix = prefix.as_ref();
let kind = ty.kind();
print_indent(depth, format!(" {}kind = {}", prefix, type_to_str(kind)));
if kind == CXType_Invalid {
return;
}
print_indent(depth, format!(" {}cconv = {}", prefix, ty.call_conv()));
print_indent(depth,
format!(" {}spelling = \"{}\"", prefix, ty.spelling()));
let num_template_args =
unsafe { clang_Type_getNumTemplateArguments(ty.x) };
if num_template_args >= 0 {
print_indent(depth,
format!(" {prefix}number-of-template-args = {num_template_args}"));
}
if let Some(num) = ty.num_elements() {
print_indent(depth,
format!(" {prefix}number-of-elements = {num}"));
}
print_indent(depth,
format!(" {}is-variadic? {}", prefix, ty.is_variadic()));
let canonical = ty.canonical_type();
if canonical != *ty {
println!();
print_type(depth, String::from(prefix) + "canonical.", &canonical);
}
if let Some(pointee) = ty.pointee_type() {
if pointee != *ty {
println!();
print_type(depth, String::from(prefix) + "pointee.", &pointee);
}
}
if let Some(elem) = ty.elem_type() {
if elem != *ty {
println!();
print_type(depth, String::from(prefix) + "elements.", &elem);
}
}
if let Some(ret) = ty.ret_type() {
if ret != *ty {
println!();
print_type(depth, String::from(prefix) + "return.", &ret);
}
}
let named = ty.named();
if named != *ty && named.is_valid() {
println!();
print_type(depth, String::from(prefix) + "named.", &named);
}
}
print_indent(depth, "(");
print_cursor(depth, "", c);
println!();
let ty = c.cur_type();
print_type(depth, "type.", &ty);
let declaration = ty.declaration();
if let Some(declaration) = declaration {
if declaration != *c && declaration.kind() != CXCursor_NoDeclFound {
println!();
print_cursor(depth, "type.declaration.", &declaration);
}
}
let mut found_children = false;
c.visit(|s| {
if !found_children {
println!();
found_children = true;
}
ast_dump(&s, depth + 1)
});
print_indent(depth, ")");
CXChildVisit_Continue
}
#[derive(Debug)]
pub struct EvalResult {
x: CXEvalResult,
}
impl EvalResult {
pub fn new(cursor: Cursor) -> Option<Self> {
if false {
return None;
}
let mut found_cant_eval = false;
cursor.visit(|c| if c.kind() == CXCursor_TypeRef &&
c.cur_type().kind() == CXType_Unexposed {
found_cant_eval = true;
CXChildVisit_Break
} else {
CXChildVisit_Recurse
});
if found_cant_eval {
return None;
}
Some(EvalResult {
x: unsafe { clang_Cursor_Evaluate(cursor.x) },
})
}
fn kind(&self) -> CXEvalResultKind {
unsafe { clang_EvalResult_getKind(self.x) }
}
pub fn as_double(&self) -> Option<f64> {
match self.kind() {
CXEval_Float => {
Some(unsafe { clang_EvalResult_getAsDouble(self.x) } as f64)
}
_ => None,
}
}
pub fn as_int(&self) -> Option<i32> {
match self.kind() {
CXEval_Int => {
Some(unsafe { clang_EvalResult_getAsInt(self.x) } as i32)
}
_ => None,
}
}
pub fn as_literal_string(&self) -> Option<Vec<u8>> {
match self.kind() {
CXEval_StrLiteral => {
let ret = unsafe {
CStr::from_ptr(clang_EvalResult_getAsStr(self.x))
};
Some(ret.to_bytes().to_vec())
}
_ => None,
}
}
}
impl Drop for EvalResult {
fn drop(&mut self) {
unsafe { clang_EvalResult_dispose(self.x) };
}
}