summaryrefslogtreecommitdiffstats
path: root/stream.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2013-12-14 23:46:32 -0800
committerKaz Kylheku <kaz@kylheku.com>2013-12-14 23:46:32 -0800
commite5cd9bbd3de84e90d5602b83a0eb7780ec4b750e (patch)
treee5a35dda0dc080b69ff6b9042cdabcbd5dd2fa70 /stream.c
parentf8ffbad0cec3ff0be1376a6352fe63a3c5e5f361 (diff)
downloadtxr-e5cd9bbd3de84e90d5602b83a0eb7780ec4b750e.tar.gz
txr-e5cd9bbd3de84e90d5602b83a0eb7780ec4b750e.tar.bz2
txr-e5cd9bbd3de84e90d5602b83a0eb7780ec4b750e.zip
Support for parsing Lisp expression out of strings and streams.
New catenated streams make the Yacc hack possible. * eval.c (eval_init): Register lisp_parse as intrinsic. * parser.h (lisp_parse): Declared. * parser.l: New lexical hack to produce SECRET_ESCAPE_E token. (regex_parse): Move declaration before statements. (lisp_parse): New function. * parser.y (SECRET_ESCAPE_E): New token type. (spec): New production rule for single expression. * stream.c (cat_stream_print, cat_get_line, cat_get_char, cat_get_byte, cat_get_prop): New static functions. (cat_stream_ops): New static function. (make_catenated_stream): New function. * stream.h (make_catenated_stream): Declared.
Diffstat (limited to 'stream.c')
-rw-r--r--stream.c80
1 files changed, 80 insertions, 0 deletions
diff --git a/stream.c b/stream.c
index 67765e8a..c20b3ca5 100644
--- a/stream.c
+++ b/stream.c
@@ -2030,6 +2030,86 @@ val open_process(val name, val mode_str, val args)
}
#endif
+static void cat_stream_print(val stream, val out)
+{
+ val streams = (val) stream->co.handle;
+ format(out, lit("#<~s catenated ~s>"), stream->co.cls, streams, nao);
+}
+
+static val cat_get_line(val stream)
+{
+ val streams = (val) stream->co.handle;
+
+ while (streams) {
+ val line = get_line(first(streams));
+ if (line)
+ return line;
+ stream->co.handle = (mem_t *) (streams = rest(streams));
+ }
+
+ return nil;
+}
+
+static val cat_get_char(val stream)
+{
+ val streams = (val) stream->co.handle;
+
+ while (streams) {
+ val ch = get_char(first(streams));
+ if (ch)
+ return ch;
+ stream->co.handle = (mem_t *) (streams = rest(streams));
+ }
+
+ return nil;
+}
+
+static val cat_get_byte(val stream)
+{
+ val streams = (val) stream->co.handle;
+
+ while (streams) {
+ val byte = get_byte(first(streams));
+ if (byte)
+ return byte;
+ stream->co.handle = (mem_t *) (streams = rest(streams));
+ }
+
+ return nil;
+}
+
+static val cat_get_prop(val stream, val ind)
+{
+ val streams = (val) stream->co.handle;
+ if (streams)
+ return stream_get_prop(first(streams), ind);
+ return nil;
+}
+
+static struct strm_ops cat_stream_ops = {
+ { cobj_equal_op,
+ cat_stream_print,
+ cobj_destroy_stub_op,
+ cobj_mark_op,
+ cobj_hash_op },
+ 0, /* put_string */
+ 0, /*_put_char */
+ 0, /* put_byte, */
+ cat_get_line,
+ cat_get_char,
+ cat_get_byte,
+ 0, /* close, */
+ 0, /* flush, */
+ 0, /* seek, */
+ cat_get_prop,
+ 0, /* set_prop */
+};
+
+val make_catenated_stream(val stream_list)
+{
+ return cobj((mem_t *) stream_list, stream_s, &cat_stream_ops.cobj_ops);
+}
+
void stream_init(void)
{
protect(&std_input, &std_output, &std_debug, &std_error, &std_null, (val *) 0);