(select-safe-coding-system): If the file
[bpt/emacs.git] / lisp / cedet / ede / pmake.el
CommitLineData
acc33231
CY
1;;; ede-pmake.el --- EDE Generic Project Makefile code generator.
2
3;;; Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
4;;; 2007, 2008, 2009 Free Software Foundation, Inc.
5
6;; Author: Eric M. Ludlam <zappo@gnu.org>
7;; Keywords: project, make
8
9;; This file is part of GNU Emacs.
10
11;; GNU Emacs is free software: you can redistribute it and/or modify
12;; it under the terms of the GNU General Public License as published by
13;; the Free Software Foundation, either version 3 of the License, or
14;; (at your option) any later version.
15
16;; GNU Emacs is distributed in the hope that it will be useful,
17;; but WITHOUT ANY WARRANTY; without even the implied warranty of
18;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19;; GNU General Public License for more details.
20
21;; You should have received a copy of the GNU General Public License
22;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
23
24;;; Commentary:
25;;
26;; Code generator for Makefiles.
27;;
28;; Here is how it should work:
29;; 1) Collect information about the project and targets
30;; 2) Insert header into the Makefile
31;; 3) Insert basic variables (target/source)
32;; 4) Conditional
33;; a) Makefile
34;; 1) Insert support variables (compiler variables, etc)
35;; 2) Insert VERSION and DISTDIR
36;; 3) Specify top build dir if necessary
37;; 4) Specify compile/link commands (c, etc)
38;; 5) Specify dependency files
39;; 6) Specify all: target
40;; 7) Include dependency files
41;; 8) Insert commonized target specify rules
42;; 9) Insert clean: and dist: rules
43;; b) Automake file
44;; 1) Insert distribution source variables for targets
45;; 2) Insert user requested rules
46
47(require 'ede/proj)
48(require 'ede/proj-obj)
49(require 'ede/proj-comp)
50
447a30f3
CY
51(declare-function ede-srecode-setup "ede/srecode")
52(declare-function ede-srecode-insert "ede/srecode")
53
acc33231
CY
54;;; Code:
55(defmethod ede-proj-makefile-create ((this ede-proj-project) mfilename)
56 "Create a Makefile for all Makefile targets in THIS.
57MFILENAME is the makefile to generate."
447a30f3 58 (require 'ede/srecode)
acc33231
CY
59 (let ((mt nil)
60 (isdist (string= mfilename (ede-proj-dist-makefile this)))
61 (depth 0)
62 (orig-buffer nil)
63 (buff-to-kill nil)
64 )
65 ;; Find out how deep this project is.
66 (let ((tmp this))
67 (while (setq tmp (ede-parent-project tmp))
68 (setq depth (1+ depth))))
69 ;; Collect the targets that belong in a makefile.
70 (mapc
71 (lambda (obj)
72 (if (and (obj-of-class-p obj 'ede-proj-target-makefile)
73 (string= (oref obj makefile) mfilename))
74 (setq mt (cons obj mt))))
75 (oref this targets))
76 ;; Fix the order so things compile in the right direction.
77 (setq mt (nreverse mt))
78 ;; Add in the header part of the Makefile*
79 (save-excursion
80 (setq orig-buffer (get-file-buffer mfilename))
81 (set-buffer (setq buff-to-kill (find-file-noselect mfilename)))
82 (goto-char (point-min))
83 (if (and
84 (not (eobp))
85 (not (looking-at "# Automatically Generated \\w+ by EDE.")))
86 (if (not (y-or-n-p (format "Really replace %s? " mfilename)))
87 (error "Not replacing Makefile"))
88 (message "Replace EDE Makefile"))
89 (erase-buffer)
90 (ede-srecode-setup)
91 ;; Insert a giant pile of stuff that is common between
92 ;; one of our Makefiles, and a Makefile.in
93 (ede-srecode-insert
94 "file:ede-empty"
95 "MAKETYPE"
96 (with-slots (makefile-type) this
97 (cond ((eq makefile-type 'Makefile) "make")
98 ((eq makefile-type 'Makefile.in) "autoconf")
99 ((eq makefile-type 'Makefile.am) "automake")
100 (t (error ":makefile-type in project invalid")))))
101
102 ;; Just this project's variables
103 (ede-proj-makefile-insert-variables this)
104
105 ;; Space
106 (insert "\n")
107
108 (cond
109 ((eq (oref this makefile-type) 'Makefile)
110 ;; Make sure the user has the right kind of make
111 (ede-make-check-version)
112
113 (let* ((targ (if isdist (oref this targets) mt))
114 (sp (oref this subproj))
115 (df (apply 'append
116 (mapcar (lambda (tg)
117 (ede-proj-makefile-dependency-files tg))
118 targ))))
119 ;; Distribution variables
120 (ede-compiler-begin-unique
121 (mapc 'ede-proj-makefile-insert-variables targ))
122 ;; Only add the distribution stuff in when depth != 0
123 (let ((top (ede-toplevel this))
124 (tmp this)
125 (subdir ""))
126 (insert "VERSION=" (oref top version) "\n"
127 "DISTDIR=$(top)" (oref top name) "-$(VERSION)")
128 (while (ede-parent-project tmp)
129 (setq subdir
130 (concat
131 "/"
132 (file-name-nondirectory
133 (directory-file-name
134 (file-name-directory (oref tmp file))))
135 subdir)
136 tmp (ede-parent-project tmp)))
137 (insert subdir "\n"))
138 ;; Some built in variables for C code
139 (if df
140 (let ((tc depth))
141 (insert "top_builddir = ")
142 (while (/= 0 tc)
143 (setq tc (1- tc))
144 (insert "..")
145 (if (/= tc 0) (insert "/")))
146 (insert "\n")))
147 (insert "\n")
148 ;; Create a variable with all the dependency files to include
149 ;; These methods borrowed from automake.
150 (if (and (oref this automatic-dependencies) df)
151 (progn
152 (insert "DEP_FILES="
153 (mapconcat (lambda (f)
154 (concat ".deps/"
155 (file-name-nondirectory
156 (file-name-sans-extension
157 f)) ".P"))
158 df " "))))
159 ;;
160 ;; Insert ALL Rule
161 ;;
162 (insert "\n\nall:")
163 (mapc (lambda (c)
164 (if (and (slot-exists-p c 'partofall) (oref c partofall))
165 ;; Only insert this rule if it is a part of ALL.
166 (insert " " (ede-proj-makefile-target-name c))))
167 targ)
168 (mapc (lambda (c)
169 (insert " " (ede-name c))
170 )
171 sp)
172 (insert "\n\n")
173 ;;
174 ;; Add in the include files
175 ;;
176 (mapc (lambda (c)
177 (insert "include " c "\n\n"))
178 (oref this include-file))
179 ;; Some C inference rules
180 ;; Dependency rules borrowed from automake.
181 ;;
182 ;; NOTE: This is GNU Make specific.
183 (if (and (oref this automatic-dependencies) df)
184 (insert "DEPS_MAGIC := $(shell mkdir .deps > /dev/null "
185 "2>&1 || :)\n"
186 "-include $(DEP_FILES)\n\n"))
187 ;;
188 ;; General makefile rules stored in the individual targets
189 ;;
190 (ede-compiler-begin-unique
191 (ede-proj-makefile-insert-rules this)
192 (mapc 'ede-proj-makefile-insert-rules targ))
193 ;;
194 ;; phony targets for sub projects
195 ;;
196 (mapc 'ede-proj-makefile-insert-subproj-rules sp)
197 ;;
198 ;; Distribution rules such as CLEAN and DIST
199 ;;
200 (when isdist
201 (ede-proj-makefile-tags this mt)
202 (ede-proj-makefile-insert-dist-rules this)))
203 (save-buffer))
204 ((eq (oref this makefile-type) 'Makefile.in)
205 (error "Makefile.in is not supported"))
206 ((eq (oref this makefile-type) 'Makefile.am)
b82525f2 207 (require 'ede/pconf)
acc33231
CY
208 ;; Distribution variables
209 (let ((targ (if isdist (oref this targets) mt)))
210 (ede-compiler-begin-unique
211 (mapc 'ede-proj-makefile-insert-automake-pre-variables targ))
212 (ede-compiler-begin-unique
213 (mapc 'ede-proj-makefile-insert-source-variables targ))
214 (ede-compiler-begin-unique
215 (mapc 'ede-proj-makefile-insert-automake-post-variables targ))
216 (ede-compiler-begin-unique
217 (ede-proj-makefile-insert-user-rules this))
218 (insert "\n# End of Makefile.am\n")
219 (save-buffer))
220 )
221 (t (error "Unknown makefile type when generating Makefile")))
222 ;; Put the cursor in a nice place
223 (goto-char (point-min)))
224 ;; If we have an original buffer, then don't kill it.
225 (when (not orig-buffer)
226 (kill-buffer buff-to-kill))
227 ))
228
229;;; VARIABLE insertion
230;;
231(defun ede-pmake-end-of-variable ()
232 "Move to the end of the variable declaration under point."
233 (end-of-line)
234 (while (= (preceding-char) ?\\)
235 (forward-char 1)
236 (end-of-line))
237 )
238
239(defmacro ede-pmake-insert-variable-shared (varname &rest body)
240 "Add VARNAME into the current Makefile.
241Execute BODY in a location where a value can be placed."
242 `(let ((addcr t) (v ,varname))
e6e267fc
CY
243 (if (re-search-backward (concat "^" v "\\s-*=") nil t)
244 (progn
245 (ede-pmake-end-of-variable)
246 (if (< (current-column) 40)
247 (if (and (/= (preceding-char) ?=)
248 (/= (preceding-char) ? ))
249 (insert " "))
250 (insert "\\\n "))
251 (setq addcr nil))
252 (insert v "="))
253 ,@body
254 (if addcr (insert "\n"))
255 (goto-char (point-max))))
256(put 'ede-pmake-insert-variable-shared 'lisp-indent-function 1)
257
acc33231
CY
258;;; SOURCE VARIABLE NAME CONSTRUCTION
259
260(defsubst ede-pmake-varname (obj)
261 "Convert OBJ into a variable name name.
262Change . to _ in the variable name."
263 (let ((name (oref obj name)))
264 (while (string-match "\\." name)
265 (setq name (replace-match "_" nil t name)))
266 name))
267
268(defmethod ede-proj-makefile-sourcevar ((this ede-proj-target))
269 "Return the variable name for THIS's sources."
270 (concat (ede-pmake-varname this) "_YOU_FOUND_A_BUG"))
271
272;;; DEPENDENCY FILE GENERATOR LISTS
273;;
274(defmethod ede-proj-makefile-dependency-files ((this ede-proj-target))
275 "Return a list of source files to convert to dependencies.
276Argument THIS is the target to get sources from."
277 nil)
278
279;;; GENERIC VARIABLES
280;;
281(defmethod ede-proj-makefile-configuration-variables ((this ede-proj-project)
282 configuration)
283 "Return a list of configuration variables from THIS.
284Use CONFIGURATION as the current configuration to query."
285 (cdr (assoc configuration (oref this configuration-variables))))
286
287(defmethod ede-proj-makefile-insert-variables-new ((this ede-proj-project))
288 "Insert variables needed by target THIS.
289
290NOTE: Not yet in use! This is part of an SRecode conversion of
291 EDE that is in progress."
292; (let ((conf-table (ede-proj-makefile-configuration-variables
293; this (oref this configuration-default)))
294; (conf-done nil))
295;
296; (ede-srecode-insert-with-dictionary
297; "declaration:ede-vars"
298;
299; ;; Insert all variables, and augment them with details from
300; ;; the current configuration.
301; (mapc (lambda (c)
302;
303; (let ((ldict (srecode-dictionary-add-section-dictionary
304; dict "VARIABLE"))
305; )
306; (srecode-dictionary-set-value ldict "NAME" (car c))
307; (if (assoc (car c) conf-table)
308; (let ((vdict (srecode-dictionary-add-section-dictionary
309; ldict "VALUE")))
310; (srecode-dictionary-set-value
311; vdict "VAL" (cdr (assoc (car c) conf-table)))
312; (setq conf-done (cons (car c) conf-done))))
313; (let ((vdict (srecode-dictionary-add-section-dictionary
314; ldict "VALUE")))
315; (srecode-dictionary-set-value vdict "VAL" (cdr c))))
316; )
317;
318; (oref this variables))
319;
320; ;; Add in all variables from the configuration not allready covered.
321; (mapc (lambda (c)
322;
323; (if (member (car c) conf-done)
324; nil
325; (let* ((ldict (srecode-dictionary-add-section-dictionary
326; dict "VARIABLE"))
327; (vdict (srecode-dictionary-add-section-dictionary
328; ldict "VALUE"))
329; )
330; (srecode-dictionary-set-value ldict "NAME" (car c))
331; (srecode-dictionary-set-value vdict "VAL" (cdr c))))
332; )
333;
334; conf-table)
335;
336
337 ;; @TODO - finish off this function, and replace the below fcn
338
339; ))
340 )
341
342(defmethod ede-proj-makefile-insert-variables ((this ede-proj-project))
343 "Insert variables needed by target THIS."
344 (let ((conf-table (ede-proj-makefile-configuration-variables
345 this (oref this configuration-default)))
346 (conf-done nil))
347 ;; Insert all variables, and augment them with details from
348 ;; the current configuration.
349 (mapc (lambda (c)
350 (insert (car c) "=")
351 (if (assoc (car c) conf-table)
352 (progn
353 (insert (cdr (assoc (car c) conf-table)) " ")
354 (setq conf-done (cons (car c) conf-done))))
355 (insert (cdr c) "\n"))
356 (oref this variables))
357 ;; Add in all variables from the configuration not allready covered.
358 (mapc (lambda (c)
359 (if (member (car c) conf-done)
360 nil
361 (insert (car c) "=" (cdr c) "\n")))
362 conf-table))
363 (let* ((top "")
364 (tmp this))
365 (while (ede-parent-project tmp)
366 (setq tmp (ede-parent-project tmp)
367 top (concat "../" top)))
368 (insert "\ntop=" top))
369 (insert "\nede_FILES=" (file-name-nondirectory (oref this file)) " "
370 (file-name-nondirectory (ede-proj-dist-makefile this)) "\n"))
371
372(defmethod ede-proj-makefile-insert-source-variables ((this ede-proj-target)
373 &optional
374 moresource)
375 "Insert the source variables needed by THIS.
376Optional argument MORESOURCE is a list of additional sources to add to the
377sources variable."
378 (let ((sv (ede-proj-makefile-sourcevar this)))
379 ;; This variable may be shared between targets
380 (ede-pmake-insert-variable-shared (cond ((listp sv) (car sv))
381 (t sv))
382 (insert (mapconcat (lambda (a) a) (oref this source) " "))
383 (if moresource
384 (insert " \\\n " (mapconcat (lambda (a) a) moresource " ") "")))))
385
386(defmethod ede-proj-makefile-insert-variables ((this ede-proj-target) &optional
387 moresource)
388 "Insert variables needed by target THIS.
389Optional argument MORESOURCE is a list of additional sources to add to the
390sources variable."
391 (ede-proj-makefile-insert-source-variables this moresource)
392 )
393
394(defmethod ede-proj-makefile-configuration-variables ((this ede-proj-target-makefile)
395 configuration)
396 "Return a list of configuration variables from THIS.
397Use CONFIGURATION as the current configuration to query."
398 (cdr (assoc configuration (oref this configuration-variables))))
399
400(defmethod ede-proj-makefile-insert-variables ((this ede-proj-target-makefile)
401 &optional moresource)
402 "Insert variables needed by target THIS.
403Optional argument MORESOURCE is a list of additional sources to add to the
404sources variable."
405 (call-next-method)
406 (let* ((proj (ede-target-parent this))
407 (conf-table (ede-proj-makefile-configuration-variables
408 this (oref proj configuration-default)))
409 (conf-done nil)
410 )
411 ;; Add in all variables from the configuration not allready covered.
412 (mapc (lambda (c)
413 (if (member (car c) conf-done)
414 nil
415 (insert (car c) "=" (cdr c) "\n")))
416 conf-table))
417 (let ((comp (ede-proj-compilers this))
418 (link (ede-proj-linkers this))
419 (name (ede-proj-makefile-target-name this))
420 (src (oref this source)))
421 (while comp
422 (ede-compiler-only-once (car comp)
423 (ede-proj-makefile-insert-object-variables (car comp) name src)
424 (ede-proj-makefile-insert-variables (car comp)))
425 (setq comp (cdr comp)))
426 (while link
427 (ede-linker-only-once (car link)
428 (ede-proj-makefile-insert-variables (car link)))
429 (setq link (cdr link)))))
430
431(defmethod ede-proj-makefile-insert-automake-pre-variables
432 ((this ede-proj-target))
433 "Insert variables needed by target THIS in Makefile.am before SOURCES."
434 nil)
435
436(defmethod ede-proj-makefile-insert-automake-post-variables
437 ((this ede-proj-target))
438 "Insert variables needed by target THIS in Makefile.am after SOURCES."
439 nil)
440
441;;; GARBAGE PATTERNS
442;;
443(defmethod ede-proj-makefile-garbage-patterns ((this ede-proj-project))
444 "Return a list of patterns that are considered garbage to THIS.
445These are removed with make clean."
446 (let ((mc (ede-map-targets
447 this (lambda (c) (ede-proj-makefile-garbage-patterns c))))
448 (uniq nil))
449 (setq mc (sort (apply 'append mc) 'string<))
450 ;; Filter out duplicates from the targets.
451 (while mc
452 (if (and (car uniq) (string= (car uniq) (car mc)))
453 nil
454 (setq uniq (cons (car mc) uniq)))
455 (setq mc (cdr mc)))
456 (nreverse uniq)))
457
458(defmethod ede-proj-makefile-garbage-patterns ((this ede-proj-target))
459 "Return a list of patterns that are considered garbage to THIS.
460These are removed with make clean."
461 ;; Get the the source object from THIS, and use the specified garbage.
462 (let ((src (ede-target-sourcecode this))
463 (garb nil))
464 (while src
465 (setq garb (append (oref (car src) garbagepattern) garb)
466 src (cdr src)))
467 garb))
468
469
470;;; RULES
471;;
472(defmethod ede-proj-makefile-insert-subproj-rules ((this ede-proj-project))
473 "Insert a rule for the project THIS which should be a subproject."
474 (insert ".PHONY:" (ede-name this))
475 (newline)
476 (insert (ede-name this) ":")
477 (newline)
478 (insert "\t$(MAKE) -C " (directory-file-name (ede-subproject-relative-path this)))
479 (newline)
480 (newline)
481 )
482
483(defmethod ede-proj-makefile-insert-rules ((this ede-proj-project))
484 "Insert rules needed by THIS target."
485 (mapc 'ede-proj-makefile-insert-rules (oref this inference-rules))
486 )
487
488(defmethod ede-proj-makefile-insert-dist-dependencies ((this ede-proj-project))
489 "Insert any symbols that the DIST rule should depend on.
490Argument THIS is the project that should insert stuff."
491 (mapc 'ede-proj-makefile-insert-dist-dependencies (oref this targets))
492 )
493
494(defmethod ede-proj-makefile-insert-dist-dependencies ((this ede-proj-target))
495 "Insert any symbols that the DIST rule should depend on.
496Argument THIS is the target that should insert stuff."
497 nil)
498
499(defmethod ede-proj-makefile-insert-dist-filepatterns ((this ede-proj-target))
500 "Insert any symbols that the DIST rule should depend on.
501Argument THIS is the target that should insert stuff."
502 (ede-proj-makefile-insert-dist-dependencies this)
503 )
504
505(defmethod ede-proj-makefile-insert-dist-rules ((this ede-proj-project))
506 "Insert distribution rules for THIS in a Makefile, such as CLEAN and DIST."
507 (let ((junk (ede-proj-makefile-garbage-patterns this))
508 tmp)
509 ;; Build CLEAN, DIST, TAG, and other rules here.
510 (if junk
511 (insert "\nclean:\n"
512 "\trm -f "
513 (mapconcat (lambda (c) c) junk " ")
514 "\n\n"))
515 ;; @TODO: ^^^ Clean should also recurse. ^^^
516
517 (insert ".PHONY: dist\n")
518 (insert "\ndist:")
519 (ede-proj-makefile-insert-dist-dependencies this)
520 (insert "\n")
521 (unless (or (ede-subproject-p this)
522 (oref this metasubproject))
523 ;; Only delete if we are the toplevel project.
524 (insert "\trm -rf $(DISTDIR)\n"))
525 (insert "\tmkdir $(DISTDIR)\n") ;We may need a -p, but I think not.
526 (setq tmp (oref this targets))
527 (insert "\tcp")
528 (while tmp
529 (let ((sv (ede-proj-makefile-sourcevar (car tmp))))
530 (if (listp sv)
531 ;; Handle special case variables.
532 (cond ((eq (cdr sv) 'share)
533 ;; This variable may be shared between multiple targets.
534 (if (re-search-backward (concat "\\$(" (car sv) ")")
535 (save-excursion
536 (beginning-of-line)
537 (point))
538 t)
539 ;; If its already in the dist target, then skip it.
540 nil
541 (setq sv (car sv))))
542 (t (setq sv (car sv)))))
543 (if (stringp sv)
544 (insert " $(" sv ")"))
545 (ede-proj-makefile-insert-dist-filepatterns (car tmp))
546 (setq tmp (cdr tmp))))
547 (insert " $(ede_FILES) $(DISTDIR)\n")
548
549 ;; Call our sub projects.
550 (ede-map-subprojects
551 this (lambda (sproj)
552 (let ((rp (directory-file-name (ede-subproject-relative-path sproj))))
553 (insert "\t$(MAKE) -C " rp " $(MFLAGS) DISTDIR=$(DISTDIR)/" rp
554 " dist"
555 "\n"))))
556
557 ;; Tar up the stuff.
558 (unless (or (ede-subproject-p this)
559 (oref this metasubproject))
560 (insert "\ttar -cvzf $(DISTDIR).tar.gz $(DISTDIR)\n"
561 "\trm -rf $(DISTDIR)\n"))
562
563 ;; Make sure the Makefile is ok.
564 (insert "\n"
565 (file-name-nondirectory (buffer-file-name)) ": "
566 (file-name-nondirectory (oref this file)) "\n"
567;; "$(EMACS) -batch Project.ede -l ede -f ede-proj-regenerate"
568 "\t@echo Makefile is out of date! "
569 "It needs to be regenerated by EDE.\n"
570 "\t@echo If you have not modified Project.ede, you can"
571 " use 'touch' to update the Makefile time stamp.\n"
572 "\t@false\n\n"
573 "\n\n# End of Makefile\n")))
574
575(defmethod ede-proj-makefile-insert-rules ((this ede-proj-target))
576 "Insert rules needed by THIS target."
577 nil)
578
579(defmethod ede-proj-makefile-insert-rules ((this ede-proj-target-makefile))
580 "Insert rules needed by THIS target."
581 (mapc 'ede-proj-makefile-insert-rules (oref this rules))
582 (let ((c (ede-proj-compilers this)))
583 (when c
584 (mapc 'ede-proj-makefile-insert-rules c)
585 (if (oref this phony)
586 (insert ".PHONY: " (ede-proj-makefile-target-name this) "\n"))
587 (insert (ede-proj-makefile-target-name this) ": "
588 (ede-proj-makefile-dependencies this) "\n")
589 (ede-proj-makefile-insert-commands this)
590 )))
591
592(defmethod ede-proj-makefile-insert-commands ((this ede-proj-target-makefile))
593 "Insert the commands needed by target THIS.
594For targets, insert the commands needed by the chosen compiler."
595 (mapc 'ede-proj-makefile-insert-commands (ede-proj-compilers this))
596 (when (object-assoc t :uselinker (ede-proj-compilers this))
597 (mapc 'ede-proj-makefile-insert-commands (ede-proj-linkers this))))
598
599
600(defmethod ede-proj-makefile-insert-user-rules ((this ede-proj-project))
601 "Insert user specified rules needed by THIS target.
602This is different from `ede-proj-makefile-insert-rules' in that this
603function won't create the building rules which are auto created with
604automake."
605 (mapc 'ede-proj-makefile-insert-user-rules (oref this inference-rules)))
606
607(defmethod ede-proj-makefile-insert-user-rules ((this ede-proj-target))
608 "Insert user specified rules needed by THIS target."
609 (mapc 'ede-proj-makefile-insert-rules (oref this rules)))
610
611(defmethod ede-proj-makefile-dependencies ((this ede-proj-target-makefile))
612 "Return a string representing the dependencies for THIS.
613Some compilers only use the first element in the dependencies, others
614have a list of intermediates (object files), and others don't care.
615This allows customization of how these elements appear."
616 (let* ((c (ede-proj-compilers this))
b90caf50 617 (io (eval (cons 'or (mapcar 'ede-compiler-intermediate-objects-p c))))
acc33231
CY
618 (out nil))
619 (if io
620 (progn
621 (while c
622 (setq out
623 (concat out "$(" (ede-compiler-intermediate-object-variable
624 (car c)
625 (ede-proj-makefile-target-name this)) ")")
626 c (cdr c)))
627 out)
628 (let ((sv (ede-proj-makefile-sourcevar this))
629 (aux (oref this auxsource)))
630 (setq out
631 (if (and (stringp sv) (not (string= sv "")))
632 (concat "$(" sv ")")
633 ""))
634 (while aux
635 (setq out (concat out " " (car aux)))
636 (setq aux (cdr aux)))
637 out))))
638
639;; Tags
640(defmethod ede-proj-makefile-tags ((this ede-proj-project) targets)
641 "Insert into the current location rules to make recursive TAGS files.
642Argument THIS is the project to create tags for.
643Argument TARGETS are the targets we should depend on for TAGS."
644 (insert "tags: ")
645 (let ((tg targets))
646 ;; Loop over all source variables and insert them
647 (while tg
648 (insert "$(" (ede-proj-makefile-sourcevar (car tg)) ") ")
649 (setq tg (cdr tg)))
650 (insert "\n")
651 (if targets
652 (insert "\tetags $^\n"))
653 ;; Now recurse into all subprojects
654 (setq tg (oref this subproj))
655 (while tg
656 (insert "\t$(MAKE) -C " (ede-subproject-relative-path (car tg)) " $(MFLAGS) $@\n")
657 (setq tg (cdr tg)))
658 (insert "\n")))
659
660
661(provide 'ede/pmake)
662
3999968a 663;; arch-tag: 7ad8e19f-cdee-484c-8caf-f15cb0fc4df2
acc33231 664;;; ede/pmake.el ends here