summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-06-10 07:39:55 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-06-10 07:39:55 -0700
commit5761d2e6c82f5234d4e63a49bc0a59ab69bbf0ed (patch)
treec3846797884f74c0958c3ce3005395c08de9178e
parente4fe68098144ea52eb9d4b35c381988dbf387be2 (diff)
downloadtxr-5761d2e6c82f5234d4e63a49bc0a59ab69bbf0ed.tar.gz
txr-5761d2e6c82f5234d4e63a49bc0a59ab69bbf0ed.tar.bz2
txr-5761d2e6c82f5234d4e63a49bc0a59ab69bbf0ed.zip
New macro: pic.
* lisplib.c (pic_instantiate, pic_set_entries): New static functions. (lisplib_init): Register autoloading of pic.tl module via new functions. * share/txr/stdlib/pic.tl: New file.
-rw-r--r--lisplib.c18
-rw-r--r--share/txr/stdlib/pic.tl107
2 files changed, 125 insertions, 0 deletions
diff --git a/lisplib.c b/lisplib.c
index 21c1ac59..90bdc987 100644
--- a/lisplib.c
+++ b/lisplib.c
@@ -918,6 +918,23 @@ static val doc_set_entries(val dlt, val fun)
return nil;
}
+static val pic_instantiate(val set_fun)
+{
+ funcall1(set_fun, nil);
+ load(scat2(stdlib_path, lit("pic")));
+ return nil;
+}
+
+static val pic_set_entries(val dlt, val fun)
+{
+ val name[] = {
+ lit("pic"),
+ nil
+ };
+ set_dlt_entries(dlt, name, fun);
+ return nil;
+}
+
val dlt_register(val dlt,
val (*instantiate)(val),
val (*set_entries)(val, val))
@@ -974,6 +991,7 @@ void lisplib_init(void)
dlt_register(dl_table, quips_instantiate, quips_set_entries);
dlt_register(dl_table, match_instantiate, match_set_entries);
dlt_register(dl_table, doc_instantiate, doc_set_entries);
+ dlt_register(dl_table, pic_instantiate, pic_set_entries);
reg_fun(intern(lit("try-load"), system_package), func_n1(lisplib_try_load));
}
diff --git a/share/txr/stdlib/pic.tl b/share/txr/stdlib/pic.tl
new file mode 100644
index 00000000..1c28e0df
--- /dev/null
+++ b/share/txr/stdlib/pic.tl
@@ -0,0 +1,107 @@
+;; Copyright 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.
+
+(defun expand-pic-num (fmt val)
+ (let* ((zero (or (starts-with "0" fmt)
+ (starts-with "+0" fmt)
+ (starts-with "-0" fmt)))
+ (plus (eql [fmt 0] #\+))
+ (minus (eql [fmt 0] #\-))
+ (exc (pos #\! fmt))
+ (dot (or exc (pos #\. fmt)))
+ (int (if dot [fmt 0..dot] fmt))
+ (fra (if dot [fmt (succ dot)..:] "")))
+ (let ((code (if (or minus plus)
+ ^(fmt ,`~@(len fmt),@(if plus "+")@(if zero "0")@(len fra)f`
+ ,val)
+ (with-gensyms (vg)
+ ^(let ((,vg ,val))
+ (if (minusp ,vg)
+ (fmt ,`~@(len fmt),@(if zero "0")@(len fra)f`
+ ,vg)
+ (rest (fmt ,`~@(succ (len fmt)),@(if zero "0")@(len fra)f`
+ ,vg))))))))
+ (if exc
+ (with-gensyms (str)
+ ^(let ((,str ,code))
+ (if (> (len ,str) ,(len fmt))
+ (mkstring ,(len fmt) #\#)
+ ,str)))
+ code))))
+
+(defun expand-pic-align (chr fmt val)
+ ^(fmt ,`~@(if chr chr)@(len fmt)a` ,val))
+
+(defun pic-join-opt (join-form)
+ (labels ((et (str) (regsub #/\~/ "~~" str)))
+ (match-case join-form
+ ((join @(stringp @s) (fmt `@fmt` . @args) . @rest)
+ (pic-join-opt ^(join (fmt ,`@(et s)@fmt` ,*args) ,*rest)))
+ ((join (fmt `@fmt` . @args) @(stringp @s) . @rest)
+ (pic-join-opt ^(join (fmt ,`@fmt@(et s)` ,*args) ,*rest)))
+ ((join (fmt `@fmt1` . @args1) (fmt `@fmt2` . @args2) . @rest)
+ (pic-join-opt ^(join (fmt ,`@fmt1@fmt2` ,*args1 ,*args2) ,*rest)))
+ ((join @(stringp @s1) @(stringp @s2) . @rest)
+ (pic-join-opt ^(join ,`@s1@s1` ,rest)))
+ ((join "" @item . @rest)
+ (pic-join-opt ^(join ,item ,*rest)))
+ ((join @item "" . @rest)
+ (pic-join-opt ^(join ,item ,*rest)))
+ ((join @item) item)
+ (@else else))))
+
+(defun expand-pic (f fmt val)
+ (unless (stringp fmt)
+ (compile-error f "~s is required to be a format string" fmt))
+ (cond
+ ([m^$ #/\~[~#<>\|\-+0.!]/ fmt] [fmt 1])
+ ([m^$ #/[+\-]?0?#+([.!]#+)?/ fmt] (expand-pic-num fmt val))
+ ([m^$ #/<+/ fmt] (expand-pic-align "<" fmt val))
+ ([m^$ #/>+/ fmt] (expand-pic-align nil fmt val))
+ ([m^$ #/\|+/ fmt] (expand-pic-align "^" fmt val))
+ (t (compile-error f "unrecognized format string ~s" fmt))))
+
+(defmacro pic (:form f :env e bigfmt . args)
+ (unless (stringp bigfmt)
+ (compile-error f "~s is required to be a format string" bigfmt))
+ (let* ((regex #/[+\-]?0?#+([.!]#+)?| \
+ <+| \
+ >+| \
+ \|+| \
+ \~[~#<>\|\-+0.!]/)
+ (items (collect-each ((piece (tok regex t bigfmt)))
+ (cond
+ ((m^$ regex piece)
+ (cond
+ ((starts-with "~" piece)
+ (expand-pic f piece nil))
+ (t (unless args
+ (compile-error f "insufficient arguments for format"))
+ (expand-pic f piece (pop args)))))
+ (t piece)))))
+ (if args
+ (compile-warning f "excess arguments"))
+ (pic-join-opt ^(join ,*items))))