summaryrefslogtreecommitdiffstats
path: root/stdlib/ifa.tl
diff options
context:
space:
mode:
Diffstat (limited to 'stdlib/ifa.tl')
-rw-r--r--stdlib/ifa.tl82
1 files changed, 82 insertions, 0 deletions
diff --git a/stdlib/ifa.tl b/stdlib/ifa.tl
new file mode 100644
index 00000000..f643cf92
--- /dev/null
+++ b/stdlib/ifa.tl
@@ -0,0 +1,82 @@
+;; Copyright 2015-2021
+;; Kaz Kylheku <kaz@kylheku.com>
+;; Vancouver, Canada
+;; All rights reserved.
+;;
+;; Redistribution and use in source and binary forms, with or without
+;; modification, are permitted provided that the following conditions are met:
+;;
+;; 1. Redistributions of source code must retain the above copyright notice, this
+;; list of conditions and the following disclaimer.
+;;
+;; 2. Redistributions in binary form must reproduce the above copyright notice,
+;; this list of conditions and the following disclaimer in the documentation
+;; and/or other materials provided with the distribution.
+;;
+;; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
+;; ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+;; WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+;; DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+;; FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+;; DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+;; SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+;; CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+;; OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+;; OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+(defmacro ifa (:env e :form f test then : else)
+ (flet ((candidate-p (form)
+ (not (or (constantp form e) (symbolp form)))))
+ (cond
+ ((or (atom test) (null (cdr test))) ^(let ((it ,test))
+ (if it ,then ,else)))
+ ((member (first test) '(not null false))
+ (unless (eql (length test) 2)
+ (compile-error f "wrong number of arguments to ~s" (first test)))
+ ^(ifa ,(second test) ,else ,then))
+ (t (let* ((sym (first test))
+ (args (if (eq 'dwim sym) (cddr test) (cdr test)))
+ (n-candidate-args [count-if candidate-p args])
+ (pos-candidate (or [pos-if candidate-p args] 0)))
+ (unless (or (lexical-fun-p e sym)
+ (and (or (functionp (symbol-function sym))
+ (eq sym 'dwim)
+ (null (symbol-function sym)))))
+ (compile-error f "test expression must be \
+ \ a simple function call"))
+ (when (> n-candidate-args 1)
+ (compile-error f "ambiguous situation: \
+ \ not clear what can be \"it\""))
+ (iflet ((it-form (macroexpand [args pos-candidate] e))
+ (is-place (place-form-p it-form e)))
+ (let ((before-it [args 0..pos-candidate])
+ (after-it [args (succ pos-candidate)..:]))
+ (let* ((btemps (mapcar (ret (gensym)) before-it))
+ (atemps (mapcar (ret (gensym)) after-it)))
+ ^(let (,*(zip btemps before-it))
+ (placelet ((it ,it-form))
+ (let (,*(zip atemps after-it))
+ (if (,sym ,*(if (eq 'dwim sym) ^(,(second test)))
+ ,*btemps it ,*atemps)
+ ,then ,else))))))
+ (let* ((temps (mapcar (ret (gensym)) args))
+ (it-temp [temps pos-candidate]))
+ ^(let* (,*(zip temps args) (it ,it-temp))
+ (if (,sym ,*(if (eq 'dwim sym) ^(,(second test)))
+ ,*temps) ,then ,else)))))))))
+
+(defmacro whena (test . body)
+ ^(ifa ,test (progn ,*body)))
+
+(defun sys:if-to-cond (f if-oper cond-oper pairs)
+ (tree-case pairs
+ (((test . forms) . rest) ^(,if-oper ,test (progn ,*forms)
+ (,cond-oper ,*rest)))
+ (() ())
+ (else (compile-error f "bad syntax: ~s" pairs))))
+
+(defmacro conda (:form f . pairs)
+ (sys:if-to-cond f 'ifa 'conda pairs))
+
+(defmacro condlet (:form f . pairs)
+ (sys:if-to-cond f 'iflet 'condlet pairs))