home *** CD-ROM | disk | FTP | other *** search
/ Amiga MA Magazine 1998 #6 / amigamamagazinepolishissue1998.iso / coders / jËzyki_programowania / clisp / src / archive / clisp.faslsp.lha / places.lsp < prev    next >
Lisp/Scheme  |  1996-04-15  |  51KB  |  1,132 lines

  1. ; CLISP - PLACES.LSP
  2. ; CLISP-spezifisch: string-concat, %rplaca, %rplacd, store, %setelt, ...
  3.  
  4. (in-package "SYSTEM")
  5. ;-------------------------------------------------------------------------------
  6. ; Funktionen zur Definition und zum Ausnutzen von places:
  7. ;-------------------------------------------------------------------------------
  8. (defun setf-symbol (symbol) ; liefert uninterniertes Symbol für SYSTEM::SETF-FUNCTION
  9.   (make-symbol
  10.     (string-concat
  11.       "(SETF "
  12.       (let ((pack (symbol-package symbol))) (if pack (package-name pack) "#"))
  13.       ":"
  14.       (symbol-name symbol)
  15.       ")"
  16. ) ) )
  17. ;-------------------------------------------------------------------------------
  18. (defun get-setf-symbol (symbol) ; liefert das Symbol bei SYSTEM::SETF-FUNCTION
  19.   (or (get symbol 'SYSTEM::SETF-FUNCTION)
  20.       (progn
  21.         (when (get symbol 'SYSTEM::SETF-EXPANDER)
  22.           (warn 
  23.            #L{
  24.            DEUTSCH "Die Funktion (~S ~S) ist durch einen SETF-Expander verborgen."
  25.            ENGLISH "The function (~S ~S) is hidden by a SETF expander."
  26.            FRANCAIS "La fonction (~S ~S) est cachée par une méthode SETF."
  27.            }
  28.            'setf symbol
  29.         ) )
  30.         (setf (get symbol 'SYSTEM::SETF-FUNCTION) (setf-symbol symbol))
  31. ) )   )
  32. ;-------------------------------------------------------------------------------
  33. (defun get-funname-symbol (funname) ; Abbildung Funktionsname --> Symbol
  34.   (if (atom funname)
  35.     funname
  36.     (get-setf-symbol (second funname))
  37. ) )
  38. ;-------------------------------------------------------------------------------
  39. (defun get-setf-method-multiple-value (form &optional (env (vector nil nil)))
  40.   (loop
  41.     ; 1. Schritt: nach globalen SETF-Definitionen suchen:
  42.     (when (and (consp form) (symbolp (car form)))
  43.       (when (global-in-fenv-p (car form) (svref env 1))
  44.         ; Operator nicht lokal definiert
  45.         (let ((plist-info (get (first form) 'SYSTEM::SETF-EXPANDER)))
  46.           (when plist-info
  47.             (return-from get-setf-method-multiple-value
  48.               (if (symbolp plist-info) ; Symbol kommt von kurzem DEFSETF
  49.                 (do* ((storevar (gensym))
  50.                       (tempvars nil (cons (gensym) tempvars))
  51.                       (tempforms nil (cons (car formr) tempforms))
  52.                       (formr (cdr form) (cdr formr)))
  53.                      ((atom formr)
  54.                       (setq tempforms (nreverse tempforms))
  55.                       (values tempvars
  56.                               tempforms
  57.                               `(,storevar)
  58.                               `(,plist-info ,@tempvars ,storevar)
  59.                               `(,(first form) ,@tempvars)
  60.                      ))
  61.                 )
  62.                 (let ((argcount (car plist-info)))
  63.                   (if (eql argcount -5)
  64.                     ; (-5 . fun) kommt von DEFINE-SETF-METHOD
  65.                     (funcall (cdr plist-info) form env)
  66.                     ; (argcount . fun) kommt von langem DEFSETF
  67.                     (let ((access-form form)
  68.                           (tempvars '())
  69.                           (tempforms '())
  70.                           (new-access-form '()))
  71.                       (let ((i 0)) ; Argumente-Zähler
  72.                         ; argcount = -1 falls keine Keyword-Argumente existieren
  73.                         ; bzw.     = Anzahl der einzelnen Argumente vor &KEY,
  74.                         ;          = nil nachdem diese abgearbeitet sind.
  75.                         (dolist (argform (cdr access-form))
  76.                           (when (eql i argcount) (setf argcount nil i 0))
  77.                           (if (and (null argcount) (evenp i))
  78.                             (if (keywordp argform)
  79.                               (push argform new-access-form)
  80.                               (error-of-type 'program-error
  81.                                 #L{
  82.                                 DEUTSCH "Das Argument ~S zu ~S sollte ein Keyword sein."
  83.                                 ENGLISH "The argument ~S to ~S should be a keyword."
  84.                                 FRANCAIS "L'argument ~S de ~S doit être un mot-clé."
  85.                                 }
  86.                                 argform (car access-form)
  87.                             ) )
  88.                             (let ((tempvar (gensym)))
  89.                               (push tempvar tempvars)
  90.                               (push argform tempforms)
  91.                               (push tempvar new-access-form)
  92.                           ) )
  93.                           (incf i)
  94.                       ) )
  95.                       (setq new-access-form
  96.                         (cons (car access-form) (nreverse new-access-form))
  97.                       )
  98.                       (let ((newval-var (gensym)))
  99.                         (values
  100.                           (nreverse tempvars)
  101.                           (nreverse tempforms)
  102.                           (list newval-var)
  103.                           (funcall (cdr plist-info) new-access-form newval-var)
  104.                           new-access-form
  105.                 ) ) ) ) )
  106.             ) )
  107.     ) ) ) )
  108.     ; 2. Schritt: macroexpandieren
  109.     (when (eq form (setq form (macroexpand-1 form env)))
  110.       (return)
  111.   ) )
  112.   ; 3. Schritt: Default-SETF-Methoden
  113.   (cond ((symbolp form)
  114.          (return-from get-setf-method-multiple-value
  115.            (let ((storevar (gensym)))
  116.              (values nil
  117.                      nil
  118.                      `(,storevar)
  119.                      `(SETQ ,form ,storevar)
  120.                      `,form
  121.         )) ) )
  122.         ((and (consp form) (symbolp (car form)))
  123.          (return-from get-setf-method-multiple-value
  124.            (do* ((storevar (gensym))
  125.                  (tempvars nil (cons (gensym) tempvars))
  126.                  (tempforms nil (cons (car formr) tempforms))
  127.                  (formr (cdr form) (cdr formr)))
  128.                 ((atom formr)
  129.                  (setq tempforms (nreverse tempforms))
  130.                  (values tempvars
  131.                          tempforms
  132.                          `(,storevar)
  133.                          `((SETF ,(first form)) ,storevar ,@tempvars)
  134.                          `(,(first form) ,@tempvars)
  135.                 ))
  136.         )) )
  137.         (t (error-of-type 'program-error
  138.              #L{
  139.              DEUTSCH "Das Argument muß eine 'SETF-place' sein, ist aber keine: ~S"
  140.              ENGLISH "Argument ~S is not a SETF place."
  141.              FRANCAIS "L'argument ~S doit représenter une place modifiable."
  142.              }
  143.              form
  144.   )     )  )
  145. )
  146. ;-------------------------------------------------------------------------------
  147. (defun get-setf-method (form &optional (env (vector nil nil)))
  148.   (multiple-value-bind (vars vals stores store-form access-form)
  149.       (get-setf-method-multiple-value form env)
  150.     (unless (and (consp stores) (null (cdr stores)))
  151.       (error-of-type 'program-error
  152.         #L{
  153.         DEUTSCH "Diese 'SETF-place' produziert mehrere 'Store-Variable': ~S"
  154.         ENGLISH "SETF place ~S produces more than one store variable."
  155.         FRANCAIS "La place modifiable ~S produit plusieurs variables de résultat."
  156.         }
  157.         form
  158.     ) )
  159.     (values vars vals stores store-form access-form)
  160. ) )
  161. ;-------------------------------------------------------------------------------
  162. ; In einfachen Zuweisungen wie (SETQ foo #:G0) darf #:G0 direkt ersetzt werden.
  163. (defun simple-assignment-p (store-form stores)
  164.   (and (eql (length stores) 1)
  165.        (consp store-form)
  166.        (eq (first store-form) 'SETQ)
  167.        (eql (length store-form) 3)
  168.        (symbolp (second store-form))
  169.        (simple-use-p (third store-form) (first stores))
  170. ) )
  171. (defun simple-use-p (form var)
  172.   (or (eq form var)
  173.       (and (consp form) (eq (first form) 'THE) (eql (length form) 3)
  174.            (simple-use-p (third form) var)
  175. ) )   )
  176. ;-------------------------------------------------------------------------------
  177. (defun documentation (symbol doctype)
  178.   (unless (function-name-p symbol)
  179.     (error-of-type 'error
  180.       #L{
  181.       DEUTSCH "~S: Das ist als erstes Argument unzulässig, da kein Symbol: ~S"
  182.       ENGLISH "~S: first argument ~S is illegal, not a symbol"
  183.       FRANCAIS "~S : Le premier argument ~S est invalide car ce n'est pas un symbole."
  184.       }
  185.       'documentation symbol
  186.   ) )
  187.   (getf (get (get-funname-symbol symbol) 'SYSTEM::DOCUMENTATION-STRINGS) doctype)
  188. )
  189. (defun SYSTEM::%SET-DOCUMENTATION (symbol doctype value)
  190.   (unless (function-name-p symbol)
  191.     (error-of-type 'error
  192.       #L{
  193.       DEUTSCH "~S: Das ist als erstes Argument unzulässig, da kein Symbol: ~S"
  194.       ENGLISH "~S: first argument ~S is illegal, not a symbol"
  195.       FRANCAIS "~S : Le premier argument ~S est invalide car ce n'est pas un symbole."
  196.       }
  197.       'documentation symbol
  198.   ) )
  199.   (setq symbol (get-funname-symbol symbol))
  200.   (if (null value)
  201.     (when (getf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype)
  202.       (remf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype)
  203.       nil
  204.     )
  205.     (setf (getf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype) value)
  206. ) )
  207. ;-------------------------------------------------------------------------------
  208. (defmacro push (item place &environment env)
  209.   (let ((itemvar (gensym)))
  210.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  211.       (let ((bindlist (mapcar #'list SM1 SM2)))
  212.         (if bindlist
  213.           (push `(,itemvar ,item) bindlist)
  214.           (setq itemvar item)
  215.         )
  216.         (let ((valform `(CONS ,itemvar ,SM5)))
  217.           (if (simple-assignment-p SM4 SM3)
  218.             (setq SM4 (subst valform (first SM3) SM4))
  219.             (setq bindlist (nconc bindlist `((,(first SM3) ,valform))))
  220.           )
  221.           (if bindlist
  222.             `(LET* ,bindlist
  223.                ,SM4
  224.              )
  225.             SM4
  226.           )
  227. ) ) ) ) )
  228. ;-------------------------------------------------------------------------------
  229. (defmacro define-setf-method (accessfn lambdalist &body body &environment env)
  230.   (unless (symbolp accessfn)
  231.     (error-of-type 'program-error
  232.       #L{
  233.       DEUTSCH "Der Name der Access-Function muß ein Symbol sein und nicht ~S."
  234.       ENGLISH "The name of the access function must be a symbol, not ~S"
  235.       FRANCAIS "Le nom de la fonction d'accès doit être un symbole et non ~S."
  236.       }
  237.       accessfn
  238.   ) )
  239.   (multiple-value-bind (body-rest declarations docstring)
  240.       (system::parse-body body t env)
  241.     (if (null body-rest) (setq body-rest '(NIL)))
  242.     (let ((name (make-symbol (string-concat "SETF-" (symbol-name accessfn)))))
  243.       (multiple-value-bind (newlambdalist envvar) (remove-env-arg lambdalist name)
  244.         (let ((SYSTEM::%ARG-COUNT 0)
  245.               (SYSTEM::%MIN-ARGS 0)
  246.               (SYSTEM::%RESTP nil)
  247.               (SYSTEM::%LET-LIST nil)
  248.               (SYSTEM::%KEYWORD-TESTS nil)
  249.               (SYSTEM::%DEFAULT-FORM nil)
  250.              )
  251.           (SYSTEM::ANALYZE1 newlambdalist '(CDR SYSTEM::%LAMBDA-LIST)
  252.                             name 'SYSTEM::%LAMBDA-LIST
  253.           )
  254.           (if (null newlambdalist)
  255.             (push `(IGNORE SYSTEM::%LAMBDA-LIST) declarations)
  256.           )
  257.           (let ((lengthtest (sys::make-length-test 'SYSTEM::%LAMBDA-LIST))
  258.                 (mainform
  259.                   `(LET* ,(nreverse SYSTEM::%LET-LIST)
  260.                      ,@(if declarations `(,(cons 'DECLARE declarations)))
  261.                      ,@SYSTEM::%KEYWORD-TESTS
  262.                      (BLOCK ,accessfn ,@body-rest)
  263.                    )
  264.                ))
  265.             (if lengthtest
  266.               (setq mainform
  267.                 `(IF ,lengthtest
  268.                    (ERROR-OF-TYPE 'PROGRAM-ERROR
  269.                      #L{
  270.                      DEUTSCH "Der SETF-Expander für ~S kann nicht mit ~S Argumenten aufgerufen werden."
  271.                      ENGLISH "The SETF expander for ~S may not be called with ~S arguments."
  272.                      FRANCAIS "L'«expandeur» SETF pour ~S ne peut pas être appelé avec ~S arguments."
  273.                      }
  274.                      (QUOTE ,accessfn) (1- (LENGTH SYSTEM::%LAMBDA-LIST))
  275.                    )
  276.                    ,mainform
  277.               )  )
  278.             )
  279.             `(EVAL-WHEN (LOAD COMPILE EVAL)
  280.                (LET ()
  281.                  (DEFUN ,name (SYSTEM::%LAMBDA-LIST ,(or envvar 'SYSTEM::ENV))
  282.                    ,@(if envvar '() '((DECLARE (IGNORE SYSTEM::ENV))))
  283.                    ,mainform
  284.                  )
  285.                  (SYSTEM::%PUT ',accessfn 'SYSTEM::SETF-EXPANDER
  286.                    (CONS -5 (FUNCTION ,name))
  287.                  )
  288.                  (SYSTEM::%SET-DOCUMENTATION ',accessfn 'SETF ',docstring)
  289.                  ',accessfn
  290.              ) )
  291. ) ) ) ) ) )
  292. ;-------------------------------------------------------------------------------
  293. (defmacro defsetf (accessfn &rest args &environment env)
  294.   (cond ((and (consp args) (not (listp (first args))) (symbolp (first args)))
  295.          `(EVAL-WHEN (LOAD COMPILE EVAL)
  296.             (LET ()
  297.               (SYSTEM::%PUT ',accessfn 'SYSTEM::SETF-EXPANDER ',(first args))
  298.               (SYSTEM::%SET-DOCUMENTATION ',accessfn 'SETF
  299.                 ,(if (and (null (cddr args))
  300.                           (or (null (second args)) (stringp (second args)))
  301.                      )
  302.                    (second args)
  303.                    (if (cddr args)
  304.                      (error-of-type 'program-error
  305.                        #L{
  306.                        DEUTSCH "Zu viele Argumente für DEFSETF: ~S"
  307.                        ENGLISH "Too many arguments to DEFSETF: ~S"
  308.                        FRANCAIS "Trop d'arguments pour DEFSETF : ~S"
  309.                        }
  310.                        (cdr args)
  311.                      )
  312.                      (error-of-type 'program-error
  313.                        #L{
  314.                        DEUTSCH "Der Dok.-String zu DEFSETF muß ein String sein: ~S"
  315.                        ENGLISH "The doc string to DEFSETF must be a string: ~S"
  316.                        FRANCAIS "La documentation pour DEFSETF doit être un chaîne : ~S"
  317.                        }
  318.                        (second args)
  319.                  ) ) )
  320.               )
  321.               ',accessfn
  322.           ) )
  323.         )
  324.         ((and (consp args) (listp (first args)) (consp (cdr args)) (listp (second args)))
  325.          (cond ((= (length (second args)) 1))
  326.                ((= (length (second args)) 0)
  327.                 (error-of-type 'program-error
  328.                   #L{
  329.                   DEUTSCH "Bei DEFSETF muß genau eine 'Store-Variable' angegeben werden."
  330.                   ENGLISH "Missing store variable in DEFSETF."
  331.                   FRANCAIS "Une variable de résultat doit être précisée dans DEFSETF."
  332.                   }
  333.                ))
  334.                (t (cerror 
  335.                    #L{
  336.                    DEUTSCH "Die überzähligen Variablen werden ignoriert."
  337.                    ENGLISH "The excess variables will be ignored."
  338.                    FRANCAIS "Les variables en excès seront ignorées."
  339.                    }
  340.                    #L{
  341.                    DEUTSCH "Bei DEFSETF ist nur eine 'Store-Variable' erlaubt."
  342.                    ENGLISH "Only one store variable is allowed in DEFSETF."
  343.                    FRANCAIS "Une seule variable de résultat est permise dans DEFSETF."
  344.                    }
  345.          )     )  )
  346.          (multiple-value-bind (body-rest declarations docstring)
  347.              (system::parse-body (cddr args) t env)
  348.            (let* (arg-count
  349.                   (setter
  350.                     (let* ((lambdalist (first args))
  351.                            (storevar (first (second args)))
  352.                            (SYSTEM::%ARG-COUNT 0)
  353.                            (SYSTEM::%MIN-ARGS 0)
  354.                            (SYSTEM::%RESTP nil)
  355.                            (SYSTEM::%LET-LIST nil)
  356.                            (SYSTEM::%KEYWORD-TESTS nil)
  357.                            (SYSTEM::%DEFAULT-FORM nil))
  358.                       (SYSTEM::ANALYZE1 lambdalist '(CDR SYSTEM::%ACCESS-ARGLIST)
  359.                                         accessfn 'SYSTEM::%ACCESS-ARGLIST
  360.                       )
  361.                       (setq arg-count (if (member '&KEY lambdalist) SYSTEM::%ARG-COUNT -1))
  362.                       (when declarations (setq declarations `((DECLARE ,@declarations))))
  363.                       `(LAMBDA (SYSTEM::%ACCESS-ARGLIST ,storevar)
  364.                          ,@(if (null lambdalist)
  365.                              `((DECLARE (IGNORE SYSTEM::%ACCESS-ARGLIST)))
  366.                            )
  367.                          ,@declarations
  368.                          (LET* ,(nreverse SYSTEM::%LET-LIST)
  369.                            ,@declarations
  370.                            ,@SYSTEM::%KEYWORD-TESTS
  371.                            (BLOCK ,accessfn ,@body-rest)
  372.                        ) )
  373.                  )) )
  374.              `(EVAL-WHEN (LOAD COMPILE EVAL)
  375.                 (LET ()
  376.                   (SYSTEM::%PUT ',accessfn 'SYSTEM::SETF-EXPANDER
  377.                     (CONS ,arg-count
  378.                           (FUNCTION ,(concat-pnames "SETF-" accessfn) ,setter)
  379.                   ) )
  380.                   (SYSTEM::%SET-DOCUMENTATION ',accessfn 'SETF ,docstring)
  381.                   ',accessfn
  382.               ) )
  383.         )) )
  384.         (t (error-of-type 'program-error
  385.              #L{
  386.              DEUTSCH "DEFSETF-Aufruf für ~S ist falsch aufgebaut."
  387.              ENGLISH "Illegal syntax in DEFSETF for ~S"
  388.              FRANCAIS "Le DEFSETF ~S est mal formé."
  389.              }
  390.              accessfn
  391. ) )     )  )
  392. ;-------------------------------------------------------------------------------
  393. (defmacro pop (place &environment env)
  394.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  395.     (do* ((SM1r SM1 (cdr SM1r))
  396.           (SM2r SM2 (cdr SM2r))
  397.           (bindlist nil))
  398.          ((null SM1r)
  399.           (let* ((valform
  400.                    (if (and (symbolp SM5) (simple-assignment-p SM4 SM3))
  401.                      SM5
  402.                      (progn (push `(,(first SM3) ,SM5) bindlist) (first SM3))
  403.                  ) )
  404.                  (newvalform `(CDR ,valform))
  405.                  (form `(PROG1
  406.                           (CAR ,valform)
  407.                           ,@(if (simple-assignment-p SM4 SM3)
  408.                               (list (subst newvalform (first SM3) SM4))
  409.                               (list `(SETQ ,(first SM3) ,newvalform) SM4)
  410.                             )
  411.                         )
  412.                 ))
  413.             (if bindlist
  414.               `(LET* ,(nreverse bindlist) ,form)
  415.               form
  416.          )) )
  417.       (push `(,(first SM1r) ,(first SM2r)) bindlist)
  418. ) ) )
  419. ;-------------------------------------------------------------------------------
  420. (defmacro psetf (&whole form &rest args &environment env)
  421.   (do ((arglist args (cddr arglist))
  422.        (bindlist nil)
  423.        (storelist nil))
  424.       ((atom arglist)
  425.        `(LET* ,(nreverse bindlist)
  426.           ,@storelist
  427.           NIL
  428.       ) )
  429.     (when (atom (cdr arglist))
  430.       (error-of-type 'program-error
  431.         #L{
  432.         DEUTSCH "~S mit einer ungeraden Zahl von Argumenten aufgerufen: ~S"
  433.         ENGLISH "~S called with an odd number of arguments: ~S"
  434.         FRANCAIS "~S fut appelé avec un nombre impair d'arguments : ~S"
  435.         }
  436.         'psetf form
  437.     ) )
  438.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method (first arglist) env)
  439.       (declare (ignore SM5))
  440.       (do* ((SM1r SM1 (cdr SM1r))
  441.             (SM2r SM2 (cdr SM2r)))
  442.            ((null SM1r))
  443.         (push `(,(first SM1r) ,(first SM2r)) bindlist)
  444.       )
  445.       (push `(,(first SM3) ,(second arglist)) bindlist)
  446.       (push SM4 storelist)
  447. ) ) )
  448. ;-------------------------------------------------------------------------------
  449. (defmacro pushnew (item place &rest keylist &environment env)
  450.   (let ((itemvar (gensym)))
  451.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  452.       (let ((bindlist (mapcar #'list SM1 SM2)))
  453.         (if bindlist
  454.           (push `(,itemvar ,item) bindlist)
  455.           (setq itemvar item)
  456.         )
  457.         (let ((valform `(ADJOIN ,itemvar ,SM5 ,@keylist)))
  458.           (if (simple-assignment-p SM4 SM3)
  459.             (setq SM4 (subst valform (first SM3) SM4))
  460.             (setq bindlist (nconc bindlist `((,(first SM3) ,valform))))
  461.           )
  462.           (if bindlist
  463.             `(LET* ,bindlist
  464.                ,SM4
  465.              )
  466.             SM4
  467.           )
  468. ) ) ) ) )
  469. ;-------------------------------------------------------------------------------
  470. (defmacro remf (place indicator &environment env)
  471.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  472.     (do* ((SM1r SM1 (cdr SM1r))
  473.           (SM2r SM2 (cdr SM2r))
  474.           (bindlist nil)
  475.           (indicatorvar (gensym))
  476.           (var1 (gensym))
  477.           (var2 (gensym)))
  478.          ((null SM1r)
  479.           (push `(,(first SM3) ,SM5) bindlist)
  480.           (push `(,indicatorvar ,indicator) bindlist)
  481.           `(LET* ,(nreverse bindlist)
  482.              (DO ((,var1 ,(first SM3) (CDDR ,var1))
  483.                   (,var2 NIL ,var1))
  484.                  ((ATOM ,var1) NIL)
  485.                (COND ((ATOM (CDR ,var1))
  486.                       (ERROR-OF-TYPE 'ERROR
  487.                         #L{
  488.                         DEUTSCH "REMF: Property-Liste ungerader Länge aufgetreten."
  489.                         ENGLISH "REMF: property list with an odd length"
  490.                         FRANCAIS "REMF : Occurence d'une liste de propriétés de longueur impaire."
  491.                         }
  492.                      ))
  493.                      ((EQ (CAR ,var1) ,indicatorvar)
  494.                       (IF ,var2
  495.                         (RPLACD (CDR ,var2) (CDDR ,var1))
  496.                         ,(let ((newvalform `(CDDR ,(first SM3))))
  497.                            (if (simple-assignment-p SM4 SM3)
  498.                              (subst newvalform (first SM3) SM4)
  499.                              `(PROGN (SETQ ,(first SM3) ,newvalform) ,SM4)
  500.                          ) )
  501.                       )
  502.                       (RETURN T)
  503.            ) ) )     )
  504.          )
  505.       (push `(,(first SM1r) ,(first SM2r)) bindlist)
  506. ) ) )
  507. ;-------------------------------------------------------------------------------
  508. (defmacro rotatef (&rest args &environment env)
  509.   (cond ((null args) NIL)
  510.         ((null (cdr args)) `(PROGN ,(car args) NIL) )
  511.         (t (do* ((arglist args (cdr arglist))
  512.                  (bindlist nil)
  513.                  (storelist nil)
  514.                  (lastvar nil)
  515.                  (firstbind nil))
  516.                 ((atom arglist)
  517.                  (setf (car firstbind) lastvar)
  518.                  `(LET* ,(nreverse bindlist) ,@(nreverse storelist) NIL)
  519.                 )
  520.              (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  521.                  (get-setf-method (first arglist) env)
  522.                (do* ((SM1r SM1 (cdr SM1r))
  523.                      (SM2r SM2 (cdr SM2r)))
  524.                     ((null SM1r))
  525.                  (push `(,(first SM1r) ,(first SM2r)) bindlist)
  526.                )
  527.                (push `(,lastvar ,SM5) bindlist)
  528.                (if (null firstbind) (setq firstbind (first bindlist)))
  529.                (push SM4 storelist)
  530.                (setq lastvar (first SM3))
  531. ) )     )  ) )
  532. ;-------------------------------------------------------------------------------
  533. (defmacro define-modify-macro (name lambdalist function &optional docstring)
  534.   (let* ((varlist nil)
  535.          (restvar nil))
  536.     (do* ((lambdalistr lambdalist (cdr lambdalistr))
  537.           (next))
  538.          ((null lambdalistr))
  539.       (setq next (first lambdalistr))
  540.       (cond ((eq next '&OPTIONAL))
  541.             ((eq next '&REST)
  542.              (if (symbolp (second lambdalistr))
  543.                (setq restvar (second lambdalistr))
  544.                (error-of-type 'program-error
  545.                  #L{
  546.                  DEUTSCH "In der Definition von ~S ist die &REST-Variable kein Symbol: ~S"
  547.                  ENGLISH "In the definition of ~S: &REST variable ~S should be a symbol."
  548.                  FRANCAIS "Dans la définition de ~S la variable pour &REST n'est pas un symbole : ~S."
  549.                  }
  550.                  name (second lambdalistr)
  551.              ) )
  552.              (if (null (cddr lambdalistr))
  553.                (return)
  554.                (error-of-type 'program-error
  555.                  #L{
  556.                  DEUTSCH "Nach &REST ist nur eine Variable erlaubt; es kam: ~S"
  557.                  ENGLISH "Only one variable is allowed after &REST, not ~S"
  558.                  FRANCAIS "Une seule variable est permise pour &REST et non ~S."
  559.                  }
  560.                  lambdalistr
  561.             )) )
  562.             ((or (eq next '&KEY) (eq next '&ALLOW-OTHER-KEYS) (eq next '&AUX))
  563.              (error-of-type 'program-error
  564.                #L{
  565.                DEUTSCH "In einer DEFINE-MODIFY-MACRO-Lambdaliste ist ~S unzulässig."
  566.                ENGLISH "Illegal in a DEFINE-MODIFY-MACRO lambda list: ~S"
  567.                FRANCAIS "~S n'est pas permis dans une liste lambda pour DEFINE-MODIFY-MACRO."
  568.                }
  569.                next
  570.             ))
  571.             ((symbolp next) (push next varlist))
  572.             ((and (listp next) (symbolp (first next)))
  573.              (push (first next) varlist)
  574.             )
  575.             (t (error-of-type 'program-error
  576.                  #L{
  577.                  DEUTSCH "Lambdalisten dürfen nur Symbole und Listen enthalten, nicht aber ~S"
  578.                  ENGLISH "lambda list may only contain symbols and lists, not ~S"
  579.                  FRANCAIS "Les listes lambda ne peuvent contenir que des symboles et des listes et non ~S."
  580.                  }
  581.                  next
  582.             )  )
  583.     ) )
  584.     (setq varlist (nreverse varlist))
  585.     `(DEFMACRO ,name (%REFERENCE ,@lambdalist &ENVIRONMENT ENV) ,docstring
  586.        (MULTIPLE-VALUE-BIND (DUMMIES VALS NEWVAL SETTER GETTER)
  587.            (GET-SETF-METHOD %REFERENCE ENV)
  588.          (DO ((D DUMMIES (CDR D))
  589.               (V VALS (CDR V))
  590.               (LET-LIST NIL (CONS (LIST (CAR D) (CAR V)) LET-LIST)))
  591.              ((NULL D)
  592.               (WHEN (SYMBOLP GETTER)
  593.                 (RETURN
  594.                   (SUBST
  595.                     (LIST* (QUOTE ,function) GETTER ,@varlist ,restvar)
  596.                     (CAR NEWVAL)
  597.                     SETTER
  598.               ) ) )
  599.               (PUSH
  600.                 (LIST
  601.                   (CAR NEWVAL)
  602.                   (IF (AND (LISTP %REFERENCE) (EQ (CAR %REFERENCE) 'THE))
  603.                     (LIST 'THE (CADR %REFERENCE)
  604.                       (LIST* (QUOTE ,function) GETTER ,@varlist ,restvar)
  605.                     )
  606.                     (LIST* (QUOTE ,function) GETTER ,@varlist ,restvar)
  607.                 ) )
  608.                 LET-LIST
  609.               )
  610.               (LIST 'LET* (NREVERSE LET-LIST) SETTER)
  611.      ) ) ) )
  612. ) )
  613. ;-------------------------------------------------------------------------------
  614. (define-modify-macro decf (&optional (delta 1)) -)
  615. ;-------------------------------------------------------------------------------
  616. (define-modify-macro incf (&optional (delta 1)) +)
  617. ;-------------------------------------------------------------------------------
  618. (defmacro setf (&whole form &rest args &environment env)
  619.   (let ((argcount (length args)))
  620.     (cond ((eql argcount 2)
  621.            (let* ((place (first args))
  622.                   (value (second args)))
  623.              (loop
  624.                ; 1. Schritt: nach globalen SETF-Definitionen suchen:
  625.                (when (and (consp place) (symbolp (car place)))
  626.                  (when (global-in-fenv-p (car place) (svref env 1))
  627.                    ; Operator nicht lokal definiert
  628.                    (let ((plist-info (get (first place) 'SYSTEM::SETF-EXPANDER)))
  629.                      (when plist-info
  630.                        (return-from setf
  631.                          (cond ((symbolp plist-info) ; Symbol kommt von kurzem DEFSETF
  632.                                 `(,plist-info ,@(cdr place) ,value)
  633.                                )
  634.                                ((and (eq (first place) 'THE) (eql (length place) 3))
  635.                                 `(SETF ,(third place) (THE ,(second place) ,value))
  636.                                )
  637.                                ((and (eq (first place) 'VALUES-LIST) (eql (length place) 2))
  638.                                 `(VALUES-LIST
  639.                                    (SETF ,(second place)
  640.                                          (MULTIPLE-VALUE-LIST ,value)
  641.                                ) ) )
  642.                                (t
  643.                                 (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  644.                                     (get-setf-method-multiple-value place env)
  645.                                   (declare (ignore SM5))
  646.                                   (do* ((SM1r SM1 (cdr SM1r))
  647.                                         (SM2r SM2 (cdr SM2r))
  648.                                         (bindlist nil))
  649.                                        ((null SM1r)
  650.                                         (if (eql (length SM3) 1) ; eine Store-Variable
  651.                                           `(LET* ,(nreverse
  652.                                                     (cons `(,(first SM3) ,value)
  653.                                                           bindlist
  654.                                                   ) )
  655.                                              ,SM4
  656.                                            )
  657.                                           ; mehrere Store-Variable
  658.                                           (if
  659.                                             ; Hat SM4 die Gestalt
  660.                                             ; (VALUES (SETQ v1 store1) ...) ?
  661.                                             (and (consp SM4) (eq (car SM4) 'VALUES)
  662.                                               (do ((SM3r SM3 (cdr SM3r))
  663.                                                    (SM4r (cdr SM4) (cdr SM4r)))
  664.                                                   ((or (null SM3r) (null SM4r))
  665.                                                    (and (null SM3r) (null SM4r))
  666.                                                   )
  667.                                                 (unless (simple-assignment-p (car SM4r) (list (car SM3r)))
  668.                                                   (return nil)
  669.                                             ) ) )
  670.                                             (let ((vlist (mapcar #'second (rest SM4))))
  671.                                               `(LET* ,(nreverse bindlist)
  672.                                                  (MULTIPLE-VALUE-SETQ ,vlist ,value)
  673.                                                  (VALUES ,@vlist)
  674.                                             )  )
  675.                                             `(LET* ,(nreverse bindlist)
  676.                                                (MULTIPLE-VALUE-BIND ,SM3 ,value
  677.                                                  ,SM4
  678.                                              ) )
  679.                                        )) )
  680.                                     (push `(,(first SM1r) ,(first SM2r)) bindlist)
  681.                        ) )     )) )
  682.                ) ) ) )
  683.                ; 2. Schritt: macroexpandieren
  684.                (when (eq place (setq place (macroexpand-1 place env)))
  685.                  (return)
  686.              ) )
  687.              ; 3. Schritt: Default-SETF-Methoden
  688.              (cond ((symbolp place)
  689.                     `(SETQ ,place ,value)
  690.                    )
  691.                    ((and (consp form) (symbolp (car form)))
  692.                     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  693.                         (get-setf-method-multiple-value place env)
  694.                       (declare (ignore SM5))
  695.                       ; SM4 hat die Gestalt `((SETF ,(first place)) ,@SM3 ,@SM1).
  696.                       ; SM3 ist überflüssig.
  697.                       `(LET* ,(mapcar #'list SM1 SM2)
  698.                          ,(subst value (first SM3) SM4)
  699.                        )
  700.                    ))
  701.                    (t (error-of-type 'program-error
  702.                         #L{
  703.                         DEUTSCH "Das ist keine erlaubte 'SETF-Place' : ~S"
  704.                         ENGLISH "Illegal SETF place: ~S"
  705.                         FRANCAIS "Ceci n'est pas une place modifiable valide : ~S"
  706.                         }
  707.                         (first args)
  708.              )     )  )
  709.           ))
  710.           ((oddp argcount)
  711.            (error-of-type 'program-error
  712.              #L{
  713.              DEUTSCH "~S mit einer ungeraden Zahl von Argumenten aufgerufen: ~S"
  714.              ENGLISH "~S called with an odd number of arguments: ~S"
  715.              FRANCAIS "~S fut appelé avec un nombre impair d'arguments : ~S"
  716.              }
  717.              'setf form
  718.           ))
  719.           (t (do* ((arglist args (cddr arglist))
  720.                    (L nil))
  721.                   ((null arglist) `(LET () (PROGN ,@(nreverse L))))
  722.                (push `(SETF ,(first arglist) ,(second arglist)) L)
  723.           )  )
  724. ) ) )
  725. ;-------------------------------------------------------------------------------
  726. (defmacro shiftf (&whole form &rest args &environment env)
  727.   (when (< (length args) 2)
  728.     (error-of-type 'program-error
  729.       #L{
  730.       DEUTSCH "SHIFTF mit zu wenig Argumenten aufgerufen: ~S"
  731.       ENGLISH "SHIFTF called with too few arguments: ~S"
  732.       FRANCAIS "SHIFTF fut appelé avec trop peu d'arguments : ~S"
  733.       }
  734.       form
  735.   ) )
  736.   (do* ((resultvar (gensym))
  737.         (arglist args (cdr arglist))
  738.         (bindlist nil)
  739.         (storelist nil)
  740.         (lastvar resultvar))
  741.        ((atom (cdr arglist))
  742.         (push `(,lastvar ,(first arglist)) bindlist)
  743.         `(LET* ,(nreverse bindlist) ,@(nreverse storelist) ,resultvar)
  744.        )
  745.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method (first arglist) env)
  746.       (do* ((SM1r SM1 (cdr SM1r))
  747.             (SM2r SM2 (cdr SM2r)))
  748.            ((null Sm1r))
  749.         (push `(,(first SM1r) ,(first SM2r)) bindlist)
  750.       )
  751.       (push `(,lastvar ,SM5) bindlist)
  752.       (push SM4 storelist)
  753.       (setq lastvar (first SM3))
  754. ) ) )
  755. ;-------------------------------------------------------------------------------
  756. ; Definition von places:
  757. ;-------------------------------------------------------------------------------
  758. (defsetf aref (array &rest indices) (value)
  759.   `(SYSTEM::STORE ,array ,@indices ,value)
  760. )
  761. ;-------------------------------------------------------------------------------
  762. (defun SYSTEM::%SETNTH (index list value)
  763.   (let ((pointer (nthcdr index list)))
  764.     (if (null pointer)
  765.       (error-of-type 'error
  766.         #L{
  767.         DEUTSCH "(SETF (NTH ...) ...) : Index ~S ist zu groß für ~S."
  768.         ENGLISH "(SETF (NTH ...) ...) : index ~S is too large for ~S"
  769.         FRANCAIS "(SETF (NTH ...) ...) : L'index ~S est trop grand pour ~S."
  770.         }
  771.         index list
  772.       )
  773.       (rplaca pointer value)
  774.     )
  775.     value
  776. ) )
  777. (defsetf nth SYSTEM::%SETNTH)
  778. ;-------------------------------------------------------------------------------
  779. (defsetf elt SYSTEM::%SETELT)
  780. ;-------------------------------------------------------------------------------
  781. (defsetf rest SYSTEM::%RPLACD)
  782. (defsetf first SYSTEM::%RPLACA)
  783. (defsetf second (list) (value) `(SYSTEM::%RPLACA (CDR ,list) ,value))
  784. (defsetf third (list) (value) `(SYSTEM::%RPLACA (CDDR ,list) ,value))
  785. (defsetf fourth (list) (value) `(SYSTEM::%RPLACA (CDDDR ,list) ,value))
  786. (defsetf fifth (list) (value) `(SYSTEM::%RPLACA (CDDDDR ,list) ,value))
  787. (defsetf sixth (list) (value) `(SYSTEM::%RPLACA (CDR (CDDDDR ,list)) ,value))
  788. (defsetf seventh (list) (value) `(SYSTEM::%RPLACA (CDDR (CDDDDR ,list)) ,value))
  789. (defsetf eighth (list) (value) `(SYSTEM::%RPLACA (CDDDR (CDDDDR ,list)) ,value))
  790. (defsetf ninth (list) (value) `(SYSTEM::%RPLACA (CDDDDR (CDDDDR ,list)) ,value))
  791. (defsetf tenth (list) (value) `(SYSTEM::%RPLACA (CDR (CDDDDR (CDDDDR ,list))) ,value))
  792.  
  793. (defsetf car SYSTEM::%RPLACA)
  794. (defsetf cdr SYSTEM::%RPLACD)
  795. (defsetf caar (list) (value) `(SYSTEM::%RPLACA (CAR ,list) ,value))
  796. (defsetf cadr (list) (value) `(SYSTEM::%RPLACA (CDR ,list) ,value))
  797. (defsetf cdar (list) (value) `(SYSTEM::%RPLACD (CAR ,list) ,value))
  798. (defsetf cddr (list) (value) `(SYSTEM::%RPLACD (CDR ,list) ,value))
  799. (defsetf caaar (list) (value) `(SYSTEM::%RPLACA (CAAR ,list) ,value))
  800. (defsetf caadr (list) (value) `(SYSTEM::%RPLACA (CADR ,list) ,value))
  801. (defsetf cadar (list) (value) `(SYSTEM::%RPLACA (CDAR ,list) ,value))
  802. (defsetf caddr (list) (value) `(SYSTEM::%RPLACA (CDDR ,list) ,value))
  803. (defsetf cdaar (list) (value) `(SYSTEM::%RPLACD (CAAR ,list) ,value))
  804. (defsetf cdadr (list) (value) `(SYSTEM::%RPLACD (CADR ,list) ,value))
  805. (defsetf cddar (list) (value) `(SYSTEM::%RPLACD (CDAR ,list) ,value))
  806. (defsetf cdddr (list) (value) `(SYSTEM::%RPLACD (CDDR ,list) ,value))
  807. (defsetf caaaar (list) (value) `(SYSTEM::%RPLACA (CAAAR ,list) ,value))
  808. (defsetf caaadr (list) (value) `(SYSTEM::%RPLACA (CAADR ,list) ,value))
  809. (defsetf caadar (list) (value) `(SYSTEM::%RPLACA (CADAR ,list) ,value))
  810. (defsetf caaddr (list) (value) `(SYSTEM::%RPLACA (CADDR ,list) ,value))
  811. (defsetf cadaar (list) (value) `(SYSTEM::%RPLACA (CDAAR ,list) ,value))
  812. (defsetf cadadr (list) (value) `(SYSTEM::%RPLACA (CDADR ,list) ,value))
  813. (defsetf caddar (list) (value) `(SYSTEM::%RPLACA (CDDAR ,list) ,value))
  814. (defsetf cadddr (list) (value) `(SYSTEM::%RPLACA (CDDDR ,list) ,value))
  815. (defsetf cdaaar (list) (value) `(SYSTEM::%RPLACD (CAAAR ,list) ,value))
  816. (defsetf cdaadr (list) (value) `(SYSTEM::%RPLACD (CAADR ,list) ,value))
  817. (defsetf cdadar (list) (value) `(SYSTEM::%RPLACD (CADAR ,list) ,value))
  818. (defsetf cdaddr (list) (value) `(SYSTEM::%RPLACD (CADDR ,list) ,value))
  819. (defsetf cddaar (list) (value) `(SYSTEM::%RPLACD (CDAAR ,list) ,value))
  820. (defsetf cddadr (list) (value) `(SYSTEM::%RPLACD (CDADR ,list) ,value))
  821. (defsetf cdddar (list) (value) `(SYSTEM::%RPLACD (CDDAR ,list) ,value))
  822. (defsetf cddddr (list) (value) `(SYSTEM::%RPLACD (CDDDR ,list) ,value))
  823. ;-------------------------------------------------------------------------------
  824. (defsetf svref SYSTEM::SVSTORE)
  825. (defsetf row-major-aref system::row-major-store)
  826. ;-------------------------------------------------------------------------------
  827. (defsetf GET (symbol indicator &optional default) (value)
  828.   (let ((storeform `(SYSTEM::%PUT ,symbol ,indicator ,value)))
  829.     (if default
  830.       `(PROGN ,default ,storeform) ; default wird nur zum Schein ausgewertet
  831.       `,storeform
  832. ) ) )
  833. ;-------------------------------------------------------------------------------
  834. ; Schreibt zu einem bestimmten Indicator einen Wert in eine gegebene
  835. ; Propertyliste. Wert ist NIL falls erfolgreich getan oder die neue
  836. ; (erweiterte) Propertyliste.
  837. (defun sys::%putf (plist indicator value)
  838.   (do ((plistr plist (cddr plistr)))
  839.       ((atom plistr) (list* indicator value plist))
  840.     (when (atom (cdr plistr))
  841.       (error-of-type 'error
  842.         #L{
  843.         DEUTSCH "(SETF (GETF ...) ...) : Property-Liste ungerader Länge aufgetaucht."
  844.         ENGLISH "(SETF (GETF ...) ...) : property list with an odd length"
  845.         FRANCAIS "(SETF (GETF ...) ...) : Occurence d'une liste de propriétés de longueur impaire."
  846.         }
  847.     ))
  848.     (when (eq (car plistr) indicator)
  849.       (rplaca (cdr plistr) value)
  850.       (return nil)
  851. ) ) )
  852. (define-setf-method getf (place indicator &optional default &environment env)
  853.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  854.     (let* ((storevar (gensym))
  855.            (indicatorvar (gensym))
  856.            (defaultvar-list (if default (list (gensym)) `()))
  857.           )
  858.       (values
  859.         `(,@SM1 ,indicatorvar ,@defaultvar-list)
  860.         `(,@SM2 ,indicator    ,@(if default `(,default) `()))
  861.         `(,storevar)
  862.         `(LET ((,(first SM3) (SYS::%PUTF ,SM5 ,indicatorvar ,storevar)))
  863.            ,@defaultvar-list ; defaultvar zum Schein auswerten
  864.            (WHEN ,(first SM3) ,SM4)
  865.            ,storevar
  866.          )
  867.         `(GETF ,SM5 ,indicatorvar ,@defaultvar-list)
  868. ) ) ) )
  869. ;-------------------------------------------------------------------------------
  870. (defsetf GETHASH (key hashtable &optional default) (value)
  871.   (let ((storeform `(SYSTEM::PUTHASH ,key ,hashtable ,value)))
  872.     (if default
  873.       `(PROGN ,default ,storeform) ; default wird nur zum Schein ausgewertet
  874.       `,storeform
  875. ) ) )
  876. ;-------------------------------------------------------------------------------
  877. #| ; siehe oben:
  878. (defun SYSTEM::%SET-DOCUMENTATION (symbol doctype value)
  879.   (unless (function-name-p symbol)
  880.     (error-of-type 'error
  881.       #L{
  882.       DEUTSCH "Das ist als erstes Argument unzulässig, da kein Symbol: ~S"
  883.       ENGLISH "first argument ~S is illegal, not a symbol"
  884.       FRANCAIS "Le premier argument ~S est invalide car ce n'est pas un symbole."
  885.       }
  886.       symbol
  887.   ) )
  888.   (setq symbol (get-funname-symbol symbol))
  889.   (if (null value)
  890.     (progn (remf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype) nil)
  891.     (setf (getf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype) value)
  892. ) )
  893. |#
  894. (defsetf documentation SYSTEM::%SET-DOCUMENTATION)
  895. ;-------------------------------------------------------------------------------
  896. (defsetf fill-pointer SYSTEM::SET-FILL-POINTER)
  897. ;-------------------------------------------------------------------------------
  898. (defsetf readtable-case SYSTEM::SET-READTABLE-CASE)
  899. ;-------------------------------------------------------------------------------
  900. (defsetf SYMBOL-VALUE SET)
  901. ;-------------------------------------------------------------------------------
  902. (defsetf SYMBOL-FUNCTION SYSTEM::%PUTD)
  903. ;-------------------------------------------------------------------------------
  904. (defsetf SYMBOL-PLIST SYSTEM::%PUTPLIST)
  905. ;-------------------------------------------------------------------------------
  906. (defun SYSTEM::SET-FDEFINITION (name value)
  907.   (setf (symbol-function (get-funname-symbol name)) value)
  908. )
  909. (defsetf FDEFINITION SYSTEM::SET-FDEFINITION)
  910. ;-------------------------------------------------------------------------------
  911. (defsetf MACRO-FUNCTION (symbol) (value)
  912.   `(PROGN
  913.      (SETF (SYMBOL-FUNCTION ,symbol) (CONS 'SYSTEM::MACRO ,value))
  914.      (REMPROP ,symbol 'SYSTEM::MACRO)
  915.      ,value
  916.    )
  917. )
  918. ;-------------------------------------------------------------------------------
  919. (defsetf CHAR SYSTEM::STORE-CHAR)
  920. (defsetf SCHAR SYSTEM::STORE-SCHAR)
  921. (defsetf BIT SYSTEM::STORE)
  922. (defsetf SBIT SYSTEM::STORE)
  923. (defsetf SUBSEQ (sequence start &optional end) (value)
  924.   `(PROGN (REPLACE ,sequence ,value :START1 ,start :END1 ,end) ,value)
  925. )
  926. ;-------------------------------------------------------------------------------
  927. (define-setf-method char-bit (char name &environment env)
  928.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method char env)
  929.     (let* ((namevar (gensym))
  930.            (storevar (gensym)))
  931.       (values `(,@SM1 ,namevar)
  932.               `(,@SM2 ,name)
  933.               `(,storevar)
  934.               `(LET ((,(first SM3) (SET-CHAR-BIT ,SM5 ,namevar ,storevar)))
  935.                  ,SM4
  936.                  ,storevar
  937.                )
  938.               `(CHAR-BIT ,SM5 ,namevar)
  939. ) ) ) )
  940. ;-------------------------------------------------------------------------------
  941. (define-setf-method LDB (bytespec integer &environment env)
  942.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method integer env)
  943.     (let* ((bytespecvar (gensym))
  944.            (storevar (gensym)))
  945.       (values (cons bytespecvar SM1)
  946.               (cons bytespec SM2)
  947.               `(,storevar)
  948.               `(LET ((,(first SM3) (DPB ,storevar ,bytespecvar ,SM5)))
  949.                  ,SM4
  950.                  ,storevar
  951.                )
  952.               `(LDB ,bytespecvar ,SM5)
  953. ) ) ) )
  954. ;-------------------------------------------------------------------------------
  955. (define-setf-method MASK-FIELD (bytespec integer &environment env)
  956.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method integer env)
  957.     (let* ((bytespecvar (gensym))
  958.            (storevar (gensym)))
  959.       (values (cons bytespecvar SM1)
  960.               (cons bytespec SM2)
  961.               `(,storevar)
  962.               `(LET ((,(first SM3) (DEPOSIT-FIELD ,storevar ,bytespecvar ,SM5)))
  963.                  ,SM4
  964.                  ,storevar
  965.                )
  966.               `(MASK-FIELD ,bytespecvar ,SM5)
  967. ) ) ) )
  968. ;-------------------------------------------------------------------------------
  969. (define-setf-method THE (type place &environment env)
  970.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  971.     (values SM1 SM2 SM3
  972.             (subst `(THE ,type ,(first SM3)) (first SM3) SM4)
  973.             `(THE ,type ,SM5)
  974. ) ) )
  975. ;-------------------------------------------------------------------------------
  976. (define-setf-method APPLY (fun &rest args &environment env)
  977.   (if (and (listp fun)
  978.            (eq (list-length fun) 2)
  979.            (eq (first fun) 'FUNCTION)
  980.            (symbolp (second fun))
  981.       )
  982.     (setq fun (second fun))
  983.     (error-of-type 'program-error
  984.       #L{
  985.       DEUTSCH "SETF von APPLY ist nur für Funktionen der Form #'symbol als Argument definiert."
  986.       ENGLISH "SETF APPLY is only defined for functions of the form #'symbol."
  987.       FRANCAIS "Un SETF de APPLY n'est défini que pour les fonctions de la forme #'symbole."
  988.       }
  989.   ) )
  990.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method (cons fun args) env)
  991.     (unless (eq (car (last args)) (car (last SM2)))
  992.       (error-of-type 'program-error
  993.         #L{
  994.         DEUTSCH "APPLY von ~S kann nicht als 'SETF-Place' aufgefaßt werden."
  995.         ENGLISH "APPLY on ~S is not a SETF place."
  996.         FRANCAIS "APPLY de ~S ne peux pas être considéré comme une place modifiable."
  997.         }
  998.         fun
  999.     ) )
  1000.     (let ((item (car (last SM1)))) ; 'item' steht für eine Argumentliste!
  1001.       (labels ((splice (arglist)
  1002.                  ; Würde man in (LIST . arglist) das 'item' nicht als 1 Element,
  1003.                  ; sondern gespliced, sozusagen als ',@item', haben wollen, so
  1004.                  ; bräuchte man die Form, die (splice arglist) liefert.
  1005.                  (if (endp arglist)
  1006.                    'NIL
  1007.                    (let ((rest (splice (cdr arglist))))
  1008.                      (if (eql (car arglist) item)
  1009.                        ; ein (APPEND item ...) davorhängen, wie bei Backquote
  1010.                        (backquote-append item rest)
  1011.                        ; ein (CONS (car arglist) ...) davorhängen, wie bei Backquote
  1012.                        (backquote-cons (car arglist) rest)
  1013.               )) ) ) )
  1014.         (flet ((call-splicing (form)
  1015.                  ; ersetzt einen Funktionsaufruf form durch einen, bei dem
  1016.                  ; 'item' nicht 1 Argument, sondern eine Argumentliste liefert
  1017.                  (let ((fun (first form))
  1018.                        (argform (splice (rest form))))
  1019.                    ; (APPLY #'fun argform) vereinfachen:
  1020.                    ; (APPLY #'fun NIL) --> (fun)
  1021.                    ; (APPLY #'fun (LIST ...)) --> (fun ...)
  1022.                    ; (APPLY #'fun (CONS x y)) --> (APPLY #'fun x y)
  1023.                    ; (APPLY #'fun (LIST* ... z)) --> (APPLY #'fun ... z)
  1024.                    (if (or (null argform)
  1025.                            (and (consp argform) (eq (car argform) 'LIST))
  1026.                        )
  1027.                      (cons fun (cdr argform))
  1028.                      (list* 'APPLY
  1029.                             (list 'FUNCTION fun)
  1030.                             (if (and (consp argform)
  1031.                                      (or (eq (car argform) 'LIST*)
  1032.                                          (eq (car argform) 'CONS)
  1033.                                 )    )
  1034.                               (cdr argform)
  1035.                               (list argform)
  1036.               )) ) ) )      )
  1037.           (values SM1 SM2 SM3 (call-splicing SM4) (call-splicing SM5))
  1038. ) ) ) ) )
  1039. ;-------------------------------------------------------------------------------
  1040. ; Zusätzliche Definitionen von places
  1041. ;-------------------------------------------------------------------------------
  1042. (define-setf-method funcall (fun &rest args &environment env)
  1043.   (unless (and (listp fun)
  1044.                (eq (list-length fun) 2)
  1045.                (let ((fun1 (first fun)))
  1046.                  (or (eq fun1 'FUNCTION) (eq fun1 'QUOTE))
  1047.                )
  1048.                (symbolp (second fun))
  1049.                (setq fun (second fun))
  1050.           )
  1051.     (error-of-type 'program-error
  1052.       #L{
  1053.       DEUTSCH "SETF von FUNCALL ist nur für Funktionen der Form #'symbol definiert."
  1054.       ENGLISH "SETF FUNCALL is only defined for functions of the form #'symbol."
  1055.       FRANCAIS "Un SETF de FUNCALL n'est défini que pour les fonctions de la forme #'symbole."
  1056.       }
  1057.   ) )
  1058.   (get-setf-method (cons fun args) env)
  1059. )
  1060. ;-------------------------------------------------------------------------------
  1061. (defsetf GET-DISPATCH-MACRO-CHARACTER
  1062.          (disp-char sub-char &optional (readtable '*READTABLE*)) (value)
  1063.   `(PROGN (SET-DISPATCH-MACRO-CHARACTER ,disp-char ,sub-char ,value ,readtable) ,value)
  1064. )
  1065. ;-------------------------------------------------------------------------------
  1066. (defsetf long-float-digits SYSTEM::%SET-LONG-FLOAT-DIGITS)
  1067. ;-------------------------------------------------------------------------------
  1068. (defsetf DEFAULT-DIRECTORY () (value)
  1069.   `(PROGN (CD ,value) ,value)
  1070. )
  1071. ;-------------------------------------------------------------------------------
  1072. #+LOGICAL-PATHNAMES
  1073. (defsetf logical-pathname-translations set-logical-pathname-translations)
  1074. ;-------------------------------------------------------------------------------
  1075. ; Handhabung von (SETF (VALUES place1 ... placek) form)
  1076. ; --> (MULTIPLE-VALUE-BIND (dummy1 ... dummyk) form
  1077. ;       (SETF place1 dummy1 ... placek dummyk)
  1078. ;       (VALUES dummy1 ... dummyk)
  1079. ;     )
  1080. (define-setf-method VALUES (&rest subplaces &environment env)
  1081.   (multiple-value-bind (temps vals stores storeforms accessforms)
  1082.       (setf-VALUES-aux subplaces env)
  1083.     (values temps
  1084.             vals
  1085.             stores
  1086.             `(VALUES ,@storeforms)
  1087.             `(VALUES ,@accessforms)
  1088. ) ) )
  1089. (defun setf-VALUES-aux (places env)
  1090.   (do ((temps nil)
  1091.        (vals nil)
  1092.        (stores nil)
  1093.        (storeforms nil)
  1094.        (accessforms nil)
  1095.        (placesr places))
  1096.       ((atom placesr)
  1097.        (setq temps (nreverse temps))
  1098.        (setq vals (nreverse vals))
  1099.        (setq stores (nreverse stores))
  1100.        (setq storeforms (nreverse storeforms))
  1101.        (setq accessforms (nreverse accessforms))
  1102.        (values temps vals stores storeforms accessforms)
  1103.       )
  1104.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  1105.         (get-setf-method (pop placesr) env)
  1106.       (setq temps (revappend SM1 temps))
  1107.       (setq vals (revappend SM2 vals))
  1108.       (setq stores (revappend SM3 stores))
  1109.       (setq storeforms (cons SM4 storeforms))
  1110.       (setq accessforms (cons SM5 accessforms))
  1111. ) ) )
  1112. ;-------------------------------------------------------------------------------
  1113. ; Analog zu (MULTIPLE-VALUE-SETQ (var1 ... vark) form) :
  1114. ; (MULTIPLE-VALUE-SETF (place1 ... placek) form)
  1115. ; --> (VALUES (SETF (VALUES place1 ... placek) form))
  1116. ; --> (MULTIPLE-VALUE-BIND (dummy1 ... dummyk) form
  1117. ;       (SETF place1 dummy1 ... placek dummyk)
  1118. ;       dummy1
  1119. ;     )
  1120. (defmacro multiple-value-setf (places form &environment env)
  1121.   (multiple-value-bind (temps vals stores storeforms accessforms)
  1122.       (setf-VALUES-aux places env)
  1123.     (declare (ignore accessforms))
  1124.     `(LET* ,(mapcar #'list temps vals)
  1125.        (MULTIPLE-VALUE-BIND ,stores ,form
  1126.          ,@storeforms
  1127.          ,(first stores) ; (null stores) -> NIL -> Wert NIL
  1128.      ) )
  1129. ) )
  1130. ;-------------------------------------------------------------------------------
  1131.  
  1132.