summaryrefslogtreecommitdiffstats
path: root/stdlib
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-06-30 06:36:44 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-06-30 06:36:44 -0700
commit09e70c914ca83b5c7405aa633864db49f27efa05 (patch)
tree02fc78c949dbfa6364a4da4115ee6c79fa643e39 /stdlib
parent0de65a6edc3631ce50ad5f5082ff82a45be867d0 (diff)
downloadtxr-09e70c914ca83b5c7405aa633864db49f27efa05.tar.gz
txr-09e70c914ca83b5c7405aa633864db49f27efa05.tar.bz2
txr-09e70c914ca83b5c7405aa633864db49f27efa05.zip
op: refactor do handling.
* stdlib/op.tl (op-expand): The logic involving the multiple expansions of do is consolidated into one step, so that everything is taken care of by the time syntax-1 is produced. Immediately when the initial unsuffixed arguments expand successfully, we check or the presence of metas and add the @1 if metas are absent. In the case of expanding with the gensym, we immediately replace that with @1 without using symacrolet. The do-gen variable is gone. Later, when calculating lambda-interior, there are no hacky additional passes to deal with; that block of code just refers to syntax-2. The do case is folded together with the have-metas case. Also adding copious comments to this hard-to-follow logic.
Diffstat (limited to 'stdlib')
-rw-r--r--stdlib/op.tl56
1 files changed, 37 insertions, 19 deletions
diff --git a/stdlib/op.tl b/stdlib/op.tl
index 06e87633..751c56e5 100644
--- a/stdlib/op.tl
+++ b/stdlib/op.tl
@@ -99,21 +99,52 @@
['compile-error f "arguments required"])
(let* ((compat (and (plusp sys:compat) (<= sys:compat 225)))
(ctx (make-struct 'sys:op-ctx ^(form ,f)))
- (do-gen)
(sys:*op-ctx* ctx)
(sym (car f))
(syntax-0 (if (eq sym 'do) args ^[,*args]))
(syntax-1 (if (or (null syntax-0) (neq sym 'do) compat)
+ ;; not do, or empty do syntax, or compat mode.
(sys:op-alpha-rename f e syntax-0 nil)
- (or (op-ignerr (sys:op-alpha-rename f e syntax-0 nil))
+ ;; try to expand args as-is, catching errors
+ (let ((syn (op-ignerr (sys:op-alpha-rename f e
+ syntax-0
+ nil))))
+ (if syn
+ ;; args expanded
+ (if (slot ctx 'gens)
+ ;; there are metas: okay, use expansion as-is.
+ syn
+ ;; no metas: add @1 at the end and expand
+ ;; again, without catching errors.
+ (sys:op-alpha-rename f e
+ (append syntax-0
+ '(@1))
+ nil))
+ ;; args didn't expand, thus need the extra
+ ;; element. But we can't add a @1 because by
+ ;; doing so we could be introducing metas.
+ ;; We add a gensym instead to try to satisfy
+ ;; the syntax without adding metas. If all works,
+ ;; we can use a symacrolet pass to replace
+ ;; that gensym with @1.
(let ((syn (sys:op-alpha-rename
f e (append syntax-0
- (list (sys:setq do-gen
- (gensym))))
+ (list (gensym)))
nil)))
+ ;; It didn't blow up with the gensym. But
+ ;; if there are metas, we don't want to be
+ ;; be adding this gensym. We know that the
+ ;; form does not expand without the gensym.
+ ;; So we repeat that expansion, but this time
+ ;; without op-ignerr. This will flush out
+ ;; the error.
(when (slot ctx 'gens)
(sys:op-alpha-rename f e syntax-0 nil))
- syn))))
+ ;; There were no metas. OK, let's augment
+ ;; syntax-0 with @1 instead of the gensym.
+ (sys:op-alpha-rename f e
+ (append syntax-0 '(@1))
+ nil))))))
(syntax-2 (sys:op-alpha-rename f e syntax-1 t))
(metas (slot ctx 'gens))
(rec (slot ctx 'rec))
@@ -129,20 +160,7 @@
;; no cadr here to avoid circular autoload
^[sys:apply ,(car (cdr syntax-2))
(append ,rest-sym (list ,*fargs-l1))]))
- (metas syntax-2)
- ((eq sym 'do)
- (cond
- (compat syntax-2)
- (do-gen
- (let ((arg1 (sys:ensure-op-arg ctx 1)))
- ^(symacrolet ((,do-gen ,arg1))
- ,syntax-2)))
- (t (let* ((arg1 (sys:ensure-op-arg ctx 1))
- (syntax-0-alt (append args (list arg1)))
- (syntax-1-alt (sys:op-alpha-rename f e
- syntax-0-alt
- nil)))
- (sys:op-alpha-rename f e syntax-1-alt t)))))
+ ((or metas (eq sym 'do)) syntax-2)
(t (append syntax-2 rest-sym))))))
(let ((metas (slot ctx 'gens)))
(cond