summaryrefslogtreecommitdiffstats
path: root/stdlib
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-08-13 06:25:50 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-08-13 06:25:50 -0700
commit483e15d9af31110d2a34ad20fe010663de5afe19 (patch)
tree6721ce5ccadde9b844ec674bff3145941fa0da17 /stdlib
parent3d7e614022c19d0c42ff32d60c8e51b226fb427d (diff)
downloadtxr-483e15d9af31110d2a34ad20fe010663de5afe19.tar.gz
txr-483e15d9af31110d2a34ad20fe010663de5afe19.tar.bz2
txr-483e15d9af31110d2a34ad20fe010663de5afe19.zip
matcher: new must-match and must-match-case macros.
* lisplib.c (match_set_entries): Intern the match-error symbol. Register autoloads for must-match and must-match-case. * stdlib/match.tl (match-error): Register exception symbol, as subtype of match-error. (must-match, must-match-case): New macros. * tests/011/patmatch.tl: Test cases. * txr.1: Documented.
Diffstat (limited to 'stdlib')
-rw-r--r--stdlib/match.tl15
1 files changed, 15 insertions, 0 deletions
diff --git a/stdlib/match.tl b/stdlib/match.tl
index 3502688b..d122d24c 100644
--- a/stdlib/match.tl
+++ b/stdlib/match.tl
@@ -28,6 +28,8 @@
(defvar *match-macro* (hash))
+(defex match-error eval-error)
+
(defstruct match-guard ()
temps
vars
@@ -643,6 +645,13 @@
,result
,else)))))
+(defmacro must-match (pat obj . body)
+ (with-gensyms (val)
+ ^(let ((,val ,obj))
+ (if-match ,pat ,val
+ (progn ,*body)
+ (throwf 'match-error "~s: ~s failed to match object ~s" 'must-match ',pat ,val)))))
+
(defmacro while-match (:form *match-form* :env e pat obj . body)
(let ((cm (compile-match pat : (get-var-list e))))
^(for ()
@@ -674,6 +683,12 @@
(or ,*clause-code)
,result-temp))))
+(defmacro must-match-case (obj . clauses)
+ (with-gensyms (else)
+ ^(match-case ,obj
+ ,*clauses
+ ((var ,else) (throwf 'match-error "~s: failed to match object ~s" 'must-match-case ,else)))))
+
(defmacro while-match-case (:form *match-form* :env e obj . clauses)
(unless [all clauses [andf proper-listp [chain len plusp]]]
(compile-error *match-form* "bad clause syntax"))