I have an instance of CXCursor
of kind CXCursor_CXXMethod
. I want to find out if the function is const
or volatile
, for example:
class Foo {
public:
void bar() const;
void baz() volatile;
void qux() const volatile;
};
I could not find anything useful in the documentation of libclang. I tried clang_isConstQualifiedType
and clang_isVolatileQualifiedType
but these always seem to return 0
on C++ member function types.
I can think of two approaches:
Using the
libclang
lexerThe code which appears in this SO answer works for me; it uses the
libclang
tokenizer to break a method declaration apart, and then records any keywords outside of the method parentheses.It does not access the AST of the code, and as far as I can tell doesn't involve the parser at all. If you are sure the code you investigate is proper C++, I believe this approach is safe.
Disadvantages: This solution does not appear to take into account preprocessing directives, so the code has to be processed first (e.g., passed through
cpp
).Example code (the file to parse must be the first argument to your program, e.g.
./a.out bla.cpp
):Using
libclang
's Unified Symbol Resolution (USR)This approach involves using the parser itself, and extracting qualifier information from the AST.
Advantages: Seems to work for code with preprocessor directives, at least for simple cases.
Disadvantages: My solution parses the USR, which is undocumented, and might change in the future. Still, it's easy to write a unit-test to guard against that.
Take a look at
$(CLANG_SRC)/tools/libclang/CIndexUSRs.cpp
, it contains the code that generates a USR, and therefore contains the information required to parse the USR string. Specifically, lines 523-529 (in LLVM 3.1's source downloaded from www.llvm.org) for the qualifier part.Add the following function somewhere:
and in
main()
,Running on
Foo::qux()
fromproduces the expected result of
Caveat: you might have noticed that
libclang
's source suggets my code should beisVolatile = x & 0x2
and not0x4
, so it might be the case you should replace 0x4 with 0x2. It's possible my implementation (OS X) has them replaced.