diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2013-12-14 23:46:32 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2013-12-14 23:46:32 -0800 |
commit | e5cd9bbd3de84e90d5602b83a0eb7780ec4b750e (patch) | |
tree | e5a35dda0dc080b69ff6b9042cdabcbd5dd2fa70 /stream.c | |
parent | f8ffbad0cec3ff0be1376a6352fe63a3c5e5f361 (diff) | |
download | txr-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.c | 80 |
1 files changed, 80 insertions, 0 deletions
@@ -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); |