1 ;;; ede/emacs.el --- Special project for Emacs
3 ;; Copyright (C) 2008-2012 Free Software Foundation, Inc.
5 ;; Author: Eric M. Ludlam <eric@siege-engine.com>
7 ;; This file is part of GNU Emacs.
9 ;; GNU Emacs is free software: you can redistribute it and/or modify
10 ;; it under the terms of the GNU General Public License as published by
11 ;; the Free Software Foundation, either version 3 of the License, or
12 ;; (at your option) any later version.
14 ;; GNU Emacs is distributed in the hope that it will be useful,
15 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
16 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 ;; GNU General Public License for more details.
19 ;; You should have received a copy of the GNU General Public License
20 ;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
24 ;; Provide a special project type just for Emacs, cause Emacs is special.
26 ;; Identifies an Emacs project automatically.
27 ;; Speedy ede-expand-filename based on extension.
28 ;; Pre-populates the preprocessor map from lisp.h
31 ;; * Add "build" options.
32 ;; * Add texinfo lookup options.
36 (declare-function semanticdb-file-table-object
"semantic/db")
37 (declare-function semanticdb-needs-refresh-p
"semantic/db")
38 (declare-function semanticdb-refresh-table
"semantic/db")
41 (defvar ede-emacs-project-list nil
42 "List of projects created by option `ede-emacs-project'.")
44 (defun ede-emacs-file-existing (dir)
45 "Find a Emacs project in the list of Emacs projects.
46 DIR is the directory to search from."
47 (let ((projs ede-emacs-project-list
)
49 (while (and projs
(not ans
))
50 (let ((root (ede-project-root-directory (car projs
))))
51 (when (string-match (concat "^" (regexp-quote root
))
52 (file-name-as-directory dir
))
53 (setq ans
(car projs
))))
54 (setq projs
(cdr projs
)))
58 (defun ede-emacs-project-root (&optional dir
)
59 "Get the root directory for DIR."
60 (when (not dir
) (setq dir default-directory
))
61 (let ((case-fold-search t
)
62 (proj (ede-emacs-file-existing dir
)))
64 (ede-up-directory (file-name-directory
66 ;; No pre-existing project. Let's take a wild-guess if we have
67 ;; an Emacs project here.
68 (when (string-match "emacs[^/]*" dir
)
69 (let ((base (substring dir
0 (match-end 0))))
70 (when (file-exists-p (expand-file-name "src/emacs.c" base
))
73 (defun ede-emacs-version (dir)
74 "Find the Emacs version for the Emacs src in DIR.
75 Return a tuple of ( EMACSNAME . VERSION )."
76 (let ((buff (get-buffer-create " *emacs-query*"))
79 (with-current-buffer buff
81 (setq default-directory
(file-name-as-directory dir
))
82 ;(call-process "egrep" nil buff nil "-n" "-e" "^version=" "Makefile")
83 (call-process "egrep" nil buff nil
"-n" "-e" "AC_INIT" "configure.in")
84 (goto-char (point-min))
85 ;(re-search-forward "version=\\([0-9.]+\\)")
88 ((file-exists-p "version.sh")
90 (insert-file-contents "version.sh")
91 (goto-char (point-min))
92 (re-search-forward "emacs_major_version=\\([0-9]+\\)
93 emacs_minor_version=\\([0-9]+\\)
94 emacs_beta_version=\\([0-9]+\\)")
95 (setq ver
(concat (match-string 1) "."
99 ;; Insert other Emacs here...
101 ;; Vaguely recent version of GNU Emacs?
103 (insert-file-contents "configure.in")
104 (goto-char (point-min))
105 (re-search-forward "AC_INIT(emacs,\\s-*\\([0-9.]+\\)\\s-*)")
106 (setq ver
(match-string 1))
112 (defclass ede-emacs-project
(ede-project eieio-instance-tracker
)
113 ((tracking-symbol :initform
'ede-emacs-project-list
)
115 "Project Type for the Emacs source code."
116 :method-invocation-order
:depth-first
)
118 (defun ede-emacs-load (dir &optional rootproj
)
119 "Return an Emacs Project object if there is a match.
120 Return nil if there isn't one.
121 Argument DIR is the directory it is created for.
122 ROOTPROJ is nil, since there is only one project."
123 (or (ede-emacs-file-existing dir
)
124 ;; Doesn't already exist, so let's make one.
125 (let* ((vertuple (ede-emacs-version dir
)))
126 (ede-emacs-project (car vertuple
)
128 :version
(cdr vertuple
)
129 :directory
(file-name-as-directory dir
)
130 :file
(expand-file-name "src/emacs.c"
132 (ede-add-project-to-global-list this
)
137 (add-to-list 'ede-project-class-files
138 (ede-project-autoload "emacs"
141 :proj-file
"src/emacs.c"
142 :proj-root
'ede-emacs-project-root
143 :load-type
'ede-emacs-load
144 :class-sym
'ede-emacs-project
148 (defclass ede-emacs-target-c
(ede-target)
150 "EDE Emacs Project target for C code.
151 All directories need at least one target.")
153 (defclass ede-emacs-target-el
(ede-target)
155 "EDE Emacs Project target for Emacs Lisp code.
156 All directories need at least one target.")
158 (defclass ede-emacs-target-misc
(ede-target)
160 "EDE Emacs Project target for Misc files.
161 All directories need at least one target.")
163 (defmethod initialize-instance ((this ede-emacs-project
)
165 "Make sure the targets slot is bound."
167 (unless (slot-boundp this
'targets
)
168 (oset this
:targets nil
)))
172 (defmethod ede-project-root-directory ((this ede-emacs-project
)
174 "Return the root for THIS Emacs project with file."
175 (ede-up-directory (file-name-directory (oref this file
))))
177 (defmethod ede-project-root ((this ede-emacs-project
))
181 (defmethod ede-find-subproject-for-directory ((proj ede-emacs-project
)
183 "Return PROJ, for handling all subdirs below DIR."
186 ;;; TARGET MANAGEMENT
188 (defun ede-emacs-find-matching-target (class dir targets
)
189 "Find a target that is a CLASS and is in DIR in the list of TARGETS."
192 (when (and (object-of-class-p T class
)
193 (string= (oref T
:path
) dir
))
198 (defmethod ede-find-target ((proj ede-emacs-project
) buffer
)
199 "Find an EDE target in PROJ for BUFFER.
200 If one doesn't exist, create a new one for this directory."
201 (let* ((ext (file-name-extension (buffer-file-name buffer
)))
202 (cls (cond ((not ext
)
203 'ede-emacs-target-misc
)
204 ((string-match "c\\|h" ext
)
206 ((string-match "elc?" ext
)
207 'ede-emacs-target-el
)
208 (t 'ede-emacs-target-misc
)))
209 (targets (oref proj targets
))
210 (dir default-directory
)
211 (ans (ede-emacs-find-matching-target cls dir targets
))
214 (setq ans
(make-instance
216 :name
(file-name-nondirectory
217 (directory-file-name dir
))
220 (object-add-to-list proj
:targets ans
)
224 ;;; UTILITIES SUPPORT.
226 (defmethod ede-preprocessor-map ((this ede-emacs-target-c
))
227 "Get the pre-processor map for Emacs C code.
228 All files need the macros from lisp.h!"
229 (require 'semantic
/db
)
230 (let* ((proj (ede-target-parent this
))
231 (root (ede-project-root proj
))
232 (table (semanticdb-file-table-object
233 (ede-expand-filename root
"lisp.h")))
234 (config (semanticdb-file-table-object
235 (ede-expand-filename root
"config.h")))
239 (when (semanticdb-needs-refresh-p table
)
240 (semanticdb-refresh-table table
))
241 (setq filemap
(append filemap
(oref table lexical-table
)))
244 (when (semanticdb-needs-refresh-p config
)
245 (semanticdb-refresh-table config
))
246 (setq filemap
(append filemap
(oref config lexical-table
)))
251 (defun ede-emacs-find-in-directories (name base dirs
)
252 "Find NAME is BASE directory sublist of DIRS."
254 (while (and dirs
(not ans
))
255 (let* ((D (car dirs
))
256 (ed (expand-file-name D base
))
257 (ef (expand-file-name name ed
)))
258 (if (file-exists-p ef
)
260 ;; Not in this dir? How about subdirs?
261 (let ((dirfile (directory-files ed t
))
264 ;; Get all the subdirs.
266 (when (and (file-directory-p DF
)
267 (not (string-match "\\.$" DF
)))
270 (setq ans
(ede-emacs-find-in-directories name ed moredirs
))
272 (setq dirs
(cdr dirs
))))
275 (defmethod ede-expand-filename-impl ((proj ede-emacs-project
) name
)
276 "Within this project PROJ, find the file NAME.
277 Knows about how the Emacs source tree is organized."
278 (let* ((ext (file-name-extension name
))
279 (root (ede-project-root proj
))
280 (dir (ede-project-root-directory root
))
283 ((string-match "h\\|c" ext
)
284 '("src" "lib-src" "lwlib"))
285 ((string-match "elc?" ext
)
287 ((string-match "texi" ext
)
291 (if (not dirs
) (call-next-method)
292 (ede-emacs-find-in-directories name dir dirs
))
298 ;; generated-autoload-file: "loaddefs.el"
299 ;; generated-autoload-load-name: "ede/emacs"
302 ;;; ede/emacs.el ends here