1 ;; copyright 2003-2005 stefan kersten <steve@k-hornz.de>
3 ;; This program is free software; you can redistribute it and/or
4 ;; modify it under the terms of the GNU General Public License as
5 ;; published by the Free Software Foundation; either version 2 of the
6 ;; License, or (at your option) any later version.
8 ;; This program is distributed in the hope that it will be useful, but
9 ;; WITHOUT ANY WARRANTY; without even the implied warranty of
10 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 ;; General Public License for more details.
13 ;; You should have received a copy of the GNU General Public License
14 ;; along with this program; if not, write to the Free Software
15 ;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
20 (require 'sclang-util
)
24 ;; =====================================================================
26 ;; =====================================================================
28 ;; FIXME: everything will fail when renaming the post buffer!
30 (defconst sclang-post-buffer
(sclang-make-buffer-name "PostBuffer")
31 "Name of the SuperCollider process output buffer.")
33 (defconst sclang-bullet-latin-1
(string-to-char (decode-coding-string "\xa5" 'utf-8
))
34 "Character for highlighting errors (latin-1).")
36 (defconst sclang-bullet-utf-8
(string-to-char (decode-coding-string "\xe2\x80\xa2" 'utf-8
))
37 "Character for highlighting errors (utf-8).")
39 (defconst sclang-parse-error-regexp
40 "^\\(WARNING\\|ERROR\\): .*\n[\t ]*in file '\\([^']\+\\)'\n[\t ]*line \\([0-9]\+\\) char \\([0-9]\+\\)"
41 "Regular expression matching parse errors during library compilation.")
43 (defcustom sclang-max-post-buffer-size
0
44 "*Maximum number of characters to insert in post buffer.
46 :group
'sclang-interface
50 (defcustom sclang-auto-scroll-post-buffer nil
51 "*Automatically scroll post buffer on output regardless of point position.
52 Default behavior is to only scroll when point is not at end of buffer."
53 :group
'sclang-interface
57 (defun sclang-get-post-buffer ()
58 (get-buffer-create sclang-post-buffer
))
60 (defmacro with-sclang-post-buffer
(&rest body
)
61 `(with-current-buffer (sclang-get-post-buffer)
64 ;; (defun sclang-post-string (string)
65 ;; (with-sclang-post-buffer
66 ;; (let ((eobp (mapcar (lambda (w)
67 ;; (cons w (= (window-point w) (point-max))))
68 ;; (get-buffer-window-list (current-buffer) nil t))))
70 ;; ;; insert STRING into process buffer
71 ;; (goto-char (point-max))
73 ;; (dolist (assoc eobp)
75 ;; (save-selected-window
76 ;; (let ((window (car assoc)))
77 ;; (select-window window)
78 ;; (set-window-point window (point-max))
79 ;; (recenter -1))))))))
81 ;; (defun sclang-post-string (string &optional proc)
82 ;; (let* ((buffer (process-buffer proc))
83 ;; (window (display-buffer buffer)))
84 ;; (with-current-buffer buffer
85 ;; (let ((moving (= (point) (process-mark proc))))
87 ;; ;; Insert the text, advancing the process marker.
88 ;; (goto-char (process-mark proc))
90 ;; (set-marker (process-mark proc) (point)))
92 ;; (goto-char (process-mark proc))
93 ;; (set-window-point window (process-mark proc)))))))
95 (defun sclang-show-post-buffer (&optional eob-p
)
96 "Show SuperCollider process buffer.
97 If EOB-P is non-nil, positions cursor at end of buffer."
99 (with-sclang-post-buffer
100 (let ((window (display-buffer (current-buffer))))
102 (goto-char (point-max))
103 (save-selected-window
104 (set-window-point window
(point-max)))))))
106 (defun sclang-clear-post-buffer ()
107 "Clear the output buffer."
109 (with-sclang-post-buffer (erase-buffer)))
111 (defun sclang-init-post-buffer ()
112 "Initialize post buffer."
113 (get-buffer-create sclang-post-buffer
)
114 (with-sclang-post-buffer
117 (set (make-local-variable 'font-lock-fontify-region-function
)
118 (lambda (&rest args
)))
119 ;; setup compilation mode
120 (compilation-minor-mode)
121 (set (make-variable-buffer-local 'compilation-error-screen-columns
) nil
)
122 (set (make-variable-buffer-local 'compilation-error-regexp-alist
)
123 (cons (list sclang-parse-error-regexp
2 3 4) compilation-error-regexp-alist
))
124 (set (make-variable-buffer-local 'compilation-parse-errors-function
)
125 (lambda (limit-search find-at-least
)
126 (compilation-parse-errors limit-search find-at-least
)))
127 (set (make-variable-buffer-local 'compilation-parse-errors-filename-function
)
130 (sclang-clear-post-buffer)
131 (sclang-show-post-buffer))
133 ;; =====================================================================
134 ;; interpreter interface
135 ;; =====================================================================
137 (defconst sclang-process
"SCLang"
138 "Name of the SuperCollider interpreter subprocess.")
140 (defcustom sclang-program
"sclang"
141 "*Name of the SuperCollider interpreter program."
142 :group
'sclang-programs
146 (defcustom sclang-runtime-directory
""
147 "*Path to the SuperCollider runtime directory."
148 :group
'sclang-options
151 :options
'(:must-match
))
153 (defcustom sclang-library-configuration-file
""
154 "*Path of the library configuration file."
155 :group
'sclang-options
158 :options
'(:must-match
))
160 (defcustom sclang-heap-size
""
161 "*Initial heap size."
162 :group
'sclang-options
166 (defcustom sclang-heap-growth
""
168 :group
'sclang-options
172 (defcustom sclang-udp-port -
1
173 "*UDP listening port."
174 :group
'sclang-options
178 (defcustom sclang-main-run nil
179 "*Call Main.run on startup."
180 :group
'sclang-options
184 (defcustom sclang-main-stop nil
185 "*Call Main.stop on shutdown."
186 :group
'sclang-options
190 ;; =====================================================================
192 ;; =====================================================================
194 (defun sclang-get-process ()
195 (get-process sclang-process
))
197 ;; =====================================================================
198 ;; library startup/shutdown
199 ;; =====================================================================
201 (defvar sclang-library-initialized-p nil
)
203 (defcustom sclang-library-startup-hook nil
204 "*Hook run after initialization of the SCLang process."
205 :group
'sclang-interface
208 (defcustom sclang-library-shutdown-hook nil
209 "*Hook run before deletion of the SCLang process."
210 :group
'sclang-interface
213 ;; library initialization works like this:
215 ;; * emacs starts sclang with SCLANG_COMMAND_FIFO set in the environment
216 ;; * sclang opens fifo for communication with emacs during class tree
218 ;; * sclang sends '_init' command
219 ;; * '_init' command handler calls sclang-on-library-startup to complete
222 (defun sclang-library-initialized-p ()
223 (and (sclang-get-process)
224 sclang-library-initialized-p
))
226 (defun sclang-on-library-startup ()
227 (sclang-message "Initializing library...")
228 (setq sclang-library-initialized-p t
)
229 (run-hooks 'sclang-library-startup-hook
)
230 (sclang-message "Initializing library...done"))
232 (defun sclang-on-library-shutdown ()
233 (when sclang-library-initialized-p
234 (run-hooks 'sclang-library-shutdown-hook
)
235 (setq sclang-library-initialized-p nil
)
236 (sclang-message "Shutting down library...")))
238 ;; =====================================================================
240 ;; =====================================================================
242 (defun sclang-process-sentinel (proc msg
)
243 (with-sclang-post-buffer
244 (goto-char (point-max))
246 (if (and (bolp) (eolp)) "\n" "\n\n")
247 (format "*** %s %s ***" proc
(substring msg
0 -
1))
249 (when (memq (process-status proc
) '(exit signal
))
250 (sclang-on-library-shutdown)
251 (sclang-stop-command-process)))
253 (defun sclang-process-filter (process string
)
254 (let ((buffer (process-buffer process
)))
255 (with-current-buffer buffer
256 (when (and (> sclang-max-post-buffer-size
0)
257 (> (buffer-size) sclang-max-post-buffer-size
))
259 (let ((move-point (or sclang-auto-scroll-post-buffer
260 (= (point) (process-mark process
)))))
262 ;; replace mac-roman bullet with unicode character
263 (subst-char-in-string sclang-bullet-latin-1 sclang-bullet-utf-8 string t
)
264 ;; insert the text, advancing the process marker.
265 (goto-char (process-mark process
))
267 (set-marker (process-mark process
) (point)))
269 (goto-char (process-mark process
))
272 (when (eq buffer
(window-buffer window
))
273 (set-window-point window
(process-mark process
))))
276 ;; =====================================================================
277 ;; process startup/shutdown
278 ;; =====================================================================
280 (defun sclang-memory-option-p (string)
281 (let ((case-fold-search nil
))
282 (string-match "^[1-9][0-9]*[km]?$" string
)))
284 (defun sclang-port-option-p (number)
285 (and (>= number
0) (<= number
#XFFFF
)))
287 (defun sclang-make-options ()
288 (let ((default-directory "")
290 (flet ((append-option
291 (option &optional value
)
292 (setq res
(append res
(list option
) (and value
(list value
))))))
293 (if (file-directory-p sclang-runtime-directory
)
294 (append-option "-d" (expand-file-name sclang-runtime-directory
)))
295 (if (file-exists-p sclang-library-configuration-file
)
296 (append-option "-l" (expand-file-name sclang-library-configuration-file
)))
297 (if (sclang-memory-option-p sclang-heap-size
)
298 (append-option "-m" sclang-heap-size
))
299 (if (sclang-memory-option-p sclang-heap-growth
)
300 (append-option "-g" sclang-heap-growth
))
301 (if (sclang-port-option-p sclang-udp-port
)
302 (append-option "-u" (number-to-string sclang-udp-port
)))
304 (append-option "-r"))
306 (append-option "-s"))
307 (append-option "-iscel")
310 (defun sclang-start ()
311 "Start SuperCollider process."
314 (sclang-on-library-shutdown)
316 (sclang-init-post-buffer)
317 (sclang-start-command-process)
318 (let ((process-connection-type nil
))
319 (let ((proc (apply 'start-process
320 sclang-process sclang-post-buffer
321 sclang-program
(sclang-make-options))))
322 (set-process-sentinel proc
'sclang-process-sentinel
)
323 (set-process-filter proc
'sclang-process-filter
)
324 (set-process-coding-system proc
'mule-utf-8
'mule-utf-8
)
325 (set-process-query-on-exit-flag proc nil
)
328 (defun sclang-kill ()
329 "Kill SuperCollider process."
331 (when (sclang-get-process)
332 (kill-process sclang-process
)
333 (delete-process sclang-process
)))
335 (defun sclang-stop ()
336 "Stop SuperCollider process."
338 (when (sclang-get-process)
339 (process-send-eof sclang-process
)
342 (while (and (sclang-get-process)
347 (sclang-stop-command-process))
349 (defun sclang-recompile ()
350 "Recompile class library."
352 (when (sclang-get-process)
353 (process-send-string sclang-process
"\x18")
356 ;; =====================================================================
358 ;; =====================================================================
360 (defcustom sclang-mkfifo-program
"mkfifo"
361 "*Name of the \"mkfifo\" program.
363 Change this if \"mkfifo\" has a non-standard name or location."
364 :group
'sclang-programs
367 (defcustom sclang-cat-program
"cat"
368 "*Name of the \"cat\" program.
370 Change this if \"cat\" has a non-standard name or location."
371 :group
'sclang-programs
374 (defconst sclang-command-process
"SCLang Command"
375 "Subprocess for receiving command results from sclang.")
377 (defconst sclang-cmd-helper-proc
"SCLang Command Helper"
378 "Dummy subprocess that will keep the command fifo open for writing
379 so reading does not fail automatically when sclang closes its own
380 writing end of the fifo")
382 (defvar sclang-command-fifo nil
383 "FIFO for communicating with the subprocess.")
385 (defun sclang-delete-command-fifo ()
386 (and sclang-command-fifo
387 (file-exists-p sclang-command-fifo
)
388 (delete-file sclang-command-fifo
)))
390 (defun sclang-release-command-fifo ()
391 (sclang-delete-command-fifo)
392 (setq sclang-command-fifo nil
))
394 (defun sclang-create-command-fifo ()
395 (setq sclang-command-fifo
(make-temp-name
397 "sclang-command-fifo." temporary-file-directory
)))
398 (sclang-delete-command-fifo)
399 (let ((res (call-process sclang-mkfifo-program
401 sclang-command-fifo
)))
403 (message "SCLang: Couldn't create command fifo")
404 (setq sclang-command-fifo nil
))))
406 (defun sclang-start-command-process ()
407 (sclang-create-command-fifo)
408 (when sclang-command-fifo
409 ;; start the dummy process to keep the fifo open
410 (let ((process-connection-type nil
))
411 (let ((proc (start-process-shell-command
412 sclang-cmd-helper-proc nil
413 (concat sclang-cat-program
" > " sclang-command-fifo
))))
414 (set-process-query-on-exit-flag proc nil
)))
415 ;; sclang gets the fifo path via the environment
416 (setenv "SCLANG_COMMAND_FIFO" sclang-command-fifo
)
417 (let ((process-connection-type nil
))
418 (let ((proc (start-process
419 sclang-command-process nil
420 sclang-cat-program sclang-command-fifo
)))
421 (set-process-filter proc
'sclang-command-process-filter
)
422 ;; this is important. use a unibyte stream without eol
423 ;; conversion for communication.
424 (set-process-coding-system proc
'no-conversion
'no-conversion
)
425 (set-process-query-on-exit-flag proc nil
)))
426 (unless (get-process sclang-command-process
)
427 (message "SCLang: Couldn't start command process"))))
429 (defun sclang-stop-command-process ()
430 (when (get-process sclang-cmd-helper-proc
)
431 (kill-process sclang-cmd-helper-proc
)
432 (delete-process sclang-cmd-helper-proc
))
433 ;; the real command process should now quit automatically,
434 ;; since there is no more writers to the command fifo
435 (sclang-release-command-fifo))
437 (defvar sclang-command-process-previous nil
438 "Unprocessed command process output.")
440 (defun sclang-command-process-filter (proc string
)
441 (when sclang-command-process-previous
442 (setq string
(concat sclang-command-process-previous string
)))
444 (while (and (> (length string
) 3)
446 (setq end
(+ 4 (sclang-string-to-int32 string
)))))
447 (sclang-handle-command-result (car (read-from-string string
4 end
)))
448 (setq string
(substring string end
))))
449 (setq sclang-command-process-previous string
))
451 ;; =====================================================================
453 ;; =====================================================================
455 ;; symbol property: sclang-command-handler
457 (defun sclang-set-command-handler (symbol function
)
458 (put symbol
'sclang-command-handler function
))
460 (defun sclang-perform-command (symbol &rest args
)
461 (sclang-eval-string (sclang-format
462 "Emacs.lispPerformCommand(%o, %o, true)"
465 (defun sclang-perform-command-no-result (symbol &rest args
)
466 (sclang-eval-string (sclang-format
467 "Emacs.lispPerformCommand(%o, %o, false)"
470 (defun sclang-default-command-handler (fun arg
)
471 "Default command handler.
472 Displays short message on error."
475 (error (sclang-message "Error in command handler") nil
)))
477 (defun sclang-debug-command-handler (fun arg
)
478 "Debugging command handler.
479 Enters debugger on error."
480 (let ((debug-on-error t
)
484 (defvar sclang-command-handler
'sclang-default-command-handler
485 "Function called when handling command result.")
487 (defun sclang-toggle-debug-command-handler (&optional arg
)
488 "Toggle debugging of command handler.
489 With arg, activate debugging iff arg is positive."
491 (setq sclang-command-handler
492 (if (or (and arg
(> arg
0))
493 (eq sclang-command-handler
'sclang-debug-command-handler
))
494 'sclang-default-command-handler
495 'sclang-default-command-handler
))
496 (sclang-message "Command handler debugging %s."
497 (if (eq sclang-command-handler
'sclang-debug-command-handler
)
501 (defun sclang-handle-command-result (list)
503 (let ((fun (get (nth 0 list
) 'sclang-command-handler
))
506 (when (functionp fun
)
507 (let ((res (funcall sclang-command-handler fun arg
)))
510 (sclang-format "Emacs.lispHandleCommandResult(%o, %o)" id res
))))))
513 ;; =====================================================================
515 ;; =====================================================================
517 (defconst sclang-token-interpret-cmd-line
(char-to-string #X1b
))
518 (defconst sclang-token-interpret-print-cmd-line
(char-to-string #X0c
))
520 (defcustom sclang-eval-line-forward t
521 "*If non-nil `sclang-eval-line' advances to the next line."
522 :group
'sclang-interface
525 (defun sclang-send-string (token string
&optional force
)
526 (let ((proc (sclang-get-process)))
527 (when (and proc
(or (sclang-library-initialized-p) force
))
528 (process-send-string proc
(concat string token
))
531 (defun sclang-eval-string (string &optional print-p
)
532 "Send STRING to the sclang process for evaluation and print the result
533 if PRINT-P is non-nil. Return STRING if successful, otherwise nil."
535 (if print-p sclang-token-interpret-print-cmd-line sclang-token-interpret-cmd-line
)
538 (defun sclang-eval-expression (string &optional silent-p
)
539 "Execute STRING as SuperCollider code."
540 (interactive "sEval: \nP")
541 (sclang-eval-string string
(not silent-p
)))
543 (defun sclang-eval-line (&optional silent-p
)
544 "Execute the current line as SuperCollider code."
546 (let ((string (sclang-line-at-point)))
548 (sclang-eval-string string
(not silent-p
)))
549 (and sclang-eval-line-forward
550 (/= (line-end-position) (point-max))
554 (defun sclang-eval-region (&optional silent-p
)
555 "Execute the region as SuperCollider code."
558 (buffer-substring-no-properties (region-beginning) (region-end))
561 (defun sclang-eval-region-or-line (&optional silent-p
)
563 (if (and transient-mark-mode mark-active
)
564 (sclang-eval-region silent-p
)
565 (sclang-eval-line silent-p
)))
567 (defun sclang-eval-defun (&optional silent-p
)
569 (let ((string (sclang-defun-at-point)))
570 (when (and string
(string-match "^(" string
))
571 (sclang-eval-string string
(not silent-p
))
574 (defun sclang-eval-document (&optional silent-p
)
575 "Execute the whole document as SuperCollider code."
580 (buffer-substring-no-properties (region-beginning) (region-end))
583 (defvar sclang-eval-results nil
584 "Save results of sync SCLang evaluation.")
586 (sclang-set-command-handler
588 (lambda (arg) (push arg sclang-eval-results
)))
590 (defun sclang-eval-sync (string)
591 "Eval STRING in sclang and return result as a lisp value."
592 (let ((proc (get-process sclang-command-process
)))
593 (if (and (processp proc
) (eq (process-status proc
) 'run
))
594 (let ((time (current-time)) (tick 10000) elt
)
595 (sclang-perform-command 'evalSCLang string time
)
596 (while (and (> (decf tick
) 0)
597 (not (setq elt
(find time sclang-eval-results
598 :key
#'car
:test
#'equal
))))
599 (accept-process-output proc
0 100))
601 (prog1 (if (eq (nth 1 elt
) 'ok
)
603 (setq sclang-eval-results
(delq elt sclang-eval-results
))
604 (signal 'sclang-error
(nth 2 elt
)))
605 (setq sclang-eval-results
(delq elt sclang-eval-results
)))
606 (error "SCLang sync eval timeout")))
607 (error "SCLang Command process not running"))))
609 ;; =====================================================================
611 ;; =====================================================================
613 ;; (defun sclang-help-file-paths ()
614 ;; "Return a list of help file paths."
617 ;; (defun sclang-grep-help-files ()
619 ;; (let ((sclang-grep-prompt "Search help files: ")
620 ;; (sclang-grep-files (mapcar 'cdr sclang-help-topic-alist)))
621 ;; (call-interactively 'sclang-grep-files)))
623 ;; (defvar sclang-grep-history nil)
625 ;; (defcustom sclang-grep-case-fold-search t
626 ;; "*Non-nil if sclang-grep-files should ignore case."
627 ;; :group 'sclang-interface
631 ;; (defvar sclang-grep-files nil)
632 ;; (defvar sclang-grep-prompt "Grep: ")
634 ;; (defun sclang-grep-files (regexp)
636 ;; (let ((grep-default (or (when current-prefix-arg (sclang-symbol-at-point))
637 ;; (car sclang-grep-history))))
638 ;; (list (read-from-minibuffer sclang-grep-prompt
640 ;; nil nil 'sclang-grep-history))))
641 ;; (grep-compute-defaults)
642 ;; (grep (concat grep-program
644 ;; (and sclang-grep-case-fold-search " -i")
646 ;; " " (mapconcat 'shell-quote-argument sclang-grep-files " "))))
648 ;; =====================================================================
650 ;; =====================================================================
652 (defcustom sclang-show-workspace-on-startup t
653 "*If non-nil show the workspace buffer on library startup."
654 :group
'sclang-interface
657 (defconst sclang-workspace-buffer
(sclang-make-buffer-name "Workspace"))
659 (defun sclang-fill-workspace-mode-map (map)
660 (define-key map
"\C-c}" 'bury-buffer
))
662 (defun sclang-switch-to-workspace ()
664 (let ((buffer (get-buffer sclang-workspace-buffer
)))
666 (setq buffer
(get-buffer-create sclang-workspace-buffer
))
667 (with-current-buffer buffer
669 (let ((map (make-sparse-keymap)))
670 (set-keymap-parent map sclang-mode-map
)
671 (sclang-fill-workspace-mode-map map
)
673 (let ((line (concat "// " (make-string 69 ?
=) "\n")))
675 (insert "// SuperCollider Workspace\n")
677 ;; (insert "// using HTML Help: C-c C-h as usual, then switch to w3m buffer\n")
678 ;; (insert "// and do M-x sclang-minor-mode in order te enable sclang code execution\n")
681 (set-buffer-modified-p nil
)
682 ;; cwd to sclang-runtime-directory
683 (if (and sclang-runtime-directory
684 (file-directory-p sclang-runtime-directory
))
685 (setq default-directory sclang-runtime-directory
))))
686 (switch-to-buffer buffer
)))
688 (add-hook 'sclang-library-startup-hook
689 (lambda () (and sclang-show-workspace-on-startup
690 (sclang-switch-to-workspace))))
692 ;; =====================================================================
694 ;; =====================================================================
696 (defun sclang-main-run ()
698 (sclang-eval-string "thisProcess.run"))
700 (defun sclang-main-stop ()
702 (sclang-eval-string "thisProcess.stop"))
704 ;; =====================================================================
705 ;; default command handlers
706 ;; =====================================================================
708 (sclang-set-command-handler '_init
(lambda (arg) (sclang-on-library-startup)))
710 (sclang-set-command-handler
714 (eval (read expr
)))))
716 ;; =====================================================================
718 ;; =====================================================================
720 ;; shutdown process cleanly
721 (add-hook 'kill-emacs-hook
(lambda () (sclang-stop)))
723 ;; add command line switches
724 (add-to-list 'command-switch-alist
729 (add-to-list 'command-switch-alist
730 (cons "-sclang-debug"
732 (sclang-toggle-debug-command-handler 1))))
734 (add-to-list 'command-switch-alist
738 (when command-line-args-left
739 (let ((file (pop command-line-args-left
)))
740 (with-current-buffer (get-buffer-create sclang-workspace-buffer
)
741 (and (file-exists-p file
) (insert-file-contents file
))
742 (set-buffer-modified-p nil
)
744 (switch-to-buffer (current-buffer))))))))
746 (provide 'sclang-interp
)