Skip to content
macros.lisp 71.3 KiB
Newer Older
ram's avatar
ram committed
;;; -*- 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 $")
ram's avatar
ram committed
;;; **********************************************************************
;;;
;;; 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.
ram's avatar
ram committed
;;;
(in-package "LISP")
(export '(defvar defparameter defconstant when unless setf
	  defsetf psetf shiftf rotatef push pushnew pop
ram's avatar
ram committed
	  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
ram's avatar
ram committed
          define-modify-macro destructuring-bind nth-value
wlott's avatar
wlott committed
          otherwise ; Sacred to CASE and related macros.
ram's avatar
ram committed

(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
ram's avatar
ram committed
  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
pw's avatar
pw committed
		   (setq doc form
			 ;; Only one doc string is allowed.
			 doc-string-allowed nil)
ram's avatar
ram committed
		   (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*
emarsden's avatar
 
emarsden committed
    (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
emarsden's avatar
 
emarsden committed
                       :package package
                       :format-control (intl:gettext "defining macro ~A")
emarsden's avatar
 
emarsden committed
                       :format-arguments (list name))
              (continue ()
                :report (lambda (stream)
			  (write-string (intl:gettext "Ignore the lock and continue") stream)))
emarsden's avatar
 
emarsden committed
              (unlock-package ()
                :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-")))
ram's avatar
ram committed
    (multiple-value-bind
	(body local-decs doc)
	(parse-defmacro lambda-list whole body name 'defmacro
			:environment environment)
      (when doc
	(intl::note-translatable intl::*default-domain* doc))
ram's avatar
ram committed
      (let ((def `(lambda (,whole ,environment)
		    ,@local-decs
		    (block ,name
		      ,body))))
pw's avatar
pw committed
	`(progn
	   (eval-when (:compile-toplevel)
	     (c::do-macro-compile-time ',name #',def))
	   (eval-when (:load-toplevel :execute)
	     (c::%defmacro ',name #',def ',lambda-list ,doc)))))))
ram's avatar
ram committed


;;; %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))
emarsden's avatar
 
emarsden committed
  (setf (eval:interpreted-function-name definition) name)
  (setf (eval:interpreted-function-arglist definition) lambda-list)
ram's avatar
ram committed
  (c::%%defmacro name definition doc))
;;;
(defun c::%%defmacro (name definition doc)
  (clear-info function where-from name)
  (setf (macro-function name) definition)
ram's avatar
ram committed
  (setf (documentation name 'function) doc)
  name)

wlott's avatar
wlott committed


;;;; DEFINE-COMPILER-MACRO

(defmacro define-compiler-macro (name lambda-list &body body)
  "Define a compiler-macro for NAME."
wlott's avatar
wlott committed
  (let ((whole (gensym "WHOLE-"))
	(environment (gensym "ENV-")))
rtoy's avatar
rtoy committed
    (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))))))))
pw's avatar
pw committed

wlott's avatar
wlott committed

(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)
toy's avatar
toy committed
  `(eval-when (:compile-toplevel :load-toplevel :execute)
     (%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))
    (:special
     (error 'simple-program-error
	    :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)
    
ram's avatar
ram committed

;;; DEFTYPE is a lot like DEFMACRO.

(defmacro deftype (name arglist &body body)
  "Syntax like DEFMACRO, but defines a new type."
ram's avatar
ram committed
  (unless (symbolp name)
    (simple-program-error (intl:gettext "~S -- Type name not a symbol.") name))
  (and lisp::*enable-package-locked-errors*
emarsden's avatar
 
emarsden committed
       (symbol-package name)
       (ext:package-definition-lock (symbol-package name))
       (restart-case
           (error 'lisp::package-locked-error
emarsden's avatar
 
emarsden committed
                  :package (symbol-package name)
                  :format-control (intl:gettext "defining type ~A")
emarsden's avatar
 
emarsden committed
                  :format-arguments (list name))
         (continue ()
           :report (lambda (stream)
		     (write-string (intl:gettext "Ignore the lock and continue") stream)))
emarsden's avatar
 
emarsden committed
         (unlock-package ()
           :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-")))
ram's avatar
ram committed
    (multiple-value-bind (body local-decs doc)
			 (parse-defmacro arglist whole body name 'deftype
					 :default-default ''*)
      (when doc
	(intl::note-translatable intl::*default-domain* doc))
toy's avatar
toy committed
      `(eval-when (:compile-toplevel :load-toplevel :execute)
	 (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
     (when *type-system-initialized*
       (error (intl:gettext "Illegal to redefine standard type: ~S.") name)))
    (:instance
     (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)
    (c::%note-type-defined name))
ram's avatar
ram committed

;;; And so is DEFINE-SETF-EXPANDER.
ram's avatar
ram committed

(defparameter defsetf-error-string "Setf expander for ~S cannot be called with ~S args.")
ram's avatar
ram committed

(defmacro define-setf-expander (access-fn lambda-list &body body)
  "Syntax like DEFMACRO, but creates a Setf-Expansion generator.  The body
ram's avatar
ram committed
  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.")
ram's avatar
ram committed
	   access-fn))

  (let ((whole (gensym "WHOLE-"))
	(environment (gensym "ENV-")))
ram's avatar
ram committed
    (multiple-value-bind (body local-decs doc)
			 (parse-defmacro lambda-list whole body access-fn
dtc's avatar
dtc committed
					 'define-setf-expander
      (when doc
	(intl::note-translatable intl::*default-domain* doc))
toy's avatar
toy committed
      `(eval-when (:compile-toplevel :load-toplevel :execute)
	 (%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
	(body local-decls)
	(parse-defmacro lambda-list arg-list-name body nil 'destructuring-bind
			:annonymousp t :doc-string-allowed nil)
      `(let ((,arg-list-name ,arg-list))
	 ,@local-decls
ram's avatar
ram committed


;;;; 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))
gerd's avatar
gerd committed
  (multiple-value-bind (valid block-name)
      (valid-function-name-p name)
    (declare (ignore valid))
    (when doc
      (intl::note-translatable intl::*default-domain* doc))
gerd's avatar
gerd committed
    (let ((def `(lambda ,lambda-list
		  ,@decls
		  (block ,block-name ,@body))))
      `(c::%defun ',name #',def ,doc ',source))))
ram's avatar
ram committed


;;; %Defun, %%Defun  --  Internal
;;;
;;;    Similar to %Defmacro, ...
;;;
(defun c::%%defun (name def doc &optional inline-expansion)
  (c::define-function-name name)
ram's avatar
ram committed
  (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)))
ram's avatar
ram committed
  (when (or inline-expansion
	    (info function inline-expansion name))
    (setf (info function inline-expansion name) inline-expansion))
  name)
ram's avatar
ram committed
(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)))
ram's avatar
ram committed

(defun set-defvar-textdomain (name domain)
  (setf (c::info variable textdomain name) domain))

ram's avatar
ram committed
;;; DEFCONSTANT  --  Public
;;;
(defmacro defconstant (var val &optional doc)
  "For defining global constants at top level.  The DEFCONSTANT says that the
ram's avatar
ram committed
  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))
ram's avatar
ram committed

;;; %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))
ram's avatar
ram committed
;;;
(defun c::%%defconstant (name value doc source-location)
ram's avatar
ram committed
  (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)))
ram's avatar
ram committed
  (setf (symbol-value name) value)
  (setf (info variable kind name) :constant)
  (clear-info variable constant-value name)
  (set-defvar-source-location name source-location)
ram's avatar
ram committed
  name)

(defmacro defvar (var &optional (val nil valp) (doc nil docp))
  "For defining global variables at top level.  Declares the variable
ram's avatar
ram committed
  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))
ram's avatar
ram committed
  `(progn
    (declaim (special ,var))
ram's avatar
ram committed
     ,@(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))
ram's avatar
ram committed
    ',var))

(defmacro defparameter (var val &optional (doc nil docp))
  "Defines a parameter that is not normally changed by the program,
ram's avatar
ram committed
  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))
ram's avatar
ram committed
  `(progn
    (declaim (special ,var))
ram's avatar
ram committed
    (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))
ram's avatar
ram committed
    ',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
ram's avatar
ram committed
  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
ram's avatar
ram committed
  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))
ram's avatar
ram committed
  `(block nil
     (let ,varlist
       ,@decls
       (tagbody ,@body))))

(defmacro prog* (varlist &parse-body (body decls))
ram's avatar
ram committed
  `(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))
ram's avatar
ram committed
	(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)))
ram's avatar
ram committed
;;;
(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))
ram's avatar
ram committed
  (if (= (length varlist) 1)
      `(let ((,(car varlist) ,value-form))
	 ,@body)
      (let ((ignore (gensym)))
rtoy's avatar
rtoy committed
	`(multiple-value-call #'(lambda (&optional ,@(mapcar #'list varlist) &rest ,ignore)
ram's avatar
ram committed
				  (declare (ignore ,ignore))
				  ,@body)
	   ,value-form))))
;;;
(defmacro multiple-value-list (value-form)
  `(multiple-value-call #'list ,value-form))

ram's avatar
ram committed

(defmacro nth-value (n form)
  "Evaluates FORM and returns the Nth value (zero based).  This involves no
ram's avatar
ram committed
  consing when N is a trivial constant integer."
  (if (integerp n)
      (let ((dummy-list nil)
        ;; 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.
          (push (gensym "IGNORE-") dummy-list))
        `(multiple-value-bind (,@dummy-list ,keeper)
                              ,form
           (declare (ignore ,@dummy-list))
	`(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)))))))
ram's avatar
ram committed

ram's avatar
ram committed

;;;; 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
ram's avatar
ram committed
;;; 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
dtc's avatar
dtc committed
;;; by DEFINE-SETF-EXPANDER.  It is a function that is called on the reference
ram's avatar
ram committed
;;; 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."
ram's avatar
ram committed
  (let (temp)
    (cond ((symbolp form)
	   (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)))))
ram's avatar
ram committed
	  ;;
	  ;; 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))
ram's avatar
ram committed
	  ((setq temp (info setf inverse (car form)))
	   (get-setf-method-inverse form `(,temp) nil))
ram's avatar
ram committed
	  ((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))))

ram's avatar
ram committed

(defun get-setf-method-inverse (form inverse setf-function)
ram's avatar
ram committed
  (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))
ram's avatar
ram committed
	    `(,(car form) ,@vars))))


(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)
    (when (cdr store-vars)
      (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)))

ram's avatar
ram committed

(defun defsetter (fn rest)
  (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))
ram's avatar
ram committed
      (values 
       `(lambda (,arglist-var ,new-var)
	  ,@local-decs
wlott's avatar
wlott committed
       doc))))
ram's avatar
ram committed


(defmacro defsetf (access-fn &rest rest)
  "Associates a SETF update function or macro with the specified access
ram's avatar
ram committed
  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)))))
ram's avatar
ram committed

;;; 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.
;;;
ram's avatar
ram committed
(defmacro setf (&rest args &environment env)
  "Takes pairs of arguments like SETQ.  The first is a place and the second
ram's avatar
ram committed
  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))))))))
      (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)))))))
ram's avatar
ram committed

(defmacro psetf (&rest args &environment env)
  "This is to SETF as PSETQ is to SETQ.  Args are alternating place
ram's avatar
ram committed
  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)))))
ram's avatar
ram committed

(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."
  (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)))))))))
ram's avatar
ram committed

(defmacro rotatef (&rest args &environment env)
  "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))))))))
ram's avatar
ram committed


(defmacro define-modify-macro (name lambda-list function &optional doc-string)
  "Creates a new read-modify-write macro like PUSH or INCF."
ram's avatar
ram committed
  (let ((other-args nil)
	(rest-arg nil)
	(env (gensym "ENV-"))
	(reference (gensym "PLACE-")))
ram's avatar
ram committed
	     
    ;; 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))
ram's avatar
ram committed
	     (if (null (cddr ll))
		 (return nil)
		 (error (intl:gettext "Illegal stuff after &rest arg in Define-Modify-Macro."))))
ram's avatar
ram committed
	    ((memq arg '(&key &allow-other-keys &aux))
	     (error (intl:gettext "~S not allowed in Define-Modify-Macro lambda list.") arg))
ram's avatar
ram committed
	    ((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.")))))
ram's avatar
ram committed
    (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)
ram's avatar
ram committed
	 (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)))