;;; destructuring-bind.lisp ;;; ;;; Copyright (C) 2003-2005 Peter Graves ;;; $Id$ ;;; ;;; This program 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. ;;; ;;; This program 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 this program; if not, write to the Free Software ;;; Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ;;; ;;; As a special exception, the copyright holders of this library give you ;;; permission to link this library with independent modules to produce an ;;; executable, regardless of the license terms of these independent ;;; modules, and to copy and distribute the resulting executable under ;;; terms of your choice, provided that you also meet, for each linked ;;; independent module, the terms and conditions of the license of that ;;; module. An independent module is a module which is not derived from ;;; or based on this library. If you modify this library, you may extend ;;; this exception to your version of the library, but you are not ;;; obligated to do so. If you do not wish to do so, delete this ;;; exception statement from your version. ;;;; Adapted from CMUCL/SBCL. (in-package #:system) (export '(parse-body)) (defun parse-body (body &optional (doc-string-allowed t)) (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)))))))) ;; We don't have DEFVAR yet... (eval-when (:compile-toplevel :load-toplevel :execute) (%defvar '*arg-tests* ()) (%defvar '*system-lets* ()) (%defvar '*user-lets* ()) (%defvar '*ignorable-vars* ()) (%defvar '*env-var* nil)) (defun arg-count-error (error-kind name arg lambda-list minimum maximum) (declare (ignore error-kind arg lambda-list minimum maximum)) (error 'program-error :format-control "Wrong number of arguments for ~S." :format-arguments (list name))) (defun bogus-sublist-error (&key kind name object lambda-list) (error 'program-error :format-control "Error while parsing arguments to ~A ~S:~%Bogus sublist:~% ~S~%to satisfy lambda-list:~% ~:S~%" :format-arguments (list kind name object lambda-list))) (defun lambda-list-broken-key-list-error (&key kind name problem info) (error 'program-error :format-control (concatenate 'string "Error while parsing arguments to ~A ~S:~%" (ecase problem (:dotted-list "Keyword/value list is dotted: ~S") (:odd-length "Odd number of elements in keyword/value list: ~S") (:duplicate "Duplicate keyword: ~S") (:unknown-keyword "~{Unknown keyword: ~S ; expected one of ~{~S~^, ~}~}"))) :format-arguments (list kind name info))) ;;; Return, as multiple values, a body, possibly a DECLARE form to put ;;; where this code is inserted, the documentation for the parsed ;;; body, and bounds on the number of arguments. (defun parse-defmacro (lambda-list arg-list-name body name context &key (anonymousp nil) (doc-string-allowed t) ((:environment env-arg-name)) (error-fun 'error) (wrap-block t)) (multiple-value-bind (forms declarations documentation) (parse-body body doc-string-allowed) (let ((*arg-tests* ()) (*user-lets* ()) (*system-lets* ()) (*ignorable-vars* ()) (*env-var* nil)) (multiple-value-bind (env-arg-used minimum maximum) (parse-defmacro-lambda-list lambda-list arg-list-name name context error-fun (not anonymousp) nil) (values `(let* (,@(when env-arg-used `((,*env-var* ,env-arg-name))) ,@(nreverse *system-lets*)) ,@(when *ignorable-vars* `((declare (ignorable ,@*ignorable-vars*)))) ,@*arg-tests* (let* ,(nreverse *user-lets*) ,@declarations ,@(if wrap-block `((block ,(fdefinition-block-name name) ,@forms)) forms))) `(,@(when (and env-arg-name (not env-arg-used)) `((declare (ignore ,env-arg-name))))) documentation minimum maximum))))) (defun defmacro-error (problem name) (error 'type-error "~S is not of type ~S~%" problem name)) (defun verify-keywords (key-list valid-keys allow-other-keys) (do ((already-processed nil) (unknown-keyword nil) (remaining key-list (cddr remaining))) ((null remaining) (if (and unknown-keyword (not allow-other-keys) (not (lookup-keyword :allow-other-keys key-list))) (values :unknown-keyword (list unknown-keyword valid-keys)) (values nil nil))) (cond ((not (and (consp remaining) (listp (cdr remaining)))) (return (values :dotted-list key-list))) ((null (cdr remaining)) (return (values :odd-length key-list))) ((or (eq (car remaining) :allow-other-keys) (memql (car remaining) valid-keys)) (push (car remaining) already-processed)) (t (setq unknown-keyword (car remaining)))))) (defun lookup-keyword (keyword key-list) (do ((remaining key-list (cddr remaining))) ((endp remaining)) (when (eq keyword (car remaining)) (return (cadr remaining))))) (defun keyword-supplied-p (keyword key-list) (do ((remaining key-list (cddr remaining))) ((endp remaining)) (when (eq keyword (car remaining)) (return t)))) (defun dot-length (cons) (do ((rest cons (cdr rest)) (length 0 (1+ length))) ((or (null rest) (atom rest)) length))) (defun parse-defmacro-lambda-list (lambda-list arg-list-name name error-kind error-fun &optional top-level env-illegal ;;env-arg-name ) (let* ((path-0 (if top-level `(cdr ,arg-list-name) arg-list-name)) (path path-0) (now-processing :required) (maximum 0) (minimum 0) (keys ()) rest-name restp allow-other-keys-p env-arg-used) ;; This really strange way to test for &WHOLE is necessary because MEMBER ;; does not have to work on dotted lists, and dotted lists are legal ;; in lambda lists. (when (and (do ((list lambda-list (cdr list))) ((atom list) nil) (when (eq (car list) '&WHOLE) (return t))) (not (eq (car lambda-list) '&WHOLE))) (error "&Whole must appear first in ~S lambda-list." error-kind)) (do ((rest-of-args lambda-list (cdr rest-of-args))) ((atom rest-of-args) (cond ((null rest-of-args) nil) ;; Varlist is dotted, treat as &rest arg and exit. (t (push-let-binding rest-of-args path nil) (setq restp t)))) (let ((var (car rest-of-args))) (cond ((eq var '&whole) (cond ((and (cdr rest-of-args) (symbolp (cadr rest-of-args))) (setq rest-of-args (cdr rest-of-args)) (push-let-binding (car rest-of-args) arg-list-name nil)) ((and (cdr rest-of-args) (consp (cadr rest-of-args))) (pop rest-of-args) (let* ((destructuring-lambda-list (car rest-of-args)) (sub (gensym "WHOLE-SUBLIST"))) (push-sub-list-binding sub arg-list-name destructuring-lambda-list name error-kind error-fun) (parse-defmacro-lambda-list destructuring-lambda-list sub name error-kind error-fun))) (t (defmacro-error "&WHOLE" name)))) ((eq var '&environment) (cond (env-illegal (error "&ENVIRONMENT is not valid with ~S." error-kind)) ((not top-level) (error "&ENVIRONMENT is only valid at top level of lambda list."))) (cond ((and (cdr rest-of-args) (symbolp (cadr rest-of-args))) (setq rest-of-args (cdr rest-of-args)) (setq *env-var* (car rest-of-args) env-arg-used t)) (t (defmacro-error "&ENVIRONMENT" name)))) ((or (eq var '&rest) (eq var '&body)) (cond ((and (cdr rest-of-args) (symbolp (cadr rest-of-args))) (setq rest-of-args (cdr rest-of-args)) (setq restp t) (push-let-binding (car rest-of-args) path nil)) ((and (cdr rest-of-args) (consp (cadr rest-of-args))) (pop rest-of-args) (setq restp t) (let* ((destructuring-lambda-list (car rest-of-args)) (sub (gensym "REST-SUBLIST"))) (push-sub-list-binding sub path destructuring-lambda-list name error-kind error-fun) (parse-defmacro-lambda-list destructuring-lambda-list sub name error-kind error-fun))) (t (defmacro-error (symbol-name var) name)))) ((eq var '&optional) (setq now-processing :optionals)) ((eq var '&key) (setq now-processing :keywords) (setq rest-name (gensym "KEYWORDS-")) (push rest-name *ignorable-vars*) (setq restp t) (push-let-binding rest-name path t)) ((eq var '&allow-other-keys) (setq allow-other-keys-p t)) ((eq var '&aux) (setq now-processing :auxs)) ((listp var) (case now-processing (:required (let ((sub-list-name (gensym "SUBLIST-"))) (push-sub-list-binding sub-list-name `(car ,path) var name error-kind error-fun) (parse-defmacro-lambda-list var sub-list-name name error-kind error-fun)) (setq path `(cdr ,path)) (incf minimum) (incf maximum)) (:optionals (when (> (length var) 3) (error "more than variable, initform, and suppliedp in &optional binding ~S" var)) (push-optional-binding (car var) (cadr var) (caddr var) `(not (null ,path)) `(car ,path) name error-kind error-fun) (setq path `(cdr ,path)) (incf maximum)) (:keywords (let* ((keyword-given (consp (car var))) (variable (if keyword-given (cadar var) (car var))) (keyword (if keyword-given (caar var) (make-keyword variable))) (supplied-p (caddr var))) (push-optional-binding variable (cadr var) supplied-p `(keyword-supplied-p ',keyword ,rest-name) `(lookup-keyword ',keyword ,rest-name) name error-kind error-fun) (push keyword keys))) (:auxs (push-let-binding (car var) (cadr var) nil)))) ((symbolp var) (case now-processing (:required (incf minimum) (incf maximum) (push-let-binding var `(car ,path) nil) (setq path `(cdr ,path))) (:optionals (incf maximum) (push-let-binding var `(car ,path) nil `(not (null ,path))) (setq path `(cdr ,path))) (:keywords (let ((key (make-keyword var))) (push-let-binding var `(lookup-keyword ,key ,rest-name) nil) (push key keys))) (:auxs (push-let-binding var nil nil)))) (t (error "non-symbol in lambda-list: ~S" var))))) ;; Generate code to check the number of arguments. (push `(unless (<= ,minimum (dot-length ,path-0) ,@(unless restp (list maximum))) ,(if (eq error-fun 'error) `(arg-count-error ',error-kind ',name ,path-0 ',lambda-list ,minimum ,(unless restp maximum)) `(,error-fun 'arg-count-error :kind ',error-kind ,@(when name `(:name ',name)) :argument ,path-0 :lambda-list ',lambda-list :minimum ,minimum ,@(unless restp `(:maximum ,maximum))))) *arg-tests*) (if keys (let ((problem (gensym "KEY-PROBLEM-")) (info (gensym "INFO-"))) (push `(multiple-value-bind (,problem ,info) (verify-keywords ,rest-name ',keys ',allow-other-keys-p) (when ,problem ,(if (eq error-fun 'error) `(lambda-list-broken-key-list-error :kind ',error-kind ,@(when name `(:name ',name)) :problem ,problem :info ,info) `(,error-fun 'defmacro-lambda-list-broken-key-list-error :kind ',error-kind ,@(when name `(:name ',name)) :problem ,problem :info ,info)))) *arg-tests*))) (values env-arg-used minimum (if (null restp) maximum nil)))) (defun push-sub-list-binding (variable path object name error-kind error-fun) (let ((var (gensym "TEMP-"))) (push `(,variable (let ((,var ,path)) (if (listp ,var) ,var ,(if (eq error-fun 'error) `(bogus-sublist-error :kind ',error-kind ,@(when name `(:name ',name)) :object ,var :lambda-list ',object) `(,error-fun 'defmacro-bogus-sublist-error :kind ',error-kind ,@(when name `(:name ',name)) :object ,var :lambda-list ',object))))) *system-lets*))) (defun push-let-binding (variable path systemp &optional condition (init-form nil)) (let ((let-form (if condition `(,variable (if ,condition ,path ,init-form)) `(,variable ,path)))) (if systemp (push let-form *system-lets*) (push let-form *user-lets*)))) (defun push-optional-binding (value-var init-form supplied-var condition path name error-kind error-fun) (unless supplied-var (setq supplied-var (gensym "SUPPLIEDP-"))) (push-let-binding supplied-var condition t) (cond ((consp value-var) (let ((whole-thing (gensym "OPTIONAL-SUBLIST-"))) (push-sub-list-binding whole-thing `(if ,supplied-var ,path ,init-form) value-var name error-kind error-fun) (parse-defmacro-lambda-list value-var whole-thing name error-kind error-fun))) ((symbolp value-var) (push-let-binding value-var path nil supplied-var init-form)) (t (error "Illegal optional variable name: ~S" value-var)))) (defmacro destructuring-bind (lambda-list arg-list &rest body) (let* ((arg-list-name (gensym "ARG-LIST-"))) (multiple-value-bind (body local-decls) (parse-defmacro lambda-list arg-list-name body nil 'destructuring-bind :anonymousp t :doc-string-allowed nil :wrap-block nil) `(let ((,arg-list-name ,arg-list)) ,@local-decls ,body)))) ;; Redefine SYS:MAKE-MACRO-EXPANDER to use PARSE-DEFMACRO. (defun make-macro-expander (definition) (let* ((name (car definition)) (lambda-list (cadr definition)) (form (gensym "WHOLE-")) (env (gensym "ENVIRONMENT-")) (body (parse-defmacro lambda-list form (cddr definition) name 'defmacro :environment env))) `(lambda (,form ,env) (block ,name ,body)))) #| These conditions might be signaled but are not defined. Probably can't define them here as clos might not be active. Taken from cmucl. (define-condition defmacro-lambda-list-bind-error (program-error) ((kind :reader defmacro-lambda-list-bind-error-kind :initarg :kind) (name :reader defmacro-lambda-list-bind-error-name :initarg :name :initform nil))) (defun print-defmacro-ll-bind-error-intro (condition stream) (if (null (defmacro-lambda-list-bind-error-name condition)) (format stream "Error while parsing arguments to ~A in ~S:~%" (defmacro-lambda-list-bind-error-kind condition) (condition-function-name condition)) (format stream "Error while parsing arguments to ~A ~S:~%" (defmacro-lambda-list-bind-error-kind condition) (defmacro-lambda-list-bind-error-name condition)))) (define-condition defmacro-bogus-sublist-error (defmacro-lambda-list-bind-error) ((object :reader defmacro-bogus-sublist-error-object :initarg :object) (lambda-list :reader defmacro-bogus-sublist-error-lambda-list :initarg :lambda-list)) (:report (lambda (condition stream) (print-defmacro-ll-bind-error-intro condition stream) (format stream "Bogus sublist:~% ~S~%to satisfy lambda-list:~% ~:S~%" (defmacro-bogus-sublist-error-object condition) (defmacro-bogus-sublist-error-lambda-list condition))))) (define-condition arg-count-error (defmacro-lambda-list-bind-error) ((argument :reader defmacro-ll-arg-count-error-argument :initarg :argument) (lambda-list :reader defmacro-ll-arg-count-error-lambda-list :initarg :lambda-list) (minimum :reader defmacro-ll-arg-count-error-minimum :initarg :minimum) (maximum :reader defmacro-ll-arg-count-error-maximum :initarg :maximum)) (:report (lambda (condition stream) (print-defmacro-ll-bind-error-intro condition stream) (format stream "Invalid number of elements in:~% ~:S~%~ to satisfy lambda-list:~% ~:S~%" (defmacro-ll-arg-count-error-argument condition) (defmacro-ll-arg-count-error-lambda-list condition)) (cond ((null (defmacro-ll-arg-count-error-maximum condition)) (format stream "Expected at least ~D" (defmacro-ll-arg-count-error-minimum condition))) ((= (defmacro-ll-arg-count-error-minimum condition) (defmacro-ll-arg-count-error-maximum condition)) (format stream "Expected exactly ~D" (defmacro-ll-arg-count-error-minimum condition))) (t (format stream "Expected between ~D and ~D" (defmacro-ll-arg-count-error-minimum condition) (defmacro-ll-arg-count-error-maximum condition)))) (format stream ", but got ~D." (length (defmacro-ll-arg-count-error-argument condition)))))) (define-condition defmacro-lambda-list-broken-key-list-error (defmacro-lambda-list-bind-error) ((problem :reader defmacro-ll-broken-key-list-error-problem :initarg :problem) (info :reader defmacro-ll-broken-key-list-error-info :initarg :info)) (:report (lambda (condition stream) (print-defmacro-ll-bind-error-intro condition stream) (format stream (ecase (defmacro-ll-broken-key-list-error-problem condition) (:dotted-list "Keyword/value list is dotted: ~S") (:odd-length "Odd number of elements in keyword/value list: ~S") (:duplicate "Duplicate keyword: ~S") (:unknown-keyword "~{Unknown keyword: ~S; expected one of ~{~S~^, ~}~}")) (defmacro-ll-broken-key-list-error-info condition))))) |#