最近在学习内核代码,由于经常用到Cedet来帮助浏览代码,所以整理了一下Cedet的使用。 这里的Cedet是Emacs 23.2中自带的,版本和sourceforge上的Cedet相同,但是其中的内 容(主要是函数名之类)有不少的变化。这里以 A Gentle introduction to Cedet 为基础,写的这个文章,结构与之类似,内容上有些地方是翻译,但大多是我根据自己的 配置整理出来的。
1 什么是Cedet
CEDET 是 Collection of Emacs Development Environment Tools的缩写, 意为"Emacs开发环境工具集",其主要目的是在Emacs中建立一个高级的开发环境。 它主要包括下列组件:
Semantic -— 多种编程语言的语法分析的基础组件。
SemanticDB-—包含在Semantic中的一个数据库,用于保存代码的语法、接口等等信息。
Senator -— 通过Semantic提取出来的信息构成的代码文件中的navegation。
Srecode -— 代码生成组件。
EDE -— 提供工程管理相关功能;
Speedbar -— 用于显示当前Buffer的侧边栏。
Eieio is a library, implementating CLOS-like (Common Lisp Object System) infrastructure for Emacs Lisp;
Cogre is a library for drawing of UML-like diagrams in Emacs buffer, with basic integration with Semantic.
其中,最后面这两个我没怎么用过。
2 Cedet的安装和启用
Emacs 23.2 中已经自带了Cedet,所以无需再单独安装,直接启用即可:
(require 'cedet)
如果你想使用Cedet的工程管理功能,可以启用EDE Mode ----
(global-ede-mode t)
3 Cedet 的定制
3.1 基本Helpter的定制
Emacs 23.2中自带的Cedet中,去掉了原来的诸如 semantic-load-enable-minimum-features, semantic-load-enable-code-helpers 等等的命令,而是通过定制子模式(semantic-default-submodes)来确 定使用哪些辅助功能,最后使用semantic-mode 来启用这些功能。
以下是我的设置:
;;;; Helper tools.
(custom-set-variables
'(semantic-default-submodes (quote (global-semantic-decoration-mode global-semantic-idle-completions-mode
global-semantic-idle-scheduler-mode global-semanticdb-minor-mode
global-semantic-idle-summary-mode global-semantic-mru-bookmark-mode)))
'(semantic-idle-scheduler-idle-time 3))
(semantic-mode)
3.2 Semantic/ia 的配置
semantic/ia 提供了机遇semantic的自动补齐,Tag信息显示等功能, Semantic/ia 可通过下面的代码来启用和优化
;; smart complitions
(require 'semantic/ia)
(setq-mode-local c-mode semanticdb-find-default-throttle
'(project unloaded system recursive))
(setq-mode-local c++-mode semanticdb-find-default-throttle
'(project unloaded system recursive))
3.3 头文件的设置
C/C++ 的开发中,和头文件要打很多交道。 头文件分为两种:系统头文件和用户自定义头文件。
3.3.1 系统头文件
如果我们使用的编译器为gcc,那么可以使用 semantic/gcc 来自动加载系统的头文件路径。
当然,在此基础上,我们也可以使用 semantic-add-system-include 来手动显示地添加某些路径。
3.3.2 用户头文件
用户自己定义的头文件,一般所在的位置可以用与当前路径的相对关系来表示出来, 然后在手动添加。
系统头文件和用户头文件的设置代码如下:
;;;; Include settings
(require 'semantic/bovine/gcc)
(require 'semantic/bovine/c)
(defconst cedet-user-include-dirs
(list ".." "../include" "../inc" "../common" "../public" "."
"../.." "../../include" "../../inc" "../../common" "../../public"))
(setq cedet-sys-include-dirs (list
"/usr/include"
"/usr/include/bits"
"/usr/include/glib-2.0"
"/usr/include/gnu"
"/usr/include/gtk-2.0"
"/usr/include/gtk-2.0/gdk-pixbuf"
"/usr/include/gtk-2.0/gtk"
"/usr/local/include"
"/usr/local/include"))
(let ((include-dirs cedet-user-include-dirs))
(setq include-dirs (append include-dirs cedet-sys-include-dirs))
(mapc (lambda (dir)
(semantic-add-system-include dir 'c++-mode)
(semantic-add-system-include dir 'c-mode))
include-dirs))
(setq semantic-c-dependency-system-include-path "/usr/include/")
3.4 IMenu的集成
Semantic 可以通过imenu集成到Emacs菜单中, 从而通过菜单来显示和访问函数、变量以及其他Tag的列表。
Emacs 23.2 中,可以通过下面的代码来实现这个功能:
;;;; TAGS Menu
(defun my-semantic-hook ()
(imenu-add-to-menubar "TAGS"))
(add-hook 'semantic-init-hooks 'my-semantic-hook)
3.5 Semanticdb 的定制
如果按照前面的代码中来定制HelperTool,sematicdb会自动启用。 这里我们首先需要自定义一下semanticdb的存放路径,例如我的:
;;;; Semantic DataBase存储位置
(setq semanticdb-default-save-directory
(expand-file-name "~/.emacs.d/semanticdb"))
semanticdb 可以使用其他工具产生的Tag,例如GNU Global产生的Tags, 可以按照下面的代码来启用:
;; 使用 gnu global 的TAGS。
(require 'semantic/db-global)
(semanticdb-enable-gnu-global-databases 'c-mode)
(semanticdb-enable-gnu-global-databases 'c++-mode)
3.6 管理C/C++的工程
建议使用EDE来管理C/C++工程,下面的代码可用来定义一个工程:
(ede-cpp-root-project "Kernel"
:name "Kernel Project"
:file "~/Work/projects/kernel/linux-2.6.34/Makefile"
:include-path '("/"
"/include"
)
:system-include-path '("/usr/include")
:spp-table '(("__FLT_MIN__" . "1.17549435e-38F")
("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_2" . "1")
("__SIZEOF_POINTER__" . "4")
("__GCC_HAVE_DWARF2_CFI_ASM" . "1")
("__FLT_MIN_EXP__" . "(-125)")
("__SIZEOF_PTRDIFF_T__" . "4")
("__FLT_HAS_INFINITY__" . "1")
("_FORTIFY_SOURCE" . "2")
("__ELF__" . "1")
("__DBL_MANT_DIG__" . "53")
("__DBL_EPSILON__" . "2.2204460492503131e-16")
("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4" . "1")
("__unix" . "1")
("__SIZEOF_WINT_T__" . "4")
("__LDBL_EPSILON__" . "1.08420217248550443401e-19L")
("__INT_MAX__" . "2147483647")
("__FLT_MAX_EXP__" . "128")
("__FLT_MAX__" . "3.40282347e+38F")
("__SIZEOF_DOUBLE__" . "8")
("__LDBL_DENORM_MIN__" . "3.64519953188247460253e-4951L")
("__DEC128_MANT_DIG__" . "34")
("__SHRT_MAX__" . "32767")
("__SIZEOF_LONG__" . "4")
("__GNUC__" . "4")
("__DEC32_MIN_EXP__" . "(-94)")
("__cplusplus" . "1")
("__tune_i486__" . "1")
("__DBL_HAS_DENORM__" . "1")
("__DEC128_SUBNORMAL_MIN__" . "0.000000000000000000000000000000001E-6143DL")
("__DEC64_MIN_EXP__" . "(-382)")
("__FLT_DENORM_MIN__" . "1.40129846e-45F")
("__PTRDIFF_TYPE__" . "int")
("__THROW" . "")
("__CHAR_BIT__" . "8")
("__linux__" . "1")
("__STDC_HOSTED__" . "1")
("__DEC32_MIN__" . "1E-95DF")
("__VERSION__" . "\"4.4.3\"")
("__DEC128_MIN_EXP__" . "(-6142)")
("__i386__" . "1")
("__DEC64_MIN__" . "1E-383DD")
("__DBL_MAX_10_EXP__" . "308")
("__DBL_MIN_10_EXP__" . "(-307)")
("__DEC64_MAX_EXP__" . "385")
("__FLT_EPSILON__" . "1.19209290e-7F")
("__DEPRECATED" . "1")
("__GXX_ABI_VERSION" . "1002")
("__DEC128_MIN__" . "1E-6143DL")
("__i486__" . "1")
("unix" . "1")
("__DEC64_MANT_DIG__" . "16")
("__DEC32_MAX_EXP__" . "97")
("linux" . "1")
("__SIZEOF_FLOAT__" . "4")
("__i486" . "1")
("__GNUC_PATCHLEVEL__" . "3")
("__DBL_DIG__" . "15")
("__SIZEOF_INT__" . "4")
("__DEC32_MAX__" . "9.999999E96DF")
("__i386" . "1")
("__DEC64_MAX__" . "9.999999999999999E384DD")
("__SIZEOF_SHORT__" . "2")
("__DEC128_MAX_EXP__" . "6145")
("__FINITE_MATH_ONLY__" . "0")
("__linux" . "1")
("__SCHAR_MAX__" . "127")
("__gnu_linux__" . "1")
("__LDBL_HAS_QUIET_NAN__" . "1")
("__BIGGEST_ALIGNMENT__" . "16")
("__DBL_HAS_INFINITY__" . "1")
("__DEC128_MAX__" . "9.999999999999999999999999999999999E6144DL")
("__DECIMAL_BID_FORMAT__" . "1")
("__NO_INLINE__" . "1")
("__FLT_MANT_DIG__" . "24")
("__FLT_HAS_QUIET_NAN__" . "1")
("__INTMAX_MAX__" . "9223372036854775807LL")
("__STDC__" . "1")
("__const" . "const")
("__restrict" . nil)
("__FLT_EVAL_METHOD__" . "2")
("__unix__" . "1")
("__DEC32_EPSILON__" . "1E-6DF")
("__GXX_RTTI" . "1")
("__SIZEOF_LONG_DOUBLE__" . "12")
("__LDBL_HAS_DENORM__" . "1")
("__LONG_LONG_MAX__" . "9223372036854775807LL")
("__DBL_MIN__" . "2.2250738585072014e-308")
("__DEC64_EPSILON__" . "1E-15DD")
("__LDBL_DIG__" . "18")
("__DBL_MIN_EXP__" . "(-1021)")
("__WCHAR_MAX__" . "2147483647")
("__DBL_DENORM_MIN__" . "4.9406564584124654e-324")
("__DECIMAL_DIG__" . "21")
("__DEC128_EPSILON__" . "1E-33DL")
("__LDBL_MANT_DIG__" . "64")
("__SIZEOF_WCHAR_T__" . "4")
("i386" . "1")
("__DEC64_SUBNORMAL_MIN__" . "0.000000000000001E-383DD")
("__LDBL_MIN__" . "3.36210314311209350626e-4932L")
("__FLT_HAS_DENORM__" . "1")
("__DBL_MAX__" . "1.7976931348623157e+308")
("__FLT_RADIX__" . "2")
("__DEC32_SUBNORMAL_MIN__" . "0.000001E-95DF")
("__LONG_MAX__" . "2147483647L")
("__EXCEPTIONS" . "1")
("__LDBL_MIN_EXP__" . "(-16381)")
("__DEC_EVAL_METHOD__" . "2")
("__DEC32_MANT_DIG__" . "7")
("__SIZEOF_LONG_LONG__" . "8")
("__GXX_WEAK__" . "1")
("__DBL_MAX_EXP__" . "1024")
("__SIZEOF_SIZE_T__" . "4")
("__FLT_MAX_10_EXP__" . "38")
("__DBL_HAS_QUIET_NAN__" . "1")
("__FLT_DIG__" . "6")
("__LDBL_MAX_10_EXP__" . "4932")
("__LDBL_MIN_10_EXP__" . "(-4931)")
("__GCC_HAVE_SYNC_COMPARE_AND_SWAP_1" . "1")
("__LDBL_MAX__" . "1.18973149535723176502e+4932L")
("__LDBL_MAX_EXP__" . "16384")
("__LDBL_HAS_INFINITY__" . "1")
("__GNUG__" . "4")
("__FLT_MIN_10_EXP__" . "(-37)")
("__GNUC_MINOR__" . "4")
("__GNUC_GNU_INLINE__" . "1")
("_GNU_SOURCE" . "1")
))
这里有几个变量值得注意:
file:
file可以为该程根目录下面的任意文件,该文件不用来解析,而只是这个工程的一个标志。
include-path:
该变量是一个相对路径,指出了自定义的include目录。 其中的"/"并不表示系统的根目录,而表示该工程的根目录。
system-include-path:
这是一个绝对路径,该路径指明了系统的Include目录。
spp-table:
spp-table 给出了预处理时的使用的宏,通常是在Makefile里使用-DXXX定义的宏。
这些宏可以在载入工程后,通过命令semantic-lex-spp-describe来获得。
4 Cedet的使用
4.1 命名补齐
这里的补齐包括函数名称,变量名等等,是很常用的一个功能。 个人以为最实用的一个补齐是 semantic-ia-complete-symbol, 他可以通过快捷键"C-c, /" 来调用。为了使用方便并和其他Package统一, 我将该函数添加到了hippie-expand中, 并将hippie-expand包进了自定义的函数indent-or-complete (从别人的配置文件中找到的)中,并将这个函数绑定到了Tab上。 这样,大多数情况下,通过Tab即可实现补齐或者对齐。 如果偶尔Tab不成功,再使用"M-/"或者"C-c, /"来修正一下。
这段配置的Lisp代码如下:
;;;; 缩进或者补齐
;;; hippie-try-expand settings
(setq hippie-expand-try-functions-list
'(
yas/hippie-try-expand
semantic-ia-complete-symbol
try-expand-dabbrev
try-expand-dabbrev-visible
try-expand-dabbrev-all-buffers
try-expand-dabbrev-from-kill
try-complete-file-name-partially
try-complete-file-name
try-expand-all-abbrevs))
(defun indent-or-complete ()
"Complete if point is at end of a word, otherwise indent line."
(interactive)
(if (looking-at "\\>")
(hippie-expand nil)
(indent-for-tab-command)
))
(defun yyc/indent-key-setup ()
"Set tab as key for indent-or-complete"
(local-set-key [(tab)] 'indent-or-complete)
)
此外,对于C和C++的struct/class结构,函数semantic-complete-self-insert 可以插入类或结构中的成员变量,将至绑定到"."或者">",会加速代码编写的效率:
;;;; C-mode-hooks .
(defun yyc/c-mode-keys ()
"description"
;; Semantic functions.
(semantic-default-c-setup)
(local-set-key "\C-c?" 'semantic-ia-complete-symbol-menu)
(local-set-key "\C-cb" 'semantic-mrub-switch-tags)
(local-set-key "\C-cR" 'semantic-symref)
(local-set-key "\C-cj" 'semantic-ia-fast-jump)
(local-set-key "\C-cp" 'semantic-ia-show-summary)
(local-set-key "\C-cl" 'semantic-ia-show-doc)
(local-set-key "\C-cr" 'semantic-symref-symbol)
(local-set-key "\C-c/" 'semantic-ia-complete-symbol)
(local-set-key [(control return)] 'semantic-ia-complete-symbol)
(local-set-key "." 'semantic-complete-self-insert)
(local-set-key ">" 'semantic-complete-self-insert)
;; Indent or complete
(local-set-key [(tab)] 'indent-or-complete)
)
(add-hook 'c-mode-common-hook 'yyc/c-mode-keys)
4.2 获取Tag信息
semantic-ia-show-doc, semantic-ia-show-summary, semantic-ia-describe-class 这三个函数可以用来获取Tag信息,显示出代码的注释(包括Doxgen的注释), 对于阅读代码有很大帮助。
这些函数可以按照自己的习惯去绑定。其中前面两个的绑定过程在前面的 yyc/c-mode-keys中可以找到。
4.3 代码中的跳转
阅读代码时候,在不同的Tag中跳转是一个很有用的功能。 Cedet提供了这个功能,但是我手头上的Emacs 23.2 中自带的Cedet在Tag 跳转上有个问题————可以用 semantic-ia-fast-jump 跳转到Tag定义, 但是每次跳转后,却不能跳回来。
按照本文3.1节的方法设置Helper以后,在配置文件中添加下面的代码可以解决这一问题:
(defadvice push-mark (around semantic-mru-bookmark activate)
"Push a mark at LOCATION with NOMSG and ACTIVATE passed to `push-mark'.
If `semantic-mru-bookmark-mode' is active, also push a tag onto
the mru bookmark stack."
(semantic-mrub-push semantic-mru-bookmark-ring
(point)
'mark)
ad-do-it)
4.4 查找函数调用
前面一节说的是在当前函数中,跳到Tag定义;而这里, 说的是查看当前光标下面的函数被哪些函数调用了。 Cedet中的 semantic-symref 实现了这一功能。 可以将该函数绑定到自己喜欢的快捷键上,如4.1中所示。
对于代码的浏览这个部分,我大部分时候使用的是global这个工具, global配置Xgtags来用,很方便。
4.5 Srecode的使用
Cedet提供了srecode,用于自动生成代码。但,个人感觉这个功能不怎么好用, 或者说是我不会用吧。
Emacs 23.2中自带的Cedet,仅提供了srecode的功能,但却没有将需要的template 一起和emacs一同发布出来。
对此,我的做法是修改了srecode-map-load-path,添加了 "~/.emacs.d/templates/srecode",
;;;; Custom template for srecode
(setq srecode-map-load-path
(list (srecode-map-base-template-dir)
(expand-file-name "~/.emacs.d/templates/srecode")
))
然后从cedet官网上 下载源码包,并把template解压到 "~/.emacs.d/templates/srecode"中。
然后将可以使用srecode-insert之类的函数来插入代码模版了。