Newer
Older
;;; -*- Log: code.log; Package: Lisp -*-
;;;
;;; **********************************************************************
;;; This code was written as part of the CMU Common Lisp project at
;;; Carnegie Mellon University, and has been placed in the public domain.
;;;
(ext:file-comment
"$Header: src/code/macros.lisp $")
;;; **********************************************************************
;;;
;;; This file contains the macros that are part of the standard
;;; Spice Lisp environment.
;;;
;;; Written by Scott Fahlman and Rob MacLachlan.
;;; Modified by Bill Chiles to adhere to the wall.
(intl:textdomain "cmucl")
(export '(defvar defparameter defconstant when unless setf
defsetf psetf shiftf rotatef push pushnew pop
incf decf remf case typecase with-open-file
with-open-stream with-input-from-string with-output-to-string
locally etypecase ctypecase ecase ccase
get-setf-expansion define-setf-expander
define-compiler-macro))
(in-package "EXTENSIONS")
(export '(do-anonymous collect iterate))
(in-package "LISP")
;;; Parse-Body -- Public
;;;
;;; Parse out declarations and doc strings, *not* expanding macros.
;;; Eventually the environment arg should be flushed, since macros can't expand
;;; into declarations anymore.
;;;
(defun parse-body (body environment &optional (doc-string-allowed t))
"This function is to parse the declarations and doc-string out of the body of
a defun-like form. Body is the list of stuff which is to be parsed.
Environment is ignored. If Doc-String-Allowed is true, then a doc string
will be parsed out of the body and returned. If it is false then a string
will terminate the search for declarations. Three values are returned: the
tail of Body after the declarations and doc strings, a list of declare forms,
and the doc-string, or NIL if none."
(declare (ignore environment))
(let ((decls ())
(doc nil))
(do ((tail body (cdr tail)))
((endp tail)
(values tail (nreverse decls) doc))
(let ((form (car tail)))
(cond ((and (stringp form) (cdr tail))
(if doc-string-allowed
(setq doc form
;; Only one doc string is allowed.
doc-string-allowed nil)
(return (values tail (nreverse decls) doc))))
((not (and (consp form) (symbolp (car form))))
(return (values tail (nreverse decls) doc)))
((eq (car form) 'declare)
(push form decls))
(t
(return (values tail (nreverse decls) doc))))))))
;;;; DEFMACRO:
;;; Defmacro -- Public
;;;
;;; Parse the definition and make an expander function. The actual
;;; definition is done by %defmacro which we expand into.
;;;
(defmacro defmacro (name lambda-list &body body)
(when lisp::*enable-package-locked-errors*
(multiple-value-bind (valid block-name)
(ext:valid-function-name-p name)
(declare (ignore valid))
(let ((package (symbol-package block-name)))
(when package
(when (ext:package-definition-lock package)
(restart-case
(error 'lisp::package-locked-error
:format-control (intl:gettext "defining macro ~A")
:report (lambda (stream)
(write-string (intl:gettext "Ignore the lock and continue") stream)))
:report (lambda (stream)
(write-string (intl:gettext "Disable the package's definition-lock then continue") stream))
(setf (ext:package-definition-lock package) nil))
(unlock-all ()
:report (lambda (stream)
(write-string (intl:gettext "Unlock all packages, then continue") stream))
(lisp::unlock-all-packages))))))))
(let ((whole (gensym "WHOLE-"))
(environment (gensym "ENV-")))
(parse-defmacro lambda-list whole body name 'defmacro
:environment environment)
(when doc
(intl::note-translatable intl::*default-domain* doc))
(let ((def `(lambda (,whole ,environment)
,@local-decs
(block ,name
,body))))
`(progn
(eval-when (:compile-toplevel)
(c::do-macro-compile-time ',name #',def))
(eval-when (:load-toplevel :execute)
(c::%defmacro ',name #',def ',lambda-list ,doc)))))))
;;; %Defmacro, %%Defmacro -- Internal
;;;
;;; Defmacro expands into %Defmacro which is a function that is treated
;;; magically the compiler. After the compiler has gotten the information it
;;; wants out of macro definition, it compiles a call to %%Defmacro which
;;; happens at load time. We have a %Defmacro function which just calls
;;; %%Defmacro in order to keep the interpreter happy.
;;;
;;; Eventually %%Defmacro should deal with clearing old compiler information
;;; for the functional value.
;;;
(defun c::%defmacro (name definition lambda-list doc)
(assert (eval:interpreted-function-p definition))
(setf (eval:interpreted-function-arglist definition) lambda-list)
(c::%%defmacro name definition doc))
;;;
(defun c::%%defmacro (name definition doc)
(clear-info function where-from name)
(setf (macro-function name) definition)
;;;; DEFINE-COMPILER-MACRO
(defmacro define-compiler-macro (name lambda-list &body body)
"Define a compiler-macro for NAME."
(let ((whole (gensym "WHOLE-"))
(environment (gensym "ENV-")))
(multiple-value-bind (validp block-name)
(valid-function-name-p name)
(unless validp
(simple-program-error (intl:gettext "~S is not a valid function name.") name))
(multiple-value-bind
(body local-decs doc)
(parse-defmacro lambda-list whole body name 'define-compiler-macro
:environment environment)
(when doc
(intl::note-translatable intl::*default-domain* doc))
(let ((def `(lambda (,whole ,environment)
,@local-decs
(block ,block-name
,body))))
`(progn
(eval-when (:compile-toplevel)
(c::do-compiler-macro-compile-time ',name #',def))
(eval-when (:load-toplevel :execute)
(c::%define-compiler-macro ',name #',def ',lambda-list ,doc))))))))
(defun c::%define-compiler-macro (name definition lambda-list doc)
(assert (eval:interpreted-function-p definition))
(setf (eval:interpreted-function-name definition)
(let ((*print-case* :upcase))
(format nil "DEFINE-COMPILER-MACRO ~S" name)))
(setf (eval:interpreted-function-arglist definition) lambda-list)
(c::%%define-compiler-macro name definition doc))
;;;
(defun c::%%define-compiler-macro (name definition doc)
(setf (compiler-macro-function name) definition)
(setf (documentation name 'compiler-macro) doc)
name)
;;;; DEFINE-SYMBOL-MACRO
;;; define-symbol-macro -- Public
;;;
(defmacro define-symbol-macro (name expansion)
(%define-symbol-macro ',name ',expansion (c::source-location))))
(defun %define-symbol-macro (name expansion source-location)
(unless (symbolp name)
(error 'simple-type-error :datum name :expected-type 'symbol
:format-control (intl:gettext "Symbol macro name is not a symbol: ~S.")
:format-arguments (list name)))
(ecase (info variable kind name)
((:macro :global nil)
(setf (info variable kind name) :macro)
(setf (info variable macro-expansion name) expansion)
(set-defvar-source-location name source-location))
:format-control (intl:gettext "Symbol macro name already declared special: ~S.")
:format-arguments (list name)))
(:constant
(error 'simple-program-error
:format-control (intl:gettext "Symbol macro name already declared constant: ~S.")
:format-arguments (list name))))
name)
;;; DEFTYPE is a lot like DEFMACRO.
(defmacro deftype (name arglist &body body)
"Syntax like DEFMACRO, but defines a new type."
(simple-program-error (intl:gettext "~S -- Type name not a symbol.") name))
(and lisp::*enable-package-locked-errors*
(symbol-package name)
(ext:package-definition-lock (symbol-package name))
(restart-case
(error 'lisp::package-locked-error
:format-control (intl:gettext "defining type ~A")
:report (lambda (stream)
(write-string (intl:gettext "Ignore the lock and continue") stream)))
:report (lambda (stream)
(write-string (intl:gettext "Disable package's definition-lock then continue") stream))
(setf (ext:package-definition-lock (symbol-package name)) nil))
(unlock-all ()
:report (lambda (stream)
(write-string (intl:gettext "Unlock all packages, then continue") stream))
(lisp::unlock-all-packages))))
(let ((whole (gensym "WHOLE-")))
(parse-defmacro arglist whole body name 'deftype
:default-default ''*)
(when doc
(intl::note-translatable intl::*default-domain* doc))
(set-defvar-source-location ',name (c::source-location))
(%deftype ',name
#'(lambda (,whole)
,@local-decs
(block ,name ,body))
,@(when doc `(,doc)))))))
;;;
(defun %deftype (name expander &optional doc)
(when (info declaration recognized name)
(error (intl:gettext "Deftype already names a declaration: ~S.") name))
(ecase (info type kind name)
(:primitive
(error (intl:gettext "Illegal to redefine standard type: ~S.") name)))
(warn (intl:gettext "Redefining class ~S to be a DEFTYPE.") name)
(undefine-structure (layout-info (%class-layout (kernel::find-class name))))
(setf (class-cell-class (find-class-cell name)) nil)
(setf (info type compiler-layout name) nil)
(setf (info type kind name) :defined))
(:defined)
((nil)
(setf (info type kind name) :defined)))
(setf (info type expander name) expander)
(when doc
(setf (documentation name 'type) doc))
;; ### Bootstrap hack -- we need to define types before %note-type-defined
;; is defined.
(when (fboundp 'c::%note-type-defined)
;;; And so is DEFINE-SETF-EXPANDER.
(defparameter defsetf-error-string "Setf expander for ~S cannot be called with ~S args.")
(defmacro define-setf-expander (access-fn lambda-list &body body)
"Syntax like DEFMACRO, but creates a Setf-Expansion generator. The body
must be a form that returns the five magical values."
(unless (symbolp access-fn)
(simple-program-error (intl:gettext "~S -- Access-function name not a symbol in DEFINE-SETF-EXPANDER.")
(let ((whole (gensym "WHOLE-"))
(environment (gensym "ENV-")))
(multiple-value-bind (body local-decs doc)
(parse-defmacro lambda-list whole body access-fn
:environment environment)
(when doc
(intl::note-translatable intl::*default-domain* doc))
(%define-setf-macro
',access-fn
#'(lambda (,whole ,environment)
,@local-decs
(block ,access-fn ,body))
nil
',doc)))))
(defmacro define-setf-method (&rest stuff)
"Obsolete, use define-setf-expander."
`(define-setf-expander ,@stuff))
;;; %DEFINE-SETF-MACRO -- Internal
;;;
;;; Do stuff for defining a setf macro.
;;;
(defun %define-setf-macro (name expander inverse doc)
(cond ((not (fboundp `(setf ,name))))
((info function accessor-for name)
(warn (intl:gettext "Defining setf macro for destruct slot accessor; redefining as ~
a normal function:~% ~S")
name)
(c::define-function-name name))
((not (eq (symbol-package name) (symbol-package 'aref)))
(warn (intl:gettext "Defining setf macro for ~S, but ~S is fbound.")
name `(setf ,name))))
(when (or inverse (info setf inverse name))
(setf (info setf inverse name) inverse))
(when (or expander (info setf expander name))
(setf (info setf expander name) expander))
(when doc
(setf (documentation name 'setf) doc))
name)
;;;; Destructuring-bind
(defmacro destructuring-bind (lambda-list arg-list &rest body)
"Bind the variables in LAMBDA-LIST to the contents of ARG-LIST."
(let* ((arg-list-name (gensym "ARG-LIST-")))
(multiple-value-bind
(parse-defmacro lambda-list arg-list-name body nil 'destructuring-bind
:annonymousp t :doc-string-allowed nil)
`(let ((,arg-list-name ,arg-list))
,body))))
;;;; Defun, Defvar, Defparameter, Defconstant:
;;; Defun -- Public
;;;
;;; Very similar to Defmacro, but simpler. We don't have to parse the
;;; lambda-list.
;;;
(defmacro defun (&whole source name lambda-list &parse-body (body decls doc))
(multiple-value-bind (valid block-name)
(valid-function-name-p name)
(declare (ignore valid))
(when doc
(intl::note-translatable intl::*default-domain* doc))
(let ((def `(lambda ,lambda-list
,@decls
(block ,block-name ,@body))))
`(c::%defun ',name #',def ,doc ',source))))
;;; %Defun, %%Defun -- Internal
;;;
;;; Similar to %Defmacro, ...
;;;
(defun c::%%defun (name def doc &optional inline-expansion)
(c::define-function-name name)
(setf (fdefinition name) def)
(when doc
(if (and (consp name) (eq (first name) 'setf))
(setf (documentation (second name) 'setf) doc)
(setf (documentation name 'function) doc)))
(when (eq (info function where-from name) :assumed)
(setf (info function where-from name) :defined)
(when (info function assumed-type name)
(setf (info function assumed-type name) nil)))
(when (or inline-expansion
(info function inline-expansion name))
(setf (info function inline-expansion name) inline-expansion))
name)
(defun c::%defun (name def doc source)
(declare (ignore source))
(assert (eval:interpreted-function-p def))
(setf (eval:interpreted-function-name def) name)
(let ((inline-expansion nil))
(when (memq (info function inlinep name) '(:inline :maybe-inline))
(multiple-value-bind (lambda-expression closure-p)
(function-lambda-expression def)
(unless closure-p
(setq inline-expansion lambda-expression))))
(c::%%defun name def doc inline-expansion)))
(defun set-defvar-textdomain (name domain)
(setf (c::info variable textdomain name) domain))
;;; DEFCONSTANT -- Public
;;;
(defmacro defconstant (var val &optional doc)
"For defining global constants at top level. The DEFCONSTANT says that the
value is constant and may be compiled into code. If the variable already has
a value, and this is not equal to the init, an error is signalled. The third
argument is an optional documentation string for the variable."
(when doc
(intl::note-translatable intl::*default-domain* doc))
`(progn
(eval-when (:compile-toplevel)
(c::do-defconstant-compile-time ',var ,val ',doc))
(eval-when (:load-toplevel :execute)
(set-defvar-textdomain ',var ,intl::*default-domain*)
(c::%%defconstant ',var ,val ',doc (c::source-location)))))
(defun set-defvar-source-location (name source-location)
(setf (info :source-location :defvar name) source-location))
;;; %Defconstant, %%Defconstant -- Internal
;;;
;;; Like the other %mumbles except that we currently actually do something
;;; interesting at load time, namely checking if the constant is being
;;; redefined.
;;;
(defun c::%defconstant (name value doc)
(c::%%defconstant name value doc nil))
(defun c::%%defconstant (name value doc source-location)
(when doc
(setf (documentation name 'variable) doc))
(when (boundp name)
(unless (equalp (symbol-value name) value)
(cerror (intl:gettext "Go ahead and change the value.")
(intl:gettext "Constant ~S being redefined.") name)))
(setf (symbol-value name) value)
(setf (info variable kind name) :constant)
(clear-info variable constant-value name)
(set-defvar-source-location name source-location)
name)
(defmacro defvar (var &optional (val nil valp) (doc nil docp))
"For defining global variables at top level. Declares the variable
SPECIAL and, optionally, initializes it. If the variable already has a
value, the old value is not clobbered. The third argument is an optional
documentation string for the variable."
(when docp
(intl::note-translatable intl::*default-domain* doc))
,@(when valp
`((unless (boundp ',var)
(setq ,var ,val))))
,@(when docp
`((setf (documentation ',var 'variable) ',doc)
(eval-when (:load-toplevel :execute)
(set-defvar-textdomain ',var ,intl::*default-domain*))))
(set-defvar-source-location ',var (c::source-location))
',var))
(defmacro defparameter (var val &optional (doc nil docp))
"Defines a parameter that is not normally changed by the program,
but that may be changed without causing an error. Declares the
variable special and sets its value to VAL. The third argument is
an optional documentation string for the parameter."
(when docp
(intl::note-translatable intl::*default-domain* doc))
`((setf (documentation ',var 'variable) ',doc)
(eval-when (:load-toplevel :execute)
(set-defvar-textdomain ',var ,intl::*default-domain*))))
(set-defvar-source-location ',var (c::source-location))
',var))
;;;; ASSORTED CONTROL STRUCTURES
(defmacro when (test &body forms)
"First arg is a predicate. If it is non-null, the rest of the forms are
evaluated as a PROGN."
`(cond (,test nil ,@forms)))
(defmacro unless (test &rest forms)
"First arg is a predicate. If it is null, the rest of the forms are
evaluated as a PROGN."
`(cond ((not ,test) nil ,@forms)))
(defmacro return (&optional (value nil))
`(return-from nil ,value))
(defmacro prog (varlist &parse-body (body decls))
`(block nil
(let ,varlist
,@decls
(tagbody ,@body))))
(defmacro prog* (varlist &parse-body (body decls))
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
`(block nil
(let* ,varlist
,@decls
(tagbody ,@body))))
;;; Prog1, Prog2 -- Public
;;;
;;; These just turn into a Let.
;;;
(defmacro prog1 (result &rest body)
(let ((n-result (gensym)))
`(let ((,n-result ,result))
,@body
,n-result)))
;;;
(defmacro prog2 (form1 result &rest body)
`(prog1 (progn ,form1 ,result) ,@body))
;;; And, Or -- Public
;;;
;;; AND and OR are defined in terms of IF.
;;;
(defmacro and (&rest forms)
(cond ((endp forms) t)
((endp (rest forms)) (first forms))
(t
`(if ,(first forms)
(and ,@(rest forms))
nil))))
;;;
(defmacro or (&rest forms)
(cond ((endp forms) nil)
((endp (rest forms)) (first forms))
(t
(let ((n-result (gensym)))
`(let ((,n-result ,(first forms)))
(if ,n-result
,n-result
(or ,@(rest forms))))))))
;;; Cond -- Public
;;;
;;; COND also turns into IF.
;;;
(defmacro cond (&rest clauses)
(if (endp clauses)
nil
(let ((clause (first clauses)))
(when (atom clause)
(error (intl:gettext "Cond clause should be a non-empty list: ~S.") clause))
(let ((test (first clause))
(forms (rest clause)))
(if (endp forms)
(let ((n-result (gensym)))
`(let ((,n-result ,test))
(if ,n-result
,n-result
(cond ,@(rest clauses)))))
`(if ,test
(progn ,@forms)
(cond ,@(rest clauses))))))))
;;;; Multiple value macros:
;;; Multiple-Value-XXX -- Public
;;;
;;; All the multiple-value receiving forms are defined in terms of
;;; Multiple-Value-Call.
;;;
(defmacro multiple-value-setq (varlist value-form)
(unless (and (listp varlist) (every #'symbolp varlist))
(simple-program-error (intl:gettext "Varlist is not a list of symbols: ~S.") varlist))
(if varlist
`(values (setf (values ,@varlist) ,value-form))
`(values ,value-form)))
;;;
(defmacro multiple-value-bind (varlist value-form &body body)
(unless (and (listp varlist) (every #'symbolp varlist))
(simple-program-error (intl:gettext "Varlist is not a list of symbols: ~S.") varlist))
(if (= (length varlist) 1)
`(let ((,(car varlist) ,value-form))
,@body)
(let ((ignore (gensym)))
`(multiple-value-call #'(lambda (&optional ,@(mapcar #'list varlist) &rest ,ignore)
(declare (ignore ,ignore))
,@body)
,value-form))))
;;;
(defmacro multiple-value-list (value-form)
`(multiple-value-call #'list ,value-form))
"Evaluates FORM and returns the Nth value (zero based). This involves no
consing when N is a trivial constant integer."
(if (integerp n)
(let ((dummy-list nil)
(keeper (gensym "KEEPER-")))
;; We build DUMMY-LIST, a list of variables to bind to useless
;; values, then we explicitly IGNORE those bindings and return
;; KEEPER, the only thing we're really interested in right now.
(dotimes (i n)
(push (gensym "IGNORE-") dummy-list))
`(multiple-value-bind (,@dummy-list ,keeper)
,form
(declare (ignore ,@dummy-list))
,keeper))
(once-only ((n n))
`(case (the (values fixnum &rest t) ,n)
(0 (nth-value 0 ,form))
(1 (nth-value 1 ,form))
(2 (nth-value 2 ,form))
(T (nth (the (values fixnum &rest t) ,n)
(multiple-value-list ,form)))))))
;;;; SETF and friends.
;;; Note: The expansions for SETF and friends sometimes create needless
;;; LET-bindings of argument values. The compiler will remove most of
;;; these spurious bindings, so SETF doesn't worry too much about creating
;;; them.
;;; The inverse for a generalized-variable reference function is stored in
;;; one of two ways:
;;;
;;; A SETF inverse property corresponds to the short form of DEFSETF. It is
;;; the name of a function takes the same args as the reference form, plus a
;;; new-value arg at the end.
;;;
;;; A SETF method expander is created by the long form of DEFSETF or
;;; by DEFINE-SETF-EXPANDER. It is a function that is called on the reference
;;; form and that produces five values: a list of temporary variables, a list
;;; of value forms, a list of the single store-value form, a storing function,
;;; and an accessing function.
(defun get-setf-expansion (form &optional environment)
"Returns five values needed by the SETF machinery: a list of temporary
variables, a list of values with which to fill them, a list of temporaries
for the new values, the setting function, and the accessing function."
(multiple-value-bind
(expansion expanded)
(macroexpand-1 form environment)
(if expanded
(get-setf-expansion expansion environment)
(let ((new-var (gensym)))
(values nil nil (list new-var)
`(setq ,form ,new-var) form)))))
;; Local functions inhibit global setf methods...
((and environment
(let ((name (car form)))
(dolist (x (c::lexenv-functions environment) nil)
(when (and (eq (car x) name)
(not (c::defined-function-p (cdr x))))
(return t)))))
(expand-or-get-setf-inverse form environment))
(get-setf-method-inverse form `(,temp) nil))
((setq temp (info setf expander (car form)))
(funcall temp form environment))
(t
(expand-or-get-setf-inverse form environment)))))
(defun get-setf-method-multiple-value (form &optional env)
"Obsolete: use GET-SETF-EXPANSION."
(get-setf-expansion form env))
;;;
;;; If a macro, expand one level and try again. If not, go for the
;;; SETF function.
(defun expand-or-get-setf-inverse (form environment)
(multiple-value-bind
(expansion expanded)
(macroexpand-1 form environment)
(if expanded
(get-setf-expansion expansion environment)
(get-setf-method-inverse form `(funcall #'(setf ,(car form)))
t))))
(defun get-setf-method-inverse (form inverse setf-function)
(let ((new-var (gensym))
(vars nil)
(vals nil))
(dolist (x (cdr form))
(push (gensym) vars)
(push x vals))
(setq vals (nreverse vals))
(values vars vals (list new-var)
(if setf-function
`(,@inverse ,new-var ,@vars)
`(,@inverse ,@vars ,new-var))
(defun get-setf-method (form &optional environment)
"Obsolete: use GET-SETF-EXPANSION and handle multiple store values."
(multiple-value-bind
(temps value-forms store-vars store-form access-form)
(get-setf-expansion form environment)
(error (intl:gettext "GET-SETF-METHOD used for a form with multiple store ~
variables:~% ~S") form))
(values temps value-forms store-vars store-form access-form)))
(let ((arglist (car rest))
(arglist-var (gensym "ARGS-"))
(new-var (car (cadr rest))))
(multiple-value-bind
(body local-decs doc)
(parse-defmacro arglist arglist-var (cddr rest) fn 'defsetf)
(when doc
(intl::note-translatable intl::*default-domain* doc))
`(lambda (,arglist-var ,new-var)
,@local-decs
(defmacro defsetf (access-fn &rest rest)
"Associates a SETF update function or macro with the specified access
function or macro. The format is complex. See the manual for
details."
(cond ((not (listp (car rest)))
`(eval-when (load compile eval)
(%define-setf-macro ',access-fn nil ',(car rest)
,(when (and (car rest) (stringp (cadr rest)))
`',(cadr rest)))))
((and (cdr rest) (listp (cadr rest)))
(destructuring-bind
(lambda-list (&rest store-variables) &body body)
rest
(let ((arglist-var (gensym "ARGS-"))
(access-form-var (gensym "ACCESS-FORM-"))
(env-var (gensym "ENVIRONMENT-")))
(multiple-value-bind
(body local-decs doc)
(parse-defmacro `(,lambda-list ,@store-variables)
arglist-var body access-fn 'defsetf
:annonymousp t)
(when doc
(intl::note-translatable intl::*default-domain* doc))
`(eval-when (load compile eval)
(%define-setf-macro
',access-fn
#'(lambda (,access-form-var ,env-var)
(declare (ignore ,env-var))
(%defsetf ,access-form-var ,(length store-variables)
#'(lambda (,arglist-var)
,@local-decs
(block ,access-fn
,body))))
nil
',doc))))))
(t
(error (intl:gettext "Ill-formed DEFSETF for ~S.") access-fn))))
(defun %defsetf (orig-access-form num-store-vars expander)
(collect ((subforms) (subform-vars) (subform-exprs) (store-vars))
(dolist (subform (cdr orig-access-form))
(if (constantp subform)
(subforms subform)
(let ((var (gensym)))
(subforms var)
(subform-vars var)
(subform-exprs subform))))
(dotimes (i num-store-vars)
(store-vars (gensym)))
(values (subform-vars)
(subform-exprs)
(store-vars)
(funcall expander (cons (subforms) (store-vars)))
`(,(car orig-access-form) ,@(subforms)))))
;;; SETF -- Public
;;;
;;; Except for atoms, we always call GET-SETF-METHOD, since it has some
;;; non-trivial semantics. But when there is a setf inverse, and G-S-M uses
;;; it, then we return a call to the inverse, rather than returning a hairy let
;;; form. This is probably important mainly as a convenince in allowing the
;;; use of setf inverses without the full interpreter.
;;;
"Takes pairs of arguments like SETQ. The first is a place and the second
is the value that is supposed to go into that place. Returns the last
value. The place argument may be any of the access forms for which SETF
knows a corresponding setting form."
(let ((nargs (length args)))
(cond
((= nargs 2)
(let ((place (first args))
(value-form (second args)))
(if (atom place)
`(setq ,place ,value-form)
(multiple-value-bind (dummies vals newval setter getter)
(get-setf-expansion place env)
(declare (ignore getter))
(let ((inverse (info setf inverse (car place))))
(if (and inverse (eq inverse (car setter)))
`(,inverse ,@(cdr place) ,value-form)
`(let* (,@(mapcar #'list dummies vals))
(multiple-value-bind ,newval ,value-form
,setter))))))))
((oddp nargs)
(error (intl:gettext "Odd number of args to SETF.")))
(t
(do ((a args (cddr a)) (l nil))
((null a) `(progn ,@(nreverse l)))
(setq l (cons (list 'setf (car a) (cadr a)) l)))))))
"This is to SETF as PSETQ is to SETQ. Args are alternating place
expressions and values to go into those places. All of the subforms and
values are determined, left to right, and only then are the locations
updated. Returns NIL."
(collect ((let*-bindings) (mv-bindings) (setters))
(do ((a args (cddr a)))
((endp a))
(if (endp (cdr a))
(simple-program-error (intl:gettext "Odd number of args to PSETF.")))
(multiple-value-bind
(dummies vals newval setter getter)
(get-setf-expansion (car a) env)
(declare (ignore getter))
(let*-bindings (mapcar #'list dummies vals))
(mv-bindings (list newval (cadr a)))
(setters setter)))
(labels ((thunk (let*-bindings mv-bindings)
(if let*-bindings
`(let* ,(car let*-bindings)
(multiple-value-bind ,@(car mv-bindings)
,(thunk (cdr let*-bindings) (cdr mv-bindings))))
`(progn ,@(setters) nil))))
(thunk (let*-bindings) (mv-bindings)))))
(defmacro shiftf (&rest args &environment env)
"One or more SETF-style place expressions, followed by a single
value expression. Evaluates all of the expressions in turn, then
assigns the value of each expression to the place on its left,
returning the value of the leftmost."
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
(when args
(collect ((let*-bindings) (mv-bindings) (setters) (getters))
;; The last arg isn't necessarily a place, so we have to handle
;; that separately.
(dolist (arg (butlast args))
(multiple-value-bind
(temps subforms store-vars setter getter)
(get-setf-expansion arg env)
(loop
for temp in temps
for subform in subforms
do (let*-bindings `(,temp ,subform)))
(mv-bindings store-vars)
(setters setter)
(getters getter)))
;; Handle the last arg specially here. Just put something to
;; force the setter so the setter for the previous var gets set,
;; and the getter is just the last arg itself.
(setters nil)
(getters (car (last args)))
(labels ((thunk (mv-bindings getters)
(if mv-bindings
`((multiple-value-bind
,(car mv-bindings)
,(car getters)
,@(thunk (cdr mv-bindings) (cdr getters))))
`(,@(butlast (setters))))))
`(let* ,(let*-bindings)
(multiple-value-bind ,(car (mv-bindings))
,(car (getters))
,@(thunk (mv-bindings) (cdr (getters)))
(values ,@(car (mv-bindings)))))))))
"Takes any number of SETF-style place expressions. Evaluates all of the
expressions in turn, then assigns to each place the value of the form to
its right. The rightmost form gets the value of the leftmost.
Returns NIL."
(when args
(collect ((let*-bindings) (mv-bindings) (setters) (getters))
(dolist (arg args)
(multiple-value-bind
(temps subforms store-vars setter getter)
(get-setf-expansion arg env)
(loop
for temp in temps
for subform in subforms
do (let*-bindings `(,temp ,subform)))
(mv-bindings store-vars)
(setters setter)
(getters getter)))
(setters nil)
(getters (car (getters)))
(labels ((thunk (mv-bindings getters)
(if mv-bindings
`((multiple-value-bind
,(car mv-bindings)
,(car getters)
,@(thunk (cdr mv-bindings) (cdr getters))))
(setters))))
`(let* ,(let*-bindings)
,@(thunk (mv-bindings) (cdr (getters))))))))
(defmacro define-modify-macro (name lambda-list function &optional doc-string)
"Creates a new read-modify-write macro like PUSH or INCF."
(env (gensym "ENV-"))
(reference (gensym "PLACE-")))
;; Parse out the variable names and rest arg from the lambda list.
(do ((ll lambda-list (cdr ll))
(arg nil))
((null ll))
(setq arg (car ll))
(cond ((eq arg '&optional))
((eq arg '&rest)
(if (symbolp (cadr ll))
(setq rest-arg (cadr ll))
(error (intl:gettext "Non-symbol &rest arg in definition of ~S.") name))
(error (intl:gettext "Illegal stuff after &rest arg in Define-Modify-Macro."))))
(error (intl:gettext "~S not allowed in Define-Modify-Macro lambda list.") arg))
((symbolp arg)
(push arg other-args))
((and (listp arg) (symbolp (car arg)))
(push (car arg) other-args))
(t (error (intl:gettext "Illegal stuff in lambda list of Define-Modify-Macro.")))))
(setq other-args (nreverse other-args))
`(defmacro ,name (,reference ,@lambda-list &environment ,env)
,doc-string
(multiple-value-bind (dummies vals newval setter getter)
(get-setf-method ,reference ,env)
(do ((d dummies (cdr d))
(v vals (cdr v))
(let-list nil (cons (list (car d) (car v)) let-list)))
((null d)
(push
(list (car newval)
,(if rest-arg
`(list* ',function getter ,@other-args ,rest-arg)
`(list ',function getter ,@other-args)))
let-list)
`(let* ,(nreverse let-list)
,setter)))))))
(defmacro push (obj place &environment env)
"Takes an object and a location holding a list. Conses the object onto
the list, returning the modified list. OBJ is evaluated before PLACE."
;; This special case for place being a symbol isn't strictly needed.
;; It's so we can do push (and pushnew) with a kernel.core.
(if (and (symbolp place)
(eq place (macroexpand place env)))
`(setq ,place (cons ,obj ,place))
(multiple-value-bind (dummies vals newval setter getter)
(get-setf-expansion place env)
(cond
((cdr newval)
;; Handle multiple values
(let ((g (mapcar #'(lambda (x)
(declare (ignore x))
(gensym))
(rest obj))))
`(multiple-value-bind ,g
,obj
(let* (,@(mapcar #'list dummies vals))
(multiple-value-bind ,newval
(values ,@(mapcar #'(lambda (a b)
(list 'cons a b))
g (rest getter)))
,setter)))))
(t
;; A single value
(let ((g (gensym)))