summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--ChangeLog22
-rw-r--r--Makefile12
-rw-r--r--eval.c5
-rw-r--r--eval.h1
-rw-r--r--lisplib.c16
-rw-r--r--share/txr/stdlib/ifa.tl44
-rw-r--r--txr.c7
-rw-r--r--txr.h1
8 files changed, 103 insertions, 5 deletions
diff --git a/ChangeLog b/ChangeLog
index a1400ee3..92fa7aaa 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,27 @@
2015-06-17 Kaz Kylheku <kaz@kylheku.com>
+ Adding anaphoric ifa macro.
+
+ * Makefile (install): Install .tl files present in stdlib directory.
+ (INSTALL): Handle argument 2 being a list.
+
+ * eval.c (load): New function.
+
+ * eval.h (load): Declared.
+
+ * lisplib.c (ifa_set_entries, ifa_instantiate): New functions
+ to lazily load ifa.tl.
+ (lisplib_init): Register new functions.
+
+ * txr.c (stdlib_path): New variable.
+ (sysroot_init): Store the stdlib path in stdlib_path.
+
+ * txr.h (stdlib_path): Declared.
+
+ * share/txr/stdlib/ifa.tl: New file.
+
+2015-06-17 Kaz Kylheku <kaz@kylheku.com>
+
Refactoring of lisplib.c
* lisplib.c (set_dlt_entries): New static function.
diff --git a/Makefile b/Makefile
index a25370fe..31d11f2f 100644
--- a/Makefile
+++ b/Makefile
@@ -345,8 +345,11 @@ define INSTALL
$(call ABBREV3,INSTALL,$(3),$(2))
$(V)mkdir -p $(3)
$(V)cp -f $(2) $(3)
- $(V)chmod $(1) $(3)/$(notdir $(2))
- $(V)for x in $(2) ; do touch -r $$x $(3)/$$(basename $$x) ; done
+ $(V)for x in $(2) ; do \
+ y=$(strip $(3))/$$(basename $$x) ; \
+ touch -r $$x $$y ; \
+ chmod $(1) $$y ; \
+ done
endef
PREINSTALL := :
@@ -358,7 +361,10 @@ install: $(PROG)
$(call INSTALL,0444,$(top_srcdir)/LICENSE,$(DESTDIR)$(datadir))
$(call INSTALL,0444,$(top_srcdir)/METALICENSE,$(DESTDIR)$(datadir))
$(call INSTALL,0444,$(top_srcdir)/txr.1,$(DESTDIR)$(mandir)/man1)
- $(call INSTALL,0444,$(top_srcdir)/share/txr/stdlib/*.txr,$(DESTDIR)$(datadir)/stdlib)
+ $(call INSTALL,0444,\
+ $(addprefix $(top_srcdir)/share/txr/stdlib/,\
+ *.txr *.tl),\
+ $(DESTDIR)$(datadir)/stdlib)
.PHONY: unixtar gnutar zip
diff --git a/eval.c b/eval.c
index 482a26bb..41b33865 100644
--- a/eval.c
+++ b/eval.c
@@ -2855,6 +2855,11 @@ static val me_load(val form, val menv)
return list(sys_load_s, name, list(quote_s, source_loc(form), nao), nao);
}
+val load(val target)
+{
+ return sys_load(target, nil);
+}
+
static val expand_catch_clause(val form, val menv)
{
val sym = first(form);
diff --git a/eval.h b/eval.h
index 0f4b6fd7..b75ea3f0 100644
--- a/eval.h
+++ b/eval.h
@@ -43,6 +43,7 @@ val apply_intrinsic(val fun, val args);
val eval_progn(val forms, val env, val ctx_form);
val eval(val form, val env, val ctx_form);
val eval_intrinsic(val form, val env);
+val load(val target);
val expand(val form, val menv);
val expand_forms(val forms, val menv);
val bindable(val obj);
diff --git a/lisplib.c b/lisplib.c
index b5de949e..b6a15203 100644
--- a/lisplib.c
+++ b/lisplib.c
@@ -36,6 +36,7 @@
#include "hash.h"
#include "gc.h"
#include "place.h"
+#include "txr.h"
#include "lisplib.h"
static val dl_table;
@@ -82,6 +83,20 @@ static val place_instantiate(val set_fun)
colon_k, lit("place.tl")), nil);
}
+static val ifa_set_entries(val dlt, val fun)
+{
+ val name[] = { lit("ifa"), nil };
+ set_dlt_entries(dlt, name, fun);
+ return nil;
+}
+
+static val ifa_instantiate(val set_fun)
+{
+ funcall1(set_fun, nil);
+ load(format(nil, lit("~a/ifa.tl"), stdlib_path, nao));
+ return nil;
+}
+
static val dlt_register(val dlt,
val (*instantiate)(val),
val (*set_entries)(val, val))
@@ -94,6 +109,7 @@ void lisplib_init(void)
prot1(&dl_table);
dl_table = make_hash(nil, nil, nil);
dlt_register(dl_table, place_instantiate, place_set_entries);
+ dlt_register(dl_table, ifa_instantiate, ifa_set_entries);
}
val lisplib_try_load(val sym)
diff --git a/share/txr/stdlib/ifa.tl b/share/txr/stdlib/ifa.tl
new file mode 100644
index 00000000..f7c4fcb0
--- /dev/null
+++ b/share/txr/stdlib/ifa.tl
@@ -0,0 +1,44 @@
+;; Copyright 2015
+;; Kaz Kylheku <kaz@kylheku.com>
+;; Vancouver, Canada
+;; All rights reserved.
+;;
+;; Redistribution of this software in source and binary forms, with or without
+;; modification, is permitted provided that the following two conditions are met.
+;;
+;; Use of this software in any manner constitutes agreement with the disclaimer
+;; which follows the two conditions.
+;;
+;; 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 ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
+;; WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
+;; MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL THE
+;; COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DAMAGES, HOWEVER CAUSED,
+;; AND UNDER ANY THEORY OF LIABILITY, ARISING IN ANY WAY OUT OF THE USE OF THIS
+;; SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+(defmacro ifa (:env e test then : else)
+ (flet ((candidate-p (expr)
+ (not (or (constantp expr e) (symbolp expr)))))
+ (cond
+ ((or (atom test) (null (cdr test))) ^(let ((it ,test))
+ (if it ,then ,else)))
+ ((member (first test) '(not null)) ^(ifa ,(second test) ,else ,then))
+ (t (let* ((sym (first test))
+ (args (rest test))
+ (n-candidate-args [count-if candidate-p args])
+ (pos-candidate (or [pos-if candidate-p args] 0)))
+ (unless (fboundp sym)
+ (error "ifa: only works with global functions."))
+ (when (> n-candidate-args 1)
+ (error "ifa: ambiguous situation: not clear what can be \"it\"."))
+ (let* ((temps (mapcar (ret (gensym)) args))
+ (it-temp [temps pos-candidate]))
+ ^(let* (,*(zip temps args) (it ,it-temp))
+ (if (,sym ,*temps) ,then ,else))))))))
diff --git a/txr.c b/txr.c
index effb3c2b..9897bc58 100644
--- a/txr.c
+++ b/txr.c
@@ -60,6 +60,7 @@ static const char *progname_u8;
static val progpath = nil;
int opt_noninteractive;
int opt_compat;
+val stdlib_path;
/*
* Can implement an emergency allocator here from a fixed storage
@@ -250,12 +251,14 @@ static void sysroot_init(void)
val slash = regex_compile(lit("\\\\"), nil);
#endif
prot1(&progpath);
+ prot1(&stdlib_path);
progpath = get_self_path();
#if HAVE_WINDOWS_H
progpath = regsub(slash, lit("/"), progpath);
#endif
- reg_var(intern(lit("stdlib"), user_package),
- sysroot(lit("share/txr/stdlib")));
+ stdlib_path = sysroot(lit("share/txr/stdlib"));
+
+ reg_var(intern(lit("stdlib"), user_package), stdlib_path);
reg_var(intern(lit("*txr-version*"), user_package),
toint(lit(TXR_VER), nil));
}
diff --git a/txr.h b/txr.h
index be82ebfc..d9b2a946 100644
--- a/txr.h
+++ b/txr.h
@@ -38,3 +38,4 @@ extern int opt_compat;
extern alloc_bytes_t opt_gc_delta;
extern const wchli_t *version;
extern const wchar_t *progname;
+extern val stdlib_path;