]> code.delx.au - gnu-emacs/blob - lisp/progmodes/inf-lisp.el
dd3f6a1262da50b2b16d0b717dae437254c1ff07
[gnu-emacs] / lisp / progmodes / inf-lisp.el
1 ;;; inf-lisp.el --- an inferior-lisp mode
2
3 ;; Copyright (C) 1988, 1993, 1994, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011
4 ;; Free Software Foundation, Inc.
5
6 ;; Author: Olin Shivers <shivers@cs.cmu.edu>
7 ;; Keywords: processes, lisp
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 ;; Hacked from tea.el by Olin Shivers (shivers@cs.cmu.edu). 8/88
27
28 ;; This file defines a lisp-in-a-buffer package (inferior-lisp mode)
29 ;; built on top of comint mode. This version is more featureful,
30 ;; robust, and uniform than the Emacs 18 version. The key bindings are
31 ;; also more compatible with the bindings of Hemlock and Zwei (the
32 ;; Lisp Machine emacs).
33
34 ;; Since this mode is built on top of the general command-interpreter-in-
35 ;; a-buffer mode (comint mode), it shares a common base functionality,
36 ;; and a common set of bindings, with all modes derived from comint mode.
37 ;; This makes these modes easier to use.
38
39 ;; For documentation on the functionality provided by comint mode, and
40 ;; the hooks available for customising it, see the file comint.el.
41 ;; For further information on inferior-lisp mode, see the comments below.
42
43 ;; Needs fixin:
44 ;; The load-file/compile-file default mechanism could be smarter -- it
45 ;; doesn't know about the relationship between filename extensions and
46 ;; whether the file is source or executable. If you compile foo.lisp
47 ;; with compile-file, then the next load-file should use foo.bin for
48 ;; the default, not foo.lisp. This is tricky to do right, particularly
49 ;; because the extension for executable files varies so much (.o, .bin,
50 ;; .lbin, .mo, .vo, .ao, ...).
51 ;;
52 ;; It would be nice if inferior-lisp (and inferior scheme, T, ...) modes
53 ;; had a verbose minor mode wherein sending or compiling defuns, etc.
54 ;; would be reflected in the transcript with suitable comments, e.g.
55 ;; ";;; redefining fact". Several ways to do this. Which is right?
56 ;;
57 ;; When sending text from a source file to a subprocess, the process-mark can
58 ;; move off the window, so you can lose sight of the process interactions.
59 ;; Maybe I should ensure the process mark is in the window when I send
60 ;; text to the process? Switch selectable?
61
62 ;;; Code:
63
64 (require 'comint)
65 (require 'lisp-mode)
66
67 \f
68 (defgroup inferior-lisp nil
69 "Run an outside Lisp in an Emacs buffer."
70 :group 'lisp
71 :version "22.1")
72
73 ;;;###autoload
74 (defcustom inferior-lisp-filter-regexp
75 (purecopy "\\`\\s *\\(:\\(\\w\\|\\s_\\)\\)?\\s *\\'")
76 "*What not to save on inferior Lisp's input history.
77 Input matching this regexp is not saved on the input history in Inferior Lisp
78 mode. Default is whitespace followed by 0 or 1 single-letter colon-keyword
79 \(as in :a, :c, etc.)"
80 :type 'regexp
81 :group 'inferior-lisp)
82
83 (defvar inferior-lisp-mode-map
84 (let ((map (copy-keymap comint-mode-map)))
85 (set-keymap-parent map lisp-mode-shared-map)
86 (define-key map "\C-x\C-e" 'lisp-eval-last-sexp)
87 (define-key map "\C-c\C-l" 'lisp-load-file)
88 (define-key map "\C-c\C-k" 'lisp-compile-file)
89 (define-key map "\C-c\C-a" 'lisp-show-arglist)
90 (define-key map "\C-c\C-d" 'lisp-describe-sym)
91 (define-key map "\C-c\C-f" 'lisp-show-function-documentation)
92 (define-key map "\C-c\C-v" 'lisp-show-variable-documentation)
93 map))
94
95 ;;; These commands augment Lisp mode, so you can process Lisp code in
96 ;;; the source files.
97 (define-key lisp-mode-map "\M-\C-x" 'lisp-eval-defun) ; Gnu convention
98 (define-key lisp-mode-map "\C-x\C-e" 'lisp-eval-last-sexp) ; Gnu convention
99 (define-key lisp-mode-map "\C-c\C-e" 'lisp-eval-defun)
100 (define-key lisp-mode-map "\C-c\C-r" 'lisp-eval-region)
101 (define-key lisp-mode-map "\C-c\C-c" 'lisp-compile-defun)
102 (define-key lisp-mode-map "\C-c\C-z" 'switch-to-lisp)
103 (define-key lisp-mode-map "\C-c\C-l" 'lisp-load-file)
104 (define-key lisp-mode-map "\C-c\C-k" 'lisp-compile-file) ; "kompile" file
105 (define-key lisp-mode-map "\C-c\C-a" 'lisp-show-arglist)
106 (define-key lisp-mode-map "\C-c\C-d" 'lisp-describe-sym)
107 (define-key lisp-mode-map "\C-c\C-f" 'lisp-show-function-documentation)
108 (define-key lisp-mode-map "\C-c\C-v" 'lisp-show-variable-documentation)
109
110
111 ;;; This function exists for backwards compatibility.
112 ;;; Previous versions of this package bound commands to C-c <letter>
113 ;;; bindings, which is not allowed by the gnumacs standard.
114
115 ;;; "This function binds many inferior-lisp commands to C-c <letter> bindings,
116 ;;;where they are more accessible. C-c <letter> bindings are reserved for the
117 ;;;user, so these bindings are non-standard. If you want them, you should
118 ;;;have this function called by the inferior-lisp-load-hook:
119 ;;; (add-hook 'inferior-lisp-load-hook 'inferior-lisp-install-letter-bindings)
120 ;;;You can modify this function to install just the bindings you want."
121 (defun inferior-lisp-install-letter-bindings ()
122 (define-key lisp-mode-map "\C-ce" 'lisp-eval-defun-and-go)
123 (define-key lisp-mode-map "\C-cr" 'lisp-eval-region-and-go)
124 (define-key lisp-mode-map "\C-cc" 'lisp-compile-defun-and-go)
125 (define-key lisp-mode-map "\C-cz" 'switch-to-lisp)
126 (define-key lisp-mode-map "\C-cl" 'lisp-load-file)
127 (define-key lisp-mode-map "\C-ck" 'lisp-compile-file)
128 (define-key lisp-mode-map "\C-ca" 'lisp-show-arglist)
129 (define-key lisp-mode-map "\C-cd" 'lisp-describe-sym)
130 (define-key lisp-mode-map "\C-cf" 'lisp-show-function-documentation)
131 (define-key lisp-mode-map "\C-cv" 'lisp-show-variable-documentation)
132
133 (define-key inferior-lisp-mode-map "\C-cl" 'lisp-load-file)
134 (define-key inferior-lisp-mode-map "\C-ck" 'lisp-compile-file)
135 (define-key inferior-lisp-mode-map "\C-ca" 'lisp-show-arglist)
136 (define-key inferior-lisp-mode-map "\C-cd" 'lisp-describe-sym)
137 (define-key inferior-lisp-mode-map "\C-cf" 'lisp-show-function-documentation)
138 (define-key inferior-lisp-mode-map "\C-cv"
139 'lisp-show-variable-documentation))
140
141 ;;;###autoload
142 (defcustom inferior-lisp-program (purecopy "lisp")
143 "*Program name for invoking an inferior Lisp in Inferior Lisp mode."
144 :type 'string
145 :group 'inferior-lisp)
146
147 ;;;###autoload
148 (defcustom inferior-lisp-load-command (purecopy "(load \"%s\")\n")
149 "*Format-string for building a Lisp expression to load a file.
150 This format string should use `%s' to substitute a file name
151 and should result in a Lisp expression that will command the inferior Lisp
152 to load that file. The default works acceptably on most Lisps.
153 The string \"(progn (load \\\"%s\\\" :verbose nil :print t) (values))\\n\"
154 produces cosmetically superior output for this application,
155 but it works only in Common Lisp."
156 :type 'string
157 :group 'inferior-lisp)
158
159 ;;;###autoload
160 (defcustom inferior-lisp-prompt (purecopy "^[^> \n]*>+:? *")
161 "Regexp to recognize prompts in the Inferior Lisp mode.
162 Defaults to \"^[^> \\n]*>+:? *\", which works pretty good for Lucid, kcl,
163 and franz. This variable is used to initialize `comint-prompt-regexp' in the
164 Inferior Lisp buffer.
165
166 This variable is only used if the variable
167 `comint-use-prompt-regexp' is non-nil.
168
169 More precise choices:
170 Lucid Common Lisp: \"^\\\\(>\\\\|\\\\(->\\\\)+\\\\) *\"
171 franz: \"^\\\\(->\\\\|<[0-9]*>:\\\\) *\"
172 kcl: \"^>+ *\"
173
174 This is a fine thing to set in your .emacs file or through Custom."
175 :type 'regexp
176 :group 'inferior-lisp)
177
178 (defvar inferior-lisp-buffer nil "*The current inferior-lisp process buffer.
179
180 MULTIPLE PROCESS SUPPORT
181 ===========================================================================
182 To run multiple Lisp processes, you start the first up
183 with \\[inferior-lisp]. It will be in a buffer named `*inferior-lisp*'.
184 Rename this buffer with \\[rename-buffer]. You may now start up a new
185 process with another \\[inferior-lisp]. It will be in a new buffer,
186 named `*inferior-lisp*'. You can switch between the different process
187 buffers with \\[switch-to-buffer].
188
189 Commands that send text from source buffers to Lisp processes --
190 like `lisp-eval-defun' or `lisp-show-arglist' -- have to choose a process
191 to send to, when you have more than one Lisp process around. This
192 is determined by the global variable `inferior-lisp-buffer'. Suppose you
193 have three inferior Lisps running:
194 Buffer Process
195 foo inferior-lisp
196 bar inferior-lisp<2>
197 *inferior-lisp* inferior-lisp<3>
198 If you do a \\[lisp-eval-defun] command on some Lisp source code,
199 what process do you send it to?
200
201 - If you're in a process buffer (foo, bar, or *inferior-lisp*),
202 you send it to that process.
203 - If you're in some other buffer (e.g., a source file), you
204 send it to the process attached to buffer `inferior-lisp-buffer'.
205 This process selection is performed by function `inferior-lisp-proc'.
206
207 Whenever \\[inferior-lisp] fires up a new process, it resets
208 `inferior-lisp-buffer' to be the new process's buffer. If you only run
209 one process, this does the right thing. If you run multiple
210 processes, you can change `inferior-lisp-buffer' to another process
211 buffer with \\[set-variable].")
212
213 ;;;###autoload
214 (defvar inferior-lisp-mode-hook '()
215 "*Hook for customizing Inferior Lisp mode.")
216
217 (put 'inferior-lisp-mode 'mode-class 'special)
218
219 (define-derived-mode inferior-lisp-mode comint-mode "Inferior Lisp"
220 "Major mode for interacting with an inferior Lisp process.
221 Runs a Lisp interpreter as a subprocess of Emacs, with Lisp I/O through an
222 Emacs buffer. Variable `inferior-lisp-program' controls which Lisp interpreter
223 is run. Variables `inferior-lisp-prompt', `inferior-lisp-filter-regexp' and
224 `inferior-lisp-load-command' can customize this mode for different Lisp
225 interpreters.
226
227 For information on running multiple processes in multiple buffers, see
228 documentation for variable `inferior-lisp-buffer'.
229
230 \\{inferior-lisp-mode-map}
231
232 Customization: Entry to this mode runs the hooks on `comint-mode-hook' and
233 `inferior-lisp-mode-hook' (in that order).
234
235 You can send text to the inferior Lisp process from other buffers containing
236 Lisp source.
237 `switch-to-lisp' switches the current buffer to the Lisp process buffer.
238 `lisp-eval-defun' sends the current defun to the Lisp process.
239 `lisp-compile-defun' compiles the current defun.
240 `lisp-eval-region' sends the current region to the Lisp process.
241 `lisp-compile-region' compiles the current region.
242
243 Prefixing the lisp-eval/compile-defun/region commands with
244 a \\[universal-argument] causes a switch to the Lisp process buffer after sending
245 the text.
246
247 Commands:\\<inferior-lisp-mode-map>
248 \\[comint-send-input] after the end of the process' output sends the text from the
249 end of process to point.
250 \\[comint-send-input] before the end of the process' output copies the sexp ending at point
251 to the end of the process' output, and sends it.
252 \\[comint-copy-old-input] copies the sexp ending at point to the end of the process' output,
253 allowing you to edit it before sending it.
254 If `comint-use-prompt-regexp' is nil (the default), \\[comint-insert-input] on old input
255 copies the entire old input to the end of the process' output, allowing
256 you to edit it before sending it. When not used on old input, or if
257 `comint-use-prompt-regexp' is non-nil, \\[comint-insert-input] behaves according to
258 its global binding.
259 \\[backward-delete-char-untabify] converts tabs to spaces as it moves back.
260 \\[lisp-indent-line] indents for Lisp; with argument, shifts rest
261 of expression rigidly with the current line.
262 \\[indent-sexp] does \\[lisp-indent-line] on each line starting within following expression.
263 Paragraphs are separated only by blank lines. Semicolons start comments.
264 If you accidentally suspend your process, use \\[comint-continue-subjob]
265 to continue it."
266 (setq comint-prompt-regexp inferior-lisp-prompt)
267 (setq mode-line-process '(":%s"))
268 (lisp-mode-variables t)
269 (setq comint-get-old-input (function lisp-get-old-input))
270 (setq comint-input-filter (function lisp-input-filter)))
271
272 (defun lisp-get-old-input ()
273 "Return a string containing the sexp ending at point."
274 (save-excursion
275 (let ((end (point)))
276 (backward-sexp)
277 (buffer-substring (point) end))))
278
279 (defun lisp-input-filter (str)
280 "t if STR does not match `inferior-lisp-filter-regexp'."
281 (not (string-match inferior-lisp-filter-regexp str)))
282
283 ;;;###autoload
284 (defun inferior-lisp (cmd)
285 "Run an inferior Lisp process, input and output via buffer `*inferior-lisp*'.
286 If there is a process already running in `*inferior-lisp*', just switch
287 to that buffer.
288 With argument, allows you to edit the command line (default is value
289 of `inferior-lisp-program'). Runs the hooks from
290 `inferior-lisp-mode-hook' (after the `comint-mode-hook' is run).
291 \(Type \\[describe-mode] in the process buffer for a list of commands.)"
292 (interactive (list (if current-prefix-arg
293 (read-string "Run lisp: " inferior-lisp-program)
294 inferior-lisp-program)))
295 (if (not (comint-check-proc "*inferior-lisp*"))
296 (let ((cmdlist (split-string cmd)))
297 (set-buffer (apply (function make-comint)
298 "inferior-lisp" (car cmdlist) nil (cdr cmdlist)))
299 (inferior-lisp-mode)))
300 (setq inferior-lisp-buffer "*inferior-lisp*")
301 (pop-to-buffer "*inferior-lisp*"))
302 ;;;###autoload (add-hook 'same-window-buffer-names (purecopy "*inferior-lisp*"))
303
304 ;;;###autoload
305 (defalias 'run-lisp 'inferior-lisp)
306
307 (defun lisp-eval-region (start end &optional and-go)
308 "Send the current region to the inferior Lisp process.
309 Prefix argument means switch to the Lisp buffer afterwards."
310 (interactive "r\nP")
311 (comint-send-region (inferior-lisp-proc) start end)
312 (comint-send-string (inferior-lisp-proc) "\n")
313 (if and-go (switch-to-lisp t)))
314
315 (defun lisp-compile-string (string)
316 "Send the string to the inferior Lisp process to be compiled and executed."
317 (comint-send-string
318 (inferior-lisp-proc)
319 (format "(funcall (compile nil (lambda () %s)))\n" string)))
320
321 (defun lisp-eval-string (string)
322 "Send the string to the inferior Lisp process to be executed."
323 (comint-send-string (inferior-lisp-proc) (concat string "\n")))
324
325 (defun lisp-do-defun (do-string do-region)
326 "Send the current defun to the inferior Lisp process.
327 The actually processing is done by `do-string' and `do-region'
328 which determine whether the code is compiled before evaluation.
329 DEFVAR forms reset the variables to the init values."
330 (save-excursion
331 (end-of-defun)
332 (skip-chars-backward " \t\n\r\f") ; Makes allegro happy
333 (let ((end (point)) (case-fold-search t))
334 (beginning-of-defun)
335 (if (looking-at "(defvar")
336 (funcall do-string
337 ;; replace `defvar' with `defparameter'
338 (concat "(defparameter "
339 (buffer-substring-no-properties (+ (point) 7) end)
340 "\n"))
341 (funcall do-region (point) end)))))
342
343 (defun lisp-eval-defun (&optional and-go)
344 "Send the current defun to the inferior Lisp process.
345 DEFVAR forms reset the variables to the init values.
346 Prefix argument means switch to the Lisp buffer afterwards."
347 (interactive "P")
348 (lisp-do-defun 'lisp-eval-string 'lisp-eval-region)
349 (if and-go (switch-to-lisp t)))
350
351 (defun lisp-eval-last-sexp (&optional and-go)
352 "Send the previous sexp to the inferior Lisp process.
353 Prefix argument means switch to the Lisp buffer afterwards."
354 (interactive "P")
355 (lisp-eval-region (save-excursion (backward-sexp) (point)) (point) and-go))
356
357 (defun lisp-compile-region (start end &optional and-go)
358 "Compile the current region in the inferior Lisp process.
359 Prefix argument means switch to the Lisp buffer afterwards."
360 (interactive "r\nP")
361 (lisp-compile-string (buffer-substring-no-properties start end))
362 (if and-go (switch-to-lisp t)))
363
364 (defun lisp-compile-defun (&optional and-go)
365 "Compile the current defun in the inferior Lisp process.
366 DEFVAR forms reset the variables to the init values.
367 Prefix argument means switch to the Lisp buffer afterwards."
368 (interactive "P")
369 (lisp-do-defun 'lisp-compile-string 'lisp-compile-region)
370 (if and-go (switch-to-lisp t)))
371
372 (defun switch-to-lisp (eob-p)
373 "Switch to the inferior Lisp process buffer.
374 With argument, positions cursor at end of buffer."
375 (interactive "P")
376 (if (get-buffer-process inferior-lisp-buffer)
377 (let ((pop-up-frames
378 ;; Be willing to use another frame
379 ;; that already has the window in it.
380 (or pop-up-frames
381 (get-buffer-window inferior-lisp-buffer t))))
382 (pop-to-buffer inferior-lisp-buffer))
383 (run-lisp inferior-lisp-program))
384 (when eob-p
385 (push-mark)
386 (goto-char (point-max))))
387
388
389 ;;; Now that lisp-compile/eval-defun/region takes an optional prefix arg,
390 ;;; these commands are redundant. But they are kept around for the user
391 ;;; to bind if he wishes, for backwards functionality, and because it's
392 ;;; easier to type C-c e than C-u C-c C-e.
393
394 (defun lisp-eval-region-and-go (start end)
395 "Send the current region to the inferior Lisp, and switch to its buffer."
396 (interactive "r")
397 (lisp-eval-region start end t))
398
399 (defun lisp-eval-defun-and-go ()
400 "Send the current defun to the inferior Lisp, and switch to its buffer."
401 (interactive)
402 (lisp-eval-defun t))
403
404 (defun lisp-compile-region-and-go (start end)
405 "Compile the current region in the inferior Lisp, and switch to its buffer."
406 (interactive "r")
407 (lisp-compile-region start end t))
408
409 (defun lisp-compile-defun-and-go ()
410 "Compile the current defun in the inferior Lisp, and switch to its buffer."
411 (interactive)
412 (lisp-compile-defun t))
413
414 ;;; A version of the form in H. Shevis' soar-mode.el package. Less robust.
415 ;;; (defun lisp-compile-sexp (start end)
416 ;;; "Compile the s-expression bounded by START and END in the inferior lisp.
417 ;;; If the sexp isn't a DEFUN form, it is evaluated instead."
418 ;;; (cond ((looking-at "(defun\\s +")
419 ;;; (goto-char (match-end 0))
420 ;;; (let ((name-start (point)))
421 ;;; (forward-sexp 1)
422 ;;; (process-send-string "inferior-lisp"
423 ;;; (format "(compile '%s #'(lambda "
424 ;;; (buffer-substring name-start
425 ;;; (point)))))
426 ;;; (let ((body-start (point)))
427 ;;; (goto-char start) (forward-sexp 1) ; Can't use end-of-defun.
428 ;;; (process-send-region "inferior-lisp"
429 ;;; (buffer-substring body-start (point))))
430 ;;; (process-send-string "inferior-lisp" ")\n"))
431 ;;; (t (lisp-eval-region start end)))))
432 ;;;
433 ;;; (defun lisp-compile-region (start end)
434 ;;; "Each s-expression in the current region is compiled (if a DEFUN)
435 ;;; or evaluated (if not) in the inferior lisp."
436 ;;; (interactive "r")
437 ;;; (save-excursion
438 ;;; (goto-char start) (end-of-defun) (beginning-of-defun) ; error check
439 ;;; (if (< (point) start) (error "region begins in middle of defun"))
440 ;;; (goto-char start)
441 ;;; (let ((s start))
442 ;;; (end-of-defun)
443 ;;; (while (<= (point) end) ; Zip through
444 ;;; (lisp-compile-sexp s (point)) ; compiling up defun-sized chunks.
445 ;;; (setq s (point))
446 ;;; (end-of-defun))
447 ;;; (if (< s end) (lisp-compile-sexp s end)))))
448 ;;;
449 ;;; End of HS-style code
450
451
452 (defvar lisp-prev-l/c-dir/file nil
453 "Record last directory and file used in loading or compiling.
454 This holds a cons cell of the form `(DIRECTORY . FILE)'
455 describing the last `lisp-load-file' or `lisp-compile-file' command.")
456
457 (defcustom lisp-source-modes '(lisp-mode)
458 "*Used to determine if a buffer contains Lisp source code.
459 If it's loaded into a buffer that is in one of these major modes, it's
460 considered a Lisp source file by `lisp-load-file' and `lisp-compile-file'.
461 Used by these commands to determine defaults."
462 :type '(repeat symbol)
463 :group 'inferior-lisp)
464
465 (defun lisp-load-file (file-name)
466 "Load a Lisp file into the inferior Lisp process."
467 (interactive (comint-get-source "Load Lisp file: " lisp-prev-l/c-dir/file
468 lisp-source-modes nil)) ; nil because LOAD
469 ; doesn't need an exact name
470 (comint-check-source file-name) ; Check to see if buffer needs saved.
471 (setq lisp-prev-l/c-dir/file (cons (file-name-directory file-name)
472 (file-name-nondirectory file-name)))
473 (comint-send-string (inferior-lisp-proc)
474 (format inferior-lisp-load-command file-name))
475 (switch-to-lisp t))
476
477
478 (defun lisp-compile-file (file-name)
479 "Compile a Lisp file in the inferior Lisp process."
480 (interactive (comint-get-source "Compile Lisp file: " lisp-prev-l/c-dir/file
481 lisp-source-modes nil)) ; nil = don't need
482 ; suffix .lisp
483 (comint-check-source file-name) ; Check to see if buffer needs saved.
484 (setq lisp-prev-l/c-dir/file (cons (file-name-directory file-name)
485 (file-name-nondirectory file-name)))
486 (comint-send-string (inferior-lisp-proc) (concat "(compile-file \""
487 file-name
488 "\"\)\n"))
489 (switch-to-lisp t))
490
491
492 \f
493 ;;; Documentation functions: function doc, var doc, arglist, and
494 ;;; describe symbol.
495 ;;; ===========================================================================
496
497 ;;; Command strings
498 ;;; ===============
499
500 (defvar lisp-function-doc-command
501 "(let ((fn '%s))
502 (format t \"Documentation for ~a:~&~a\"
503 fn (documentation fn 'function))
504 (values))\n"
505 "Command to query inferior Lisp for a function's documentation.")
506
507 (defvar lisp-var-doc-command
508 "(let ((v '%s))
509 (format t \"Documentation for ~a:~&~a\"
510 v (documentation v 'variable))
511 (values))\n"
512 "Command to query inferior Lisp for a variable's documentation.")
513
514 (defvar lisp-arglist-command
515 "(let ((fn '%s))
516 (format t \"Arglist for ~a: ~a\" fn (arglist fn))
517 (values))\n"
518 "Command to query inferior Lisp for a function's arglist.")
519
520 (defvar lisp-describe-sym-command
521 "(describe '%s)\n"
522 "Command to query inferior Lisp for a variable's documentation.")
523
524
525 ;;; Ancillary functions
526 ;;; ===================
527
528 ;;; Reads a string from the user.
529 (defun lisp-symprompt (prompt default)
530 (list (let* ((prompt (if default
531 (format "%s (default %s): " prompt default)
532 (concat prompt ": ")))
533 (ans (read-string prompt)))
534 (if (zerop (length ans)) default ans))))
535
536
537 ;;; Adapted from function-called-at-point in help.el.
538 (defun lisp-fn-called-at-pt ()
539 "Returns the name of the function called in the current call.
540 The value is nil if it can't find one."
541 (condition-case nil
542 (save-excursion
543 (save-restriction
544 (narrow-to-region (max (point-min) (- (point) 1000)) (point-max))
545 (backward-up-list 1)
546 (forward-char 1)
547 (let ((obj (read (current-buffer))))
548 (and (symbolp obj) obj))))
549 (error nil)))
550
551
552 ;;; Adapted from variable-at-point in help.el.
553 (defun lisp-var-at-pt ()
554 (condition-case ()
555 (save-excursion
556 (forward-sexp -1)
557 (skip-chars-forward "'")
558 (let ((obj (read (current-buffer))))
559 (and (symbolp obj) obj)))
560 (error nil)))
561
562
563 ;;; Documentation functions: fn and var doc, arglist, and symbol describe.
564 ;;; ======================================================================
565
566 (defun lisp-show-function-documentation (fn)
567 "Send a command to the inferior Lisp to give documentation for function FN.
568 See variable `lisp-function-doc-command'."
569 (interactive (lisp-symprompt "Function doc" (lisp-fn-called-at-pt)))
570 (comint-proc-query (inferior-lisp-proc)
571 (format lisp-function-doc-command fn)))
572
573 (defun lisp-show-variable-documentation (var)
574 "Send a command to the inferior Lisp to give documentation for function FN.
575 See variable `lisp-var-doc-command'."
576 (interactive (lisp-symprompt "Variable doc" (lisp-var-at-pt)))
577 (comint-proc-query (inferior-lisp-proc) (format lisp-var-doc-command var)))
578
579 (defun lisp-show-arglist (fn)
580 "Send a query to the inferior Lisp for the arglist for function FN.
581 See variable `lisp-arglist-command'."
582 (interactive (lisp-symprompt "Arglist" (lisp-fn-called-at-pt)))
583 (comint-proc-query (inferior-lisp-proc) (format lisp-arglist-command fn)))
584
585 (defun lisp-describe-sym (sym)
586 "Send a command to the inferior Lisp to describe symbol SYM.
587 See variable `lisp-describe-sym-command'."
588 (interactive (lisp-symprompt "Describe" (lisp-var-at-pt)))
589 (comint-proc-query (inferior-lisp-proc)
590 (format lisp-describe-sym-command sym)))
591
592 \f
593 ;; "Returns the current inferior Lisp process.
594 ;; See variable `inferior-lisp-buffer'."
595 (defun inferior-lisp-proc ()
596 (let ((proc (get-buffer-process (if (derived-mode-p 'inferior-lisp-mode)
597 (current-buffer)
598 inferior-lisp-buffer))))
599 (or proc
600 (error "No Lisp subprocess; see variable `inferior-lisp-buffer'"))))
601
602
603 ;;; Do the user's customisation...
604 ;;;===============================
605 (defvar inferior-lisp-load-hook nil
606 "This hook is run when the library `inf-lisp' is loaded.")
607
608 (run-hooks 'inferior-lisp-load-hook)
609
610 ;;; CHANGE LOG
611 ;;; ===========================================================================
612 ;;; 7/21/92 Jim Blandy
613 ;;; - Changed all uses of the cmulisp name or prefix to inferior-lisp;
614 ;;; this is now the official inferior lisp package. Use the global
615 ;;; ChangeLog from now on.
616 ;;; 5/24/90 Olin
617 ;;; - Split cmulisp and cmushell modes into separate files.
618 ;;; Not only is this a good idea, it's apparently the way it'll be rel 19.
619 ;;; - Upgraded process sends to use comint-send-string instead of
620 ;;; process-send-string.
621 ;;; - Explicit references to process "cmulisp" have been replaced with
622 ;;; (cmulisp-proc). This allows better handling of multiple process bufs.
623 ;;; - Added process query and var/function/symbol documentation
624 ;;; commands. Based on code written by Douglas Roberts.
625 ;;; - Added lisp-eval-last-sexp, bound to C-x C-e.
626 ;;;
627 ;;; 9/20/90 Olin
628 ;;; Added a save-restriction to lisp-fn-called-at-pt. This bug and fix
629 ;;; reported by Lennart Staflin.
630 ;;;
631 ;;; 3/12/90 Olin
632 ;;; - lisp-load-file and lisp-compile-file no longer switch-to-lisp.
633 ;;; Tale suggested this.
634 ;;; - Reversed this decision 7/15/91. You need the visual feedback.
635 ;;;
636 ;;; 7/25/91 Olin
637 ;;; Changed all keybindings of the form C-c <letter>. These are
638 ;;; supposed to be reserved for the user to bind. This affected
639 ;;; mainly the compile/eval-defun/region[-and-go] commands.
640 ;;; This was painful, but necessary to adhere to the gnumacs standard.
641 ;;; For some backwards compatibility, see the
642 ;;; cmulisp-install-letter-bindings
643 ;;; function.
644 ;;;
645 ;;; 8/2/91 Olin
646 ;;; - The lisp-compile/eval-defun/region commands now take a prefix arg,
647 ;;; which means switch-to-lisp after sending the text to the Lisp process.
648 ;;; This obsoletes all the -and-go commands. The -and-go commands are
649 ;;; kept around for historical reasons, and because the user can bind
650 ;;; them to key sequences shorter than C-u C-c C-<letter>.
651 ;;; - If M-x cmulisp is invoked with a prefix arg, it allows you to
652 ;;; edit the command line.
653
654 (provide 'inf-lisp)
655
656 ;;; inf-lisp.el ends here