diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-07-10 21:45:26 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-07-10 21:45:26 -0700 |
commit | 5b5c9770337e3f3e042f3cfe8a3e0edd6109c548 (patch) | |
tree | 05fb45d7d0f9fad1b431631d9f8475b60584abb1 /share | |
parent | 0c15b355ad3a62e13e27d752047b4566da97153f (diff) | |
download | txr-5b5c9770337e3f3e042f3cfe8a3e0edd6109c548.tar.gz txr-5b5c9770337e3f3e042f3cfe8a3e0edd6109c548.tar.bz2 txr-5b5c9770337e3f3e042f3cfe8a3e0edd6109c548.zip |
New placelet macro.
* lisplib.c (place_set_entries): Add placelet to list of names.
* share/txr/stdlib/place.tl (sys:placelet1, placelet): New macros.
(defplace dwim): Do not retrieve the
place's value into a local variable and have the getter
expand to that variable. Rather, have the getter retrieve the
value. A getter that refers to a cached copy breaks
the semantics of placelet, and any place operator which can
evaluate the location after it is stored.
* txr.1: Documented placelet.
Diffstat (limited to 'share')
-rw-r--r-- | share/txr/stdlib/place.tl | 45 |
1 files changed, 34 insertions, 11 deletions
diff --git a/share/txr/stdlib/place.tl b/share/txr/stdlib/place.tl index 72e054fd..af03070f 100644 --- a/share/txr/stdlib/place.tl +++ b/share/txr/stdlib/place.tl @@ -421,17 +421,16 @@ (,index-sym (sys:l1-val ,index)) ,*(if have-default-p ^((,dflval-sym (sys:l1-val ,default))))) - (let ((,oldval-sym [,obj-sym - ,index-sym - ,*(if have-default-p ^(,dflval-sym))])) - (macrolet ((,getter () ',oldval-sym) - (,setter (val) - ^(rlet ((,',newval-sym ,val)) - (,',osetter-sym - (sys:dwim-set ,',obj-sym - ,',index-sym ,',newval-sym)) - ,',newval-sym))) - ,body))))))) + (macrolet ((,getter () + '[,obj-sym ,index-sym + ,*(if have-default-p ^(,dflval-sym))]) + (,setter (val) + ^(rlet ((,',newval-sym ,val)) + (,',osetter-sym + (sys:dwim-set ,',obj-sym + ,',index-sym ,',newval-sym)) + ,',newval-sym))) + ,body)))))) (ssetter (with-gensyms (osetter-sym ogetter-sym obj-sym newval-sym index-sym) @@ -560,3 +559,27 @@ ^(defmacro ,name (:env env ,place-sym ,*lambda-list) (with-update-expander (getter setter) ,place-sym env ^(,setter (,',function (,getter) ,,*cleaned-lambda-list))))))) + +(defmacro sys:placelet-1 (((sym place)) . body :env env) + (with-gensyms (tmp-place pl-getter pl-setter steal-getter) + (unwind-protect + (progn + (sethash *place-update-expander* tmp-place + (lambda (tmp-getter tmp-setter tmp-place tmp-body) + ^(macrolet ((,tmp-getter () ^(,',pl-getter)) + (,tmp-setter (val) ^(,',pl-setter ,val))) + ,tmp-body))) + (call-update-expander pl-getter pl-setter place env + ^(macrolet ((,tmp-place () ^(,',pl-getter))) + ,(sys:expand + ^(symacrolet ((,sym (,tmp-place))) ,*body) + env)))) + (remhash *place-update-expander* tmp-place)))) + +(defmacro placelet (sym-place-pairs . body) + (tree-case sym-place-pairs + (() ^(progn ,*body)) + (((sym place)) ^(sys:placelet-1 ((,sym ,place)) ,*body)) + (((sym place) . rest-pairs) ^(sys:placelet-1 ((,sym ,place)) + (placelet (,*rest-pairs) ,*body))) + (obj (throwf 'eval-error "placelet: bad syntax: ~s" obj)))) |