summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-01-30 23:11:47 -0800
committerKaz Kylheku <kaz@kylheku.com>2021-01-30 23:11:47 -0800
commit7b26a8a95f544c04c20af9fdeac16624685a15b6 (patch)
tree9b86323c6f6323d914fa0e69a3624db9bd97bc7e
parentfd30d22a593b3e5a07855db6473241a0c8ced99c (diff)
downloadtxr-7b26a8a95f544c04c20af9fdeac16624685a15b6.tar.gz
txr-7b26a8a95f544c04c20af9fdeac16624685a15b6.tar.bz2
txr-7b26a8a95f544c04c20af9fdeac16624685a15b6.zip
@(rebind): bugfix: don't clobber right side variable.
* Makefile (tst/tests/000/binding.ok): Pass -B to txr for this new test. * match.c (v_rebind): Fix gaping copy-and-paste bug here, which causes rebind to take on the behavior of local/forget; it takes all symbols that appear as its arguments from the environment and produces an environment in which they don't exist. What we want is to remove the left variables from the environment, and since that is a nested pattern, the right way to do that is to flatten it. Bug reported by Frank Schwidom. * tests/000/binding.txr: New file. * tests/000/binding.expected: New file. * txr.1: Improve documentation of @(rebind), also making improvements in @(set) documentation.
-rw-r--r--Makefile1
-rw-r--r--match.c2
-rw-r--r--tests/000/binding.expected6
-rw-r--r--tests/000/binding.txr5
-rw-r--r--txr.176
5 files changed, 79 insertions, 11 deletions
diff --git a/Makefile b/Makefile
index 1311dab7..a7d71143 100644
--- a/Makefile
+++ b/Makefile
@@ -370,6 +370,7 @@ TESTS_OK := $(addprefix tst/,\
tests: $(TESTS_OK)
$(V)echo "** tests passed!"
+tst/tests/000/binding.ok: TXR_OPTS := -B
tst/tests/001/%: TXR_ARGS := tests/001/data
tst/tests/001/query-1.ok: TXR_OPTS := -B
tst/tests/001/query-2.ok: TXR_OPTS := -B
diff --git a/match.c b/match.c
index 3846060d..1010062d 100644
--- a/match.c
+++ b/match.c
@@ -3707,7 +3707,7 @@ static val v_rebind(match_files_ctx *c)
val form = second(args);
val val = txeval(specline, form, c->bindings);
- c->bindings = alist_remove(c->bindings, args);
+ c->bindings = alist_remove(c->bindings, flatten(pattern));
c->bindings = dest_bind(specline, c->bindings,
pattern, val, equal_f);
diff --git a/tests/000/binding.expected b/tests/000/binding.expected
new file mode 100644
index 00000000..9e5303f0
--- /dev/null
+++ b/tests/000/binding.expected
@@ -0,0 +1,6 @@
+x="1"
+y="2"
+z[0]="1"
+z[1]="2"
+a="1"
+b="0"
diff --git a/tests/000/binding.txr b/tests/000/binding.txr
new file mode 100644
index 00000000..95faa20d
--- /dev/null
+++ b/tests/000/binding.txr
@@ -0,0 +1,5 @@
+@(bind x 1)
+@(bind y 2)
+@(rebind z (x y))
+@(bind (a b) (0 1))
+@(rebind (a b) (b a))
diff --git a/txr.1 b/txr.1
index 619b87f8..6a87b6ac 100644
--- a/txr.1
+++ b/txr.1
@@ -6857,6 +6857,14 @@ produce equal values.
.dir set
+The syntax of the
+.code set
+directive is:
+
+.mono
+.mets @(set < pattern << bind-expression )
+.onom
+
The
.code set
directive syntactically resembles
@@ -6939,21 +6947,45 @@ Because it is preceded by the
.code @
escape, it is a Lisp variable, and not a pattern variable.
+The
+.code set
+directive also doesn't support Lisp expressions in the
+.metn pattern ,
+which must consist only of variables.
+
.dir rebind
+The syntax of the
+.code rebind
+directive is:
+
+.mono
+.mets @(rebind < pattern << bind-expression )
+.onom
+
The
.code rebind
directive resembles
-.code set
-but it is not an assignment.
+.codn bind .
It combines the semantics of
-.codn local ,
-.code bind
+.code local
and
-.codn set .
-The expression on the right hand side is evaluated in the current
-environment. Then the variables in the pattern on the left are introduced
-as new bindings, whose values come from the pattern.
+.code bind
+into a single directive.
+The
+.meta bind-expression
+is evaluated in the current
+environment, and its value remembered. Then a new
+environment is produced in which all the variables specified in
+.meta pattern
+are absent. Then, the pattern is newly bound in
+that environment against the previously produced value, as if using
+.codn bind .
+
+The old environment with the previous variables is not modified;
+it continues to exist. This is in contrast with the
+.code set
+directive, which mutates existing bindings.
.code rebind
makes it easy to create temporary bindings based on existing bindings.
@@ -6967,8 +6999,8 @@ makes it easy to create temporary bindings based on existing bindings.
.brev
When the function terminates, the previous value of recursion-level
-is restored. The effect is like the following, but much easier
-to write and faster to execute:
+is restored. The effect is less verbose and more efficient than
+the following equivalent
.verb
@(define pattern-function (arg))
@@ -6981,6 +7013,30 @@ to write and faster to execute:
@(end)
.brev
+Like
+.codn bind ,
+.code rebind
+supports nested patterns, such as
+
+.verb
+ @(rebind (a (b c)) (1 (2 3))
+.brev
+
+but it does not support any keyword arguments. The filtering
+features of
+.code bind
+do not make sense in
+.code rebind
+because the variables are always re-introduced into an environment
+in which they don't exist, whereas filtering applies in
+situations when bound variables are matched against values.
+
+The
+.code rebind
+directive also doesn't support Lisp expressions in the
+.metn pattern ,
+which must consist only of variables.
+
.dir forget
The