scratch/edwin/autold.scm

240 lines
7.3 KiB
Scheme
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#| -*-Scheme-*-
Copyright (C) 1986, 1987, 1988, 1989, 1990, 1991, 1992, 1993, 1994,
1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016,
2017, 2018, 2019, 2020 Massachusetts Institute of Technology
This file is part of MIT/GNU Scheme.
MIT/GNU Scheme is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or (at
your option) any later version.
MIT/GNU Scheme is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with MIT/GNU Scheme; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301,
USA.
|#
;;;; Autoloads for Edwin
;;;; Definitions
(define (make-autoloading-procedure library-name get-procedure)
(letrec ((apply-hook
(make-apply-hook
(lambda arguments
((ref-command load-library) library-name 'NO-WARN)
(let ((procedure (get-procedure)))
(set-apply-hook-procedure! apply-hook procedure)
(apply procedure arguments)))
(cons autoloading-procedure-tag library-name))))
apply-hook))
(define autoloading-procedure-tag "autoloading-procedure-tag")
(define (autoloading-procedure? object)
(and (apply-hook? object)
(eq? autoloading-procedure-tag (car (apply-hook-extra object)))))
(define-integrable (autoloading-procedure/library-name procedure)
(cdr (apply-hook-extra procedure)))
(define (define-autoload-procedure name package library-name)
(let ((environment (->environment package)))
(environment-define environment
name
(make-autoloading-procedure
library-name
(lambda () (environment-lookup environment name))))))
(define (define-autoload-major-mode name super-mode-name display-name
library-name description)
(define mode
(make-mode name #t display-name
(and super-mode-name (->mode super-mode-name))
description
(make-autoloading-procedure library-name
(lambda ()
(mode-initialization mode)))))
(environment-define (->environment '(EDWIN))
(mode-name->scheme-name name)
mode)
name)
(define (define-autoload-minor-mode name display-name library-name description)
(define mode
(make-mode name #f display-name #f description
(make-autoloading-procedure library-name
(lambda ()
(mode-initialization mode)))))
(environment-define (->environment '(EDWIN))
(mode-name->scheme-name name)
mode)
name)
(define (autoloading-mode? mode)
(autoloading-procedure? (mode-initialization mode)))
(define (define-autoload-command name library-name description)
(define command
(make-command name description '()
(make-autoloading-procedure library-name
(lambda ()
(command-procedure command)))))
(environment-define (->environment '(EDWIN))
(command-name->scheme-name name)
command)
name)
(define (autoloading-command? command)
(autoloading-procedure? (command-procedure command)))
(define (guarantee-command-loaded command)
(let ((procedure (command-procedure command)))
(if (autoloading-procedure? procedure)
((ref-command load-library)
(autoloading-procedure/library-name procedure)
'NO-WARN))))
;;;; Libraries
(define known-libraries
'())
(define (define-library name . entries)
(let ((entry (assq name known-libraries)))
(if entry
(set-cdr! entry entries)
(set! known-libraries
(cons (cons name entries)
known-libraries))))
name)
(define loaded-libraries
'())
(define (library-loaded? name)
(memq name loaded-libraries))
(define library-load-hooks
'())
(define (add-library-load-hook! name hook)
(if (library-loaded? name)
(hook)
(let ((entry (assq name library-load-hooks)))
(if entry
(append! entry (list hook))
(set! library-load-hooks
(cons (list name hook)
library-load-hooks))))))
(define (run-library-load-hooks! name)
(let ((entry (assq name library-load-hooks)))
(define (loop)
(if (null? (cdr entry))
(set! library-load-hooks (delq! entry library-load-hooks))
(let ((hook (cadr entry)))
(set-cdr! entry (cddr entry))
(hook)
(loop))))
(if entry (loop))))
(define (load-library library)
(for-each (lambda (entry)
(let ((file (car entry))
(environment (->environment (cadr entry)))
(purify? (if (pair? (cddr entry)) (caddr entry) #t)))
(cond ((built-in-object-file
(merge-pathnames file (pathname-as-directory "edwin")))
=> (lambda (obj)
(if purify? (purify obj))
(scode-eval obj environment)))
(else
(load (merge-pathnames file (edwin-binary-directory))
environment
'DEFAULT
purify?)))))
(cdr library))
(if (not (memq (car library) loaded-libraries))
(set! loaded-libraries
(cons (car library) loaded-libraries)))
(run-library-load-hooks! (car library)))
;;;; Loading
(define-command load-library
"Load the Edwin library NAME.
Second arg FORCE? controls what happens if the library is already loaded:
'NO-WARN means do nothing,
#f means display a warning message in the minibuffer,
anything else means load it anyway.
Second arg is prefix arg when called interactively."
(lambda ()
(list
(prompt-for-alist-value "Load library"
(map (lambda (library)
(cons (symbol->string (car library))
(car library)))
known-libraries))
(command-argument)))
(lambda (name force?)
(load-edwin-library name force? #t)))
(define (load-edwin-library name #!optional force? interactive?)
(let ((force? (if (default-object? force?) #f force?))
(interactive? (if (default-object? interactive?) #f interactive?)))
(let ((do-it
(lambda ()
(let ((library (assq name known-libraries)))
(if (not library)
(error "Unknown library name:" name))
(if interactive?
(with-output-to-transcript-buffer
(lambda ()
(bind-condition-handler (list condition-type:error)
evaluation-error-handler
(lambda ()
(parameterize
((param:suppress-loading-message? #t))
((message-wrapper #f "Loading " (car library))
(lambda ()
(load-library library))))))))
(load-library library))))))
(cond ((not (library-loaded? name))
(do-it))
((not force?)
(if interactive? (message "Library already loaded: " name)))
((not (eq? force? 'NO-WARN))
(do-it))))))
(define-command load-file
"Load the Edwin binary file FILENAME.
Second arg PURIFY? means purify the file's contents after loading;
this is the prefix arg when called interactively."
"fLoad file\nP"
(lambda (filename purify?)
((message-wrapper #f "Loading " filename)
(lambda ()
(load-edwin-file filename '(EDWIN) purify?)))))
(define (load-edwin-file filename environment purify?)
(with-output-to-transcript-buffer
(lambda ()
(bind-condition-handler (list condition-type:error)
evaluation-error-handler
(lambda ()
(parameterize ((param:suppress-loading-message? #t))
(load filename environment 'DEFAULT purify?)))))))