diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-09-11 21:28:24 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-09-11 21:28:24 -0700 |
commit | a349b8cbb6ba961d04a57c477f601800b45950ef (patch) | |
tree | 677da97beb99dd9bc5cfc16bcc15fd8177bf7004 /parser.c | |
parent | ce61431bff702afcbd6a2ef52babd859203cccf8 (diff) | |
download | txr-a349b8cbb6ba961d04a57c477f601800b45950ef.tar.gz txr-a349b8cbb6ba961d04a57c477f601800b45950ef.tar.bz2 txr-a349b8cbb6ba961d04a57c477f601800b45950ef.zip |
Better EOF check in read-eval stream function.
This makes a difference if we read from a TTY.
Without this fix, two Ctrl-D's have to be issued
before the code registers EOF and returns.
The trick is that we know, after calling the
parser, whether it hit EOF, because we have
the recent token. If its value is zero, that's
the EOF token. We can bail out of the loop
instead of calling the parser again which will
require another EOF (because prime_parser
will not push an EOF token!)
* parser.h (parser_eof): Declared.
* parser.c (read_eval_stream): After lisp_parse
and eval, check for EOF with the parser_eof
function, and bail if so.
(parser_eof): New function.
Diffstat (limited to 'parser.c')
-rw-r--r-- | parser.c | 11 |
1 files changed, 10 insertions, 1 deletions
@@ -351,14 +351,17 @@ val read_eval_stream(val stream, val error_stream, val hash_bang_support) for (;;) { val form = lisp_parse(stream, error_stream, error_val, name, colon_k); + val parser = get_parser(stream); if (form == error_val) { - if (parser_errors(get_parser(stream)) == zero) + if (parser_errors(parser) == zero) break; return nil; } (void) eval_intrinsic(form, nil); + if (parser_eof(parser)) + break; } return t; @@ -638,6 +641,12 @@ val parser_errors(val parser) return num(p->errors); } +val parser_eof(val parser) +{ + parser_t *p = coerce(parser_t *, cobj_handle(parser, parser_s)); + return tnil(p->recent_tok.yy_char == 0); +} + void parse_init(void) { parser_s = intern(lit("parser"), user_package); |