如何使用libclang检索函数调用参数值

时间:2014-08-27 07:10:11

标签: python libclang

是否可以检索 clang.cindex.CursorKind.CALL_EXPR 游标的参数值?

当我使用编译器(clang ++ -ast-dump source.cpp)转储AST时,我得到有关函数调用(调用表达式)及其参数的信息。但是我无法使用python的绑定来复制它(使用libclang的解析器检索AST)。

这是我正在使用的源代码:

#include <iostream>
#include <GL/glut.h>
#include <EGL/egl.h>

#define MULTILINE(...) #__VA_ARGS__

void renderFunction()
{
    glClearColor(0.0, 0.0, 0.0, 0.0);
    glClear(GL_COLOR_BUFFER_BIT);
    glColor3f(1.0, 0.0, 1.0);
    glOrtho(-1.0, 1.0, -1.0, 1.0, -1.0, 1.0);
    glBegin(GL_QUADS);
        glVertex2f(-0.5, -0.5);
        glVertex2f(-0.5, 0.5);
        glVertex2f(0.5, 0.5);
        glVertex2f(0.5, -0.5);
    glEnd();
    glFlush();
}

int main(int argc, char *argv[])
{          
    glutInit(&argc, argv);       
    glutInitDisplayMode(GLUT_SINGLE);
    glutInitWindowSize(500,500);
    glutInitWindowPosition(100,100);
    glutCreateWindow("OpenGL - First window demo");
    glutDisplayFunc(renderFunction);
    glutMainLoop();    

    return 0;
}

这是AST转储的一部分:

|   |-CallExpr 0x430b540 <line:10:5, col:32> 'void'
|   | |-ImplicitCastExpr 0x430b528 <col:5> 'void (*)(GLbitfield)' <FunctionToPointerDecay>
|   | | `-DeclRefExpr 0x430b4d0 <col:5> 'void (GLbitfield)' lvalue Function 0x3d3b060 'glClear' 'void (GLbitfield)'
|   | `-ImplicitCastExpr 0x430b570 </usr/include/GL/gl.h:691:31> 'GLbitfield':'unsigned int' <IntegralCast>
|   |   `-IntegerLiteral 0x430b4b0 <col:31> 'int' 16384

我想通过评估调用表达式游标来检索最后一行中的 IntegerLiteral 值部分。

1 个答案:

答案 0 :(得分:3)

你可以从令牌列表中获取这些信息,对于IntegerLiteral,第一个令牌将是你的号码(不是那么整洁,但总比没有好!)。

示例cpp程序:

#define FOO 6

void foo(int x) {}

int main()
{
    foo(FOO);
    return 0;
}

用于解析它并打印出IntegerLiteral值的示例python代码(使用lib clang):

import clang.cindex
import sys

path = '/your/path/to/libclang.so'
clang.cindex.Config.set_library_file(path)

def get_ts(source_path):
    index = clang.cindex.Index.create()
    return index.parse(source_path)

def print_node(node):
    if node.kind == clang.cindex.CursorKind.INTEGER_LITERAL:
        print node.type.kind, node.get_tokens().next().spelling
    map(print_node, node.get_children())

ts = get_ts('test.cpp')
map(print_node, ts.cursor.get_children())

输出:

TypeKind.INT 6
TypeKind.INT 0