最近在学习内核代码,由于经常用到Cedet来帮助浏览代码,所以整理了一下Cedet的使用。 这里的Cedet是Emacs 23.2中自带的,版本和sourceforge上的Cedet相同,但是其中的内 容(主要是函数名之类)有不少的变化。这里以 A Gentle introduction to Cedet 为基础,写的这个文章,结构与之类似,内容上有些地方是翻译,但大多是我根据自己的 配置整理出来的。
本文来源:点击查看
CEDET 是 Collection of Emacs Development Environment Tools的缩写, 意为"Emacs开发环境工具集",其主要目的是在Emacs中建立一个高级的开发环境。 它主要包括下列组件:
其中,最后面这两个我没怎么用过。
Emacs 23.2 中已经自带了Cedet,所以无需再单独安装,直接启用即可:
(require 'cedet) |
如果你想使用Cedet的工程管理功能,可以启用EDE Mode ----
(global-ede-mode t) |
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) |
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)) |
C/C++ 的开发中,和头文件要打很多交道。 头文件分为两种:系统头文件和用户自定义头文件。
如果我们使用的编译器为gcc,那么可以使用 semantic/gcc 来自动加载系统的头文件路径。
当然,在此基础上,我们也可以使用 semantic-add-system-include 来手动显示地添加某些路径。
用户自己定义的头文件,一般所在的位置可以用与当前路径的相对关系来表示出来, 然后在手动添加。
系统头文件和用户头文件的设置代码如下:
;;;; 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/") |
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) |
如果按照前面的代码中来定制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) |
建议使用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") ) |
这里有几个变量值得注意:
这里的补齐包括函数名称,变量名等等,是很常用的一个功能。 个人以为最实用的一个补齐是 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) |
semantic-ia-show-doc, semantic-ia-show-summary, semantic-ia-describe-class 这三个函数可以用来获取Tag信息,显示出代码的注释(包括Doxgen的注释), 对于阅读代码有很大帮助。
这些函数可以按照自己的习惯去绑定。其中前面两个的绑定过程在前面的 yyc/c-mode-keys中可以找到。
阅读代码时候,在不同的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) |
前面一节说的是在当前函数中,跳到Tag定义;而这里, 说的是查看当前光标下面的函数被哪些函数调用了。 Cedet中的 semantic-symref 实现了这一功能。 可以将该函数绑定到自己喜欢的快捷键上,如4.1中所示。
对于代码的浏览这个部分,我大部分时候使用的是global这个工具, global配置Xgtags来用,很方便。
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之类的函数来插入代码模版了。