summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-06-01 07:05:35 -0700
committerKaz Kylheku <kaz@kylheku.com>2016-06-01 07:05:35 -0700
commitcd8cf4f8fd827e428c53f2e6d7fcce5cd9727e7f (patch)
tree5acb1cb50e758409457f971d3e2fefa484ed9890
parent8a443d67ef95021529db7eb451479e79fb39b272 (diff)
downloadtxr-cd8cf4f8fd827e428c53f2e6d7fcce5cd9727e7f.tar.gz
txr-cd8cf4f8fd827e428c53f2e6d7fcce5cd9727e7f.tar.bz2
txr-cd8cf4f8fd827e428c53f2e6d7fcce5cd9727e7f.zip
Methods for turning objects into sequences.
Struct objects can now define methods car, cdr and nullify. With these, they can participate in operations on sequences. * eval.h (car_s, cdr_s): Declared. * lib.c (nullify_s): New symbol variable. (car, cdr): Implement for struct objects via, respectively, their car and cdr methods. (tolist): Handle objects by mapping through identity. (nullify): Implement for objects optionally: if an object is a struct with a nullify method, use it, otherwise go through default case of just returning the object. (empty): Implement for objects that have nullify method. (obj_init): Initialize nullify_s. * struct.c (maybe_slot): New function. * struct.h (maybe_slot): Declared. * txr.1: Documented car, cdr and nullify method convention.
-rw-r--r--eval.h1
-rw-r--r--lib.c22
-rw-r--r--struct.c14
-rw-r--r--struct.h1
-rw-r--r--txr.149
5 files changed, 85 insertions, 2 deletions
diff --git a/eval.h b/eval.h
index 7af95058..fb9c34f4 100644
--- a/eval.h
+++ b/eval.h
@@ -28,6 +28,7 @@ extern val dwim_s, lambda_s, vector_lit_s, vec_list_s, list_s;
extern val hash_lit_s, hash_construct_s, struct_lit_s, qref_s;
extern val eval_error_s, if_s, call_s;
extern val eq_s, eql_s, equal_s;
+extern val car_s, cdr_s;
extern val last_form_evaled, last_form_expanded;
extern val self_load_path_s;
diff --git a/lib.c b/lib.c
index 31301b6d..92d218a6 100644
--- a/lib.c
+++ b/lib.c
@@ -101,7 +101,7 @@ val error_s, type_error_s, internal_error_s, panic_s;
val numeric_error_s, range_error_s;
val query_error_s, file_error_s, process_error_s, syntax_error_s;
val timeout_error_s, system_error_s;
-val gensym_counter_s;
+val gensym_counter_s, nullify_s;
val nothrow_k, args_k, colon_k, auto_k, fun_k;
val wrap_k, reflect_k;
@@ -305,6 +305,9 @@ val car(val cons)
if (zerop(length_str(cons)))
return nil;
return chr_str(cons, zero);
+ case COBJ:
+ if (structp(cons))
+ return funcall1(slot(cons, car_s), cons);
default:
type_mismatch(lit("~s is not a cons"), cons, nao);
}
@@ -333,6 +336,9 @@ val cdr(val cons)
if (le(length(cons), one))
return nil;
return sub(cons, one, t);
+ case COBJ:
+ if (structp(cons))
+ return funcall1(slot(cons, cdr_s), cons);
default:
type_mismatch(lit("~s is not a cons"), cons, nao);
}
@@ -692,6 +698,8 @@ val tolist(val seq)
case LIT:
case LSTR:
return list_str(seq);
+ case COBJ:
+ return mapcar(identity_f, seq);
case NIL:
case CONS:
case LCONS:
@@ -715,6 +723,12 @@ val nullify(val seq)
return if3(length_str_gt(seq, zero), seq, nil);
case VEC:
return if3(length_vec(seq) != zero, seq, nil);
+ case COBJ:
+ if (structp(seq)) {
+ val nullify_meth = maybe_slot(seq, nullify_s);
+ if (nullify_meth)
+ return funcall1(nullify_meth, seq);
+ }
default:
return seq;
}
@@ -8041,7 +8055,10 @@ val empty(val seq)
case COBJ:
if (seq->co.cls == hash_s)
return eq(hash_count(seq), zero);
- /* fallthrough */
+ if (structp(seq)) {
+ val nullify_meth = maybe_slot(seq, nullify_s);
+ return if3(nullify_meth && funcall1(nullify_meth, seq), nil, seq);
+ }
default:
type_mismatch(lit("empty: ~s is not a sequence"), seq, nao);
}
@@ -8635,6 +8652,7 @@ static void obj_init(void)
timeout_error_s = intern(lit("timeout-error"), user_package);
assert_s = intern(lit("assert"), user_package);
name_s = intern(lit("name"), user_package);
+ nullify_s = intern(lit("nullify"), user_package);
args_k = intern(lit("args"), keyword_package);
nothrow_k = intern(lit("nothrow"), keyword_package);
diff --git a/struct.c b/struct.c
index 21a3a18a..2075b42f 100644
--- a/struct.c
+++ b/struct.c
@@ -759,6 +759,20 @@ val slot(val strct, val sym)
no_such_slot(self, si->type->self, sym);
}
+val maybe_slot(val strct, val sym)
+{
+ const val self = lit("slot");
+ struct struct_inst *si = struct_handle(strct, self);
+
+ if (symbolp(sym)) {
+ loc ptr = lookup_slot_load(strct, si, sym);
+ if (!nullocp(ptr))
+ return deref(ptr);
+ }
+
+ return nil;
+}
+
val slotset(val strct, val sym, val newval)
{
const val self = lit("slotset");
diff --git a/struct.h b/struct.h
index e71f717b..46dc6542 100644
--- a/struct.h
+++ b/struct.h
@@ -39,6 +39,7 @@ val replace_struct(val target, val source);
val reset_struct(val strct);
val find_struct_type(val sym);
val slot(val strct, val sym);
+val maybe_slot(val strct, val sym);
val slotset(val strct, val sym, val newval);
val static_slot(val stype, val sym);
val static_slot_set(val stype, val sym, val newval);
diff --git a/txr.1 b/txr.1
index e79ec1d7..6ba20ac0 100644
--- a/txr.1
+++ b/txr.1
@@ -10300,6 +10300,13 @@ are made to work with strings and vectors:
(car #(1 2 3)) -> 1
.cble
+Moreover, structure types which define the methods
+.codn car ,
+.code cdr
+and
+.code nullify
+can also be treated in the same way.
+
The
.code ldiff
function is also extended in a special way. When the right parameter
@@ -19288,6 +19295,48 @@ is defined for a structure type, then it is used for pretty-printing instances
of that type. The method takes one argument (in addition to the object), which
specifies the output stream.
+.NP* Sequence Operations on Structures
+
+Structures may be treated as sequences if they define methods named
+by the symbols
+.codn car ,
+.codn cdr ,
+and
+.codn nullify .
+
+If a structure supports these methods, then these methods are used
+by the functions
+.codn car ,
+.codn cdr ,
+.codn nullify ,
+.code empty
+and various other sequence manipulating functions derived from them, when those
+functions are applied to that object.
+
+An object which implements these three methods can be considered to denote
+an abstract sequence. The object's
+.code car
+method should return the first value in that abstract sequence, or else
+.code nil
+if that sequence is empty.
+
+The object's
+.code cdr
+method should return an object denoting the remainder of the sequence,
+or else
+.code nil
+if the sequence is empty or contains only one value. This returned object can
+be of any type: it may be of the same structure type as that object, a
+different structure type, a list, or whatever else. If a non-sequence object
+is returned.
+
+The
+.code nullify
+method should return
+.code nil
+if the object is considered to denote an empty sequence. Otherwise it
+should return that object itself.
+
.coNP Macro @ defstruct
.synb
.mets (defstruct >> { name | >> ( name << arg *)} < super