clang-tags--上下文关联的c++调用搜索工具及其vim脚本

实际上,用vim的一个很大的缺点,是没有一个好用的,能理解上下文的c++调用关系搜索工具以及代码补全工具。在IDE里面,鼠标右键查找引用,查找调用,出来的结果是经过语法分析的,补全也是根据上下文头文件弄的。但是vim里面没有这个东西,毕竟vim是编辑器,他不知道编译信息。这个问题从我开始工作,就困扰着我,一直是我没法愉快使用vim写cpp的原因,毕竟菜。

后来,有了clang,以及clang complete等vim插件,解决了cpp代码补全的问题。YouCompleteMe加入了转到定义的功能,部分解决了跳转到定义的问题(但是不是特别准,有点问题)。

但是,调用关系搜索,一直是个解决不了的问题。

Clang静态分析

前一阵子突然想到要解决这个问题,就想到了clang。clang可以做代码静态分析,那怎么就不能做调用关系分析呢?实际上编译的时候肯定是需要调用关系的一个数据库的,不然没法愉快分析链接过程。那么clang有暴露接口嘛?有的,于是想到了用暴露出来的调用关系接口,做一个简单的调用关系数据库,然后通过查询数据库,查找对应的调用栈等信息。

Clang-Tags

我能想到,自然也有网友能想到,于是我在git上找到了clang-tags。这个项目已经把我的大部分想法都实现了,虽然已经有些年头了,但是总的来说还是能正常运行,只不过它需要strace用于生成编译参数,在mac上不是特别友好(mac不支持strace,并且该脚本兼容性也不是特别好),我fork了一个自己clang-tags,将strace移除,修复了一些crash问题,并且加入了虚函数调用关系的搜索。

安装

这个东西的架构不复杂,就是一个c++后台加一个python前端,需要使用libclang应该是他最大的坑了。但是一个想用vim写cpp的码农,这点问题应该是难不倒他的。

原repo给出的安装指引已经很好地解释了如何安装了。基本上requrie的库有,就能直接cmake过去。如果使用我的repo,可以跳过strace依赖,其他都是一样的。并且原repo主人使用的是emacs,所以还带有emacs的相关脚本,由于我不用emacs,有兴趣的伙伴自行尝试。

直接使用clang-tags

使用方法可以参考愿文章的quck start,里面有详细的文档,这里简单介绍下其中一种使用。

以我写的hello.cpp为例:

➜  clang cat hello.cpp

class Hello
{
public:
    int hello(int a, int b)
    {
        return a + b;
    }
    int hello(int a)
    {
        return hello(a, 1);
    }

    Hello();
    virtual ~Hello();
};

int main(int argc, char *argv[])
{
    Hello hi;
    hi.hello(10);
    return 0;
}

clang-tags是一个backend+frontend的架构,所以每次启动的时候要在工作目录使用start命令将它起起来:

➜  clang clang-tags start
Starting server...

有时候上一次非法推出,就会有crash,需要执行clean

➜  clang clang-tags start
ERROR: socket already exists!
➜  clang clang-tags clean
➜  clang clang-tags start
Starting server...

之后使用scan扫描一次本地目录,生成需要索引的文件编译配置compile_commands.json:

➜  clang ➜  clang clang-tags scan .
➜  clang ls
compile_commands.json  hello.cpp
➜  clang cat compile_commands.json
[
    {
        "directory": ".",
        "command": "gcc /home/johnzeng/learning/clang/hello.cpp ",
        "file": "/home/johnzeng/learning/clang/hello.cpp"
    }
]%

scan后可以跟编译参数。

之后使用load将文件载入backend的文件列表,在使用index进行第一次索引:

➜  clang clang-tags load
Server response:
  /home/johnzeng/learning/clang/hello.cpp
➜  clang clang-tags index
Server response:

-- Indexing project
/home/johnzeng/learning/clang/hello.cpp:
  parsing...	0.002643s.
  indexing...
  indexing...	0.002183s.
0.007176s.
现在载入好了,只需要调用对应的命令进行查询即可,比如int hello(int a)这个函数,hello的h在文件中的偏移(也就是一个byte一个byte从头开始数,这个是第几个byte)是96,那么可以调用下列命令查找他的定义
➜  clang clang-tags find-def /home/johnzeng/learning/clang/hello.cpp 95 -m
Server response:
-- hello(int a) { return hello(a, 1); } -- CXXMethod hello
   hello.cpp:9-12:9-5: hello
   USR: c:@S@Hello@F@hello#I#
   isVirtual: 0
可以看到他的函数签名是
c:@S@Hello@F@hello#I#

定义在hello.cpp的第9行,不是虚函数,是一个普通的cxx方法。

然后就可以根据这个签名找调用了:

➜  clang clang-tags grep 'c:@S@Hello@F@hello#I#'
Server response:
hello.cpp:9:    int hello(int a)
hello.cpp:21:    hi.hello(10);
hello.cpp:21:    hi.hello(10);
由于服务器没有做去重,第21行返回了两次。还是挺好用的。

Vim-Clang-Tags

同样,这样的好东西,也是有人已经弄了,在git找到 vim-clang-tags ,试用了下,嗯,还行。但是有些小问题,也fork了一个出来,自己改了一些功能来适应自己的需求,就有了我自己的vim-clang-tags。查找结果全部使用quickfix的窗口。效果如下:


这里查的是Reader::ReadMessage的结果。效果非常赞。


猜你喜欢

转载自blog.csdn.net/zerooffdate/article/details/79514616