vim+cscope+ctags 是Linux看代码利器,在网上搜vim教程一搜一大堆,很多讲的都不错,可惜自己都没学会。好早之前,我的老科长波哥曾经教过我一次,ctags我算是 学会大概,cscope只能算是入门级水平,比如查看某个函数调用,还得用:cs find c function这种比较慢的方法。我自己比较懒,花在熟练掌握工具的时间太少了,这么看来,自己还不是个好程序员。

    cscope的存在让vim有了媲美Source Insight的能力,那么对大工程文件如何使用cscope呢?

    最近在看PostgreSQL的源码,我们知道,PostgreSQL源码属于一个大型工程,src下面好多目录,不同目录下好多文件。好多教程讲述建立csope数据库的方法,都是下面的方法,以我的PostgreSQL为例)   


  1. root@manu:/usr/local/src/pgsrc/postgresql-9.2.3/src# cscope -Rbq
  2. root@manu:/usr/local/src/pgsrc/postgresql-9.2.3/src# ll
  3. .....
  4. -rw-r--r--  1 root root  2138112  3月 30 23:35 cscope.in.out
    -rw-r--r--  1 root root 19451512  3月 30 23:35 cscope.out
    -rw-r--r--  1 root root 14147844  3月 30 23:35 cscope.po.out
  5. .....

   这种方法为带来问题(也可能是我功底比较浅),会带来什么问题呢?

   讲这个之前,我们先学习下cscope的应用指令:在vim命令行下执行:cs help 


使用vim看代码:cscope_postgresql


    cs show是显示当前可用的数据库,cs add是添加一个数据库。我们看下


  1. root@manu:/usr/local/src/pgsrc/postgresql-9.2.3/src# cd backend/storage/file/
  2. root@manu:/usr/local/src/pgsrc/postgresql-9.2.3/src/backend/storage/file# vi fd.c


    我们执行添加cscope 测试数据库的操作,在vim的命令行模式执行


  1. :cs add /usr/local/src/pgsrc/postgresql-9.2.3/src/cscope.out
  2. :cs show


    可以看到如下db连接:


使用vim看代码:cscope_数据库_02

    目前还正常,当我们使用:cs find 去查找函数调用的关系的时候:比如我读到closeAllVfds 函数实现的时候,我关心谁调用了fd.c中的closeAllVfds这个function。:cs find c 是干这个活的,请看上面的help。


  使用vim看代码:cscope_postgresql_03 


     结果是没找到。


 使用vim看代码:cscope_postgresql_04     



   我们不是已经添加了数据库了吗,为啥没找到了呢。原因就是我们生成cscope.out的时候,路径都是相对的路径。对于大工程我们怎么用cscope做到文件间的函数跳转呢?         

下面介绍官方文档推荐的方法:​​Using Cscope on large projects (example: the Linux kernel)​​,英文好的筒子可以直接去看原文,不用听我罗嗦。

   1 建立cscope目录,我在自己的家目录下专门存放cscope数据库的目录:




  1. root@manu:~/cscope# ll
    总用量 16
    drwxr-xr-x  2 root root  4096  3月 31 00:49 ./
    drwxr-xr-x 72 manu manu 12288  3月 31 00:48 ../


    2 创建一个脚本来做建立cscope数据库的事情



  1. root@manu:~/cscope# cat /usr/bin/makecscope.sh
  2. #!/bin/sh
  3. usage()
  4. {
  5. echo "usage : makecscope src_path project_name"
  6. echo "I will create cscope db in ~/cscope/project_name"
  7. }
  8. if [ $# -ne 2 ]
  9. then
  10. usage
  11. exit
  12. fi
  13. SRC_PATH=$1
  14. CSCOPE_PATH=/home/manu/cscope/$2
  15. mkdir -p $CSCOPE_PATH
  16. cd $CSCOPE_PATH
  17. find $SRC_PATH -name "*.h" -o -name "*.c" -o -name "Makefile" -o -name "makefile" > cscope.files
  18. cscope -bkq -i ./cscope.files

我们建立了一个sh脚本来干这个事情,sh脚本会接受两个参数

  • 源码路径:脚本会搜索源代码路径下的所有.c .h Makefile makefile文件
  • 项目的名称:脚本会在~/cscope目录下建立以第二个参数为名的目录,同时将生成的cscope数据库放入  该目录下


     我们将这个makecscope.sh脚本放入/usr/bin/目录下,就可以在任意路径下执行这个脚本了



    3 为任意项目创建cscope数据库,我们还是以PostgreSQL为例:



  1. root@manu:~/cscope# makecscope.sh /usr/local/src/pgsrc/postgresql-9.2.3/src/ postgres

我为路径为 /usr/local/src/pgsrc/postgresql-9.2.3/src/下的所有C文件 头文件和Makefile文件创建了cscope数据库,并将数据库放在了~/cscope/postgres路径下。


  1. root@manu:~/cscope# cd postgres/
  2. root@manu:~/cscope/postgres# ll
  3. 总用量 31532
  4. drwxr-xr-x 2 root root     4096 3月 31 00:54 ./
  5. drwxr-xr-x 3 root root     4096 3月 31 00:54 ../
  6. -rw-r--r-- 1 root root   117338 3月 31 00:54 cscope.files
  7. -rw-r--r-- 1 root root  1499136 3月 31 00:54 cscope.in.out
  8. -rw-r--r-- 1 root root 17656981 3月 31 00:54 cscope.out
  9. -rw-r--r-- 1 root root 13006272 3月 31 00:54 cscope.po.out
  10. root@manu:~/cscope/postgres#


 创建好了数据库,我们看工程代码的时候,就可以添加这个cscope数据库了,然后就可以查看函数调用关系了。





=====================================================================================

    这个问题解决了,我们还有一个问题是敲 :cs find c closeAllVfds这太不人性了,函数名很长的情况下,要人命。快捷键映射就很重要了。看下我的.vimrc里面的快捷键映射。(关心.vimrc 的筒子可以CU的草根老师博客下载那个vimrc的安装包,文章路径在此:​​配置自己的vim​​)



  1. nmap <leader>sa :cs add cscope.out<cr>
  2. nmap <leader>ss :cs find s <C-R>=expand("")<cr><cr>
  3. nmap <leader>sg :cs find g <C-R>=expand("")<cr><cr>
  4. nmap <leader>sc :cs find c <C-R>=expand("")<cr><cr>
  5. nmap <leader>st :cs find t <C-R>=expand("")<cr><cr>
  6. nmap <leader>se :cs find e <C-R>=expand("")<cr><cr>
  7. nmap <leader>sf :cs find f <C-R>=expand("")<cr><cr>
  8. nmap <leader>si :cs find i <C-R>=expand("")<cr><cr>
  9. nmap <leader>sd :cs find d <C-R>=expand("")<cr><cr>



    nmap表示在vim的普通模式下映射,当然相对与编辑模式和可视模式而言的,我们不多说。

    =expand("cword")总体是为了得到:光标下的变量或函数。cword 表示:cursor word, 类似的还有:cfile表示光标所在处的文件名。

   这个键在我的vim中对应的是逗号, 所以,上面的意思就 只要光标在我们关心的变量或者函数下,命令行模式敲,sc就相当与执行:cs find c 光标处函数名,注意,只需,sc三个键,不要敲冒号: 其他的键的映射也是类似的。    


    有了这个快捷键,我们使用cscope的效率就提升了,加上ctrl+],ctrl+o,ctrl+t 我们看项目代码基本就比较舒服了。

    有网友bottles在我的博文​​vim格式化C代码​​中问到:


  1. 这个插件在不同文件当中都能跳转到定义吗?还有某些文件我想直接输入文件名不输入路径就open这个文件,这个功能用您说的插件能实现吗?我之前就是因为不能跳转到定义才用source insight里去的。

    对于这个问题,答案是肯定的,我们只需要执行:cs find f filename,就可以跳转到名字为filename的文件中去,注意不需要是全路径名。比如我可以从/usr/local/src/pgsrc/postgresql-9.2.3/src/backend/storage/file/fd.c文件中,在命令行模式下敲


  1. :cs find f bgwriter.c

就会跳转到/usr/local/src/pgsrc/postgresql-9.2.3/src/backend/postmaster/bgwriter.c。尽管他们不在同一路径下。


    对于头文件的跳转,比如fd.c中有如下头文件:


  1. #include "miscadmin.h"

我们只要将光标置于miscadmin.h处,执行快捷键,sf三个键,就能跳转到miscadmin.h头文件处。只需要三个键,跳转的特别快。

   我的博文vim格式化C代码中,当时没有考虑到makefile,因为makefile的tab键不能展开为4个空格,这次一并在博文中修复了,对makefile做了特殊处理。






参考文献:


1 ​​Vim+cscope+ctags+tags阅读源代码
​2 ​​CSCOPE使用中问题小解​


3 ​​Using Cscope on large projects (example: the Linux kernel)​