diff options
-rw-r--r-- | ChangeLog | 48 | ||||
-rw-r--r-- | lib.c | 7 | ||||
-rw-r--r-- | lib.h | 2 | ||||
-rw-r--r-- | match.c | 4 | ||||
-rw-r--r-- | stream.c | 176 | ||||
-rw-r--r-- | stream.h | 6 | ||||
-rw-r--r-- | txr.c | 4 | ||||
-rw-r--r-- | unwind.c | 1 |
8 files changed, 190 insertions, 58 deletions
@@ -1,3 +1,51 @@ +2009-11-06 Kaz Kylheku <kkylheku@gmail.com> + + Throw exception on stream error during close, or I/O operations. This + is needed for pipes that terminate abnormally or return failed + termination. Pipe and stdio streams have an extra description field + so they are printed in a readable way. + + * lib.c (process_error): New global defined. + (obj_init): New symbol interned. + (lazy_stream_func): Pass t to close_stream, so exception + is thrown if the close fails. + (lazy_stream_cons): Ditto. + + * lib.h (process_error): Declared. + + * match.c (complex_snarf): Pass new desr argument to make_stdio_stream + and make_pipe_stream. + + * stream.c (strm_ops): New argument on close function pointer. + (common_destroy): Close without throwing exception. For objects + being finalized, we don't care if the close works or not; the + program has shown that it doesn't care about the stream by + letting it become unreachable, so we don't bother the program + by throwing an exception. + (stdio_handle): New struct. + (stdio_stream_print, stdio_stream_destroy, stdio_stream_mark, + stdio_maybe_read_error, stdio_maybe_write_error): New functions. + (stdio_put_string, stdio_put_char, stdio_get_line, + stdio_get_char, stdio_vcformat, stdio_close): Updated to new + handle format, and throw errors now. + (stdio_ops, pipe_ops): Redirected to new functions stdio_stream_print, + stdio_stream_destroy and stdio_stream_mark. + (pipe_close): Updated to new handle format. Parses status from + pclose and throws exceptions appropriate to the situation. + (dir_close): Takes extra argument. + (make_stdio_stream, make_pipe_stream): New argument added. + (make_string_output_stream): Some casts added. + (close_stream): Pass new argument down to virtual function. + (stream_init): Pass new argument to make_stdio_stream + when creating streams for stdin, stdout and stderr. + + * stream.h (make_stdio_stream, make_pipe_stream, close_stream): + Declarations updated. + + * txr.c (txr_main): Pass new argument to make_stdio_stream. + + * unwind.c (uw_init): Register process_error. + 2009-11-01 Kaz Kylheku <kkylheku@gmail.com> Version 020 @@ -53,7 +53,7 @@ obj_t *define, *output, *single, *frst, *lst, *empty, *repeat, *rep; obj_t *flattn, *forget, *local, *mrge, *bind, *cat, *args; obj_t *try, *catch, *finally, *nothrow, *throw, *defex; obj_t *error, *type_error, *internal_err, *numeric_err, *range_err; -obj_t *query_error, *file_error; +obj_t *query_error, *file_error, *process_error; obj_t *zero, *one, *two, *negone, *maxint, *minint; obj_t *null_string; @@ -1338,7 +1338,7 @@ static obj_t *lazy_stream_func(obj_t *env, obj_t *lcons) lcons->lc.func = nil; if (!next || !ahead) - close_stream(stream); + close_stream(stream, t); if (ahead) push(ahead, cdr_l(env)); @@ -1351,7 +1351,7 @@ obj_t *lazy_stream_cons(obj_t *stream) obj_t *first = get_line(stream); if (!first) { - close_stream(stream); + close_stream(stream, t); return nil; } @@ -1753,6 +1753,7 @@ static void obj_init(void) range_err = intern(string("range_error")); query_error = intern(string("query_error")); file_error = intern(string("file_error")); + process_error = intern(string("process_error")); interned_syms = cons(nil, interned_syms); @@ -164,7 +164,7 @@ extern obj_t *define, *output, *single, *frst, *lst, *empty, *repeat, *rep; extern obj_t *flattn, *forget, *local, *mrge, *bind, *cat, *args; extern obj_t *try, *catch, *finally, *nothrow, *throw, *defex; extern obj_t *error, *type_error, *internal_err, *numeric_err, *range_err; -extern obj_t *query_error, *file_error; +extern obj_t *query_error, *file_error, *process_error; extern obj_t *zero, *one, *two, *negone, *maxint, *minint; extern obj_t *null_string; @@ -683,9 +683,9 @@ obj_t *complex_snarf(fpip_t fp, obj_t *name) { switch (fp.close) { case fpip_fclose: - return lazy_stream_cons(make_stdio_stream(fp.f, t, nil)); + return lazy_stream_cons(make_stdio_stream(fp.f, name, t, nil)); case fpip_pclose: - return lazy_stream_cons(make_pipe_stream(fp.f, t, nil)); + return lazy_stream_cons(make_pipe_stream(fp.f, name, t, nil)); case fpip_closedir: return lazy_stream_cons(make_dir_stream(fp.d)); } @@ -31,6 +31,7 @@ #include <stdlib.h> #include <assert.h> #include <setjmp.h> +#include <errno.h> #include "lib.h" #include "gc.h" #include "unwind.h" @@ -46,7 +47,7 @@ struct strm_ops { obj_t *(*get_char)(obj_t *); obj_t *(*vcformat)(obj_t *, const char *fmt, va_list vl); obj_t *(*vformat)(obj_t *, const char *fmt, va_list vl); - obj_t *(*close)(obj_t *); + obj_t *(*close)(obj_t *, obj_t *); }; static obj_t *common_equal(obj_t *self, obj_t *other) @@ -56,7 +57,7 @@ static obj_t *common_equal(obj_t *self, obj_t *other) static void common_destroy(obj_t *obj) { - (void) close_stream(obj); + (void) close_stream(obj, nil); } obj_t *common_vformat(obj_t *stream, const char *fmt, va_list vl) @@ -100,16 +101,62 @@ obj_t *common_vformat(obj_t *stream, const char *fmt, va_list vl) return t; } +struct stdio_handle { + FILE *f; + obj_t *descr; +}; + +void stdio_stream_print(obj_t *stream, obj_t *out) +{ + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + format(out, "#<~s ~s>", stream->co.cls, h->descr, nao); +} + +void stdio_stream_destroy(obj_t *stream) +{ + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + common_destroy(stream); + free(h); +} + +void stdio_stream_mark(obj_t *stream) +{ + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + gc_mark(h->descr); +} + +static obj_t *stdio_maybe_read_error(obj_t *stream) +{ + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + if (ferror(h->f)) { + clearerr(h->f); + uw_throwf(file_error, "error reading ~a: ~a/~s", + stream, num(errno), string(strerror(errno))); + } + return nil; +} + +static obj_t *stdio_maybe_write_error(obj_t *stream) +{ + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + if (ferror(h->f)) { + clearerr(h->f); + uw_throwf(file_error, "error writing ~a: ~a/~s", + stream, num(errno), string(strerror(errno))); + } + return nil; +} + static obj_t *stdio_put_string(obj_t *stream, const char *s) { - FILE *f = (FILE *) stream->co.handle; - return (f && fputs(s, f) != EOF) ? t : nil; + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + return (h->f && fputs(s, h->f) != EOF) ? t : stdio_maybe_write_error(stream); } static obj_t *stdio_put_char(obj_t *stream, int ch) { - FILE *f = (FILE *) stream->co.handle; - return (f && putc(ch, f) != EOF) ? t : nil; + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + return (h->f && putc(ch, h->f) != EOF) ? t : stdio_maybe_write_error(stream); } static char *snarf_line(FILE *in) @@ -149,41 +196,46 @@ static obj_t *stdio_get_line(obj_t *stream) if (stream->co.handle == 0) { return nil; } else { - char *line = snarf_line((FILE *) stream->co.handle); + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + char *line = snarf_line(h->f); if (!line) - return nil; + return stdio_maybe_read_error(stream); return string_own(line); } } obj_t *stdio_get_char(obj_t *stream) { - FILE *f = (FILE *) stream->co.handle; - if (f) { - int ch = getc(f); - return (ch != EOF) ? chr(ch) : nil; + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + if (h->f) { + int ch = getc(h->f); + return (ch != EOF) ? chr(ch) : stdio_maybe_read_error(stream); } return nil; } obj_t *stdio_vcformat(obj_t *stream, const char *fmt, va_list vl) { - FILE *f = (FILE *) stream->co.handle; - if (f) { - int n = vfprintf(f, fmt, vl); - return (n >= 0) ? num(n) : nil; + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + + if (h->f) { + int n = vfprintf(h->f, fmt, vl); + return (n >= 0) ? num(n) : stdio_maybe_write_error(stream); } return nil; } -static obj_t *stdio_close(obj_t *stream) +static obj_t *stdio_close(obj_t *stream, obj_t *throw_on_error) { + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; - FILE *f = (FILE *) stream->co.handle; - - if (f != 0 && f != stdin && f != stdout) { - int result = fclose(f); - stream->co.handle = 0; + if (h->f != 0 && h->f != stdin && h->f != stdout) { + int result = fclose(h->f); + h->f = 0; + if (result == EOF && throw_on_error) { + uw_throwf(file_error, "error closing ~a: ~a/~s", + stream, num(errno), string(strerror(errno))); + } return result != EOF ? t : nil; } return nil; @@ -191,9 +243,9 @@ static obj_t *stdio_close(obj_t *stream) static struct strm_ops stdio_ops = { { common_equal, - cobj_print_op, - common_destroy, - 0 }, + stdio_stream_print, + stdio_stream_destroy, + stdio_stream_mark }, stdio_put_string, stdio_put_char, stdio_get_line, @@ -203,23 +255,47 @@ static struct strm_ops stdio_ops = { stdio_close }; -static obj_t *pipe_close(obj_t *stream) -{ - FILE *f = (FILE *) stream->co.handle; +static obj_t *pipe_close(obj_t *stream, obj_t *throw_on_error) +{ + struct stdio_handle *h = (struct stdio_handle *) stream->co.handle; + + if (h->f != 0) { + int status = pclose(h->f); + + h->f = 0; + + if (status != 0 && throw_on_error) { + if (status < 0) { + uw_throwf(process_error, "unable to obtain status of command ~a: ~a/~s", + stream, num(errno), string(strerror(errno)), nao); + } else if (WIFEXITED(status)) { + int exitstatus = WEXITSTATUS(status); + uw_throwf(process_error, "pipe ~a terminated with status ~a", + stream, num(exitstatus), nao); + } else if (WIFSIGNALED(status)) { + int termsig = WTERMSIG(status); + uw_throwf(process_error, "pipe ~a terminated by signal ~a", + stream, num(termsig), nao); + + } else if (WIFSTOPPED(status) || WIFCONTINUED(status)) { + uw_throwf(process_error, "processes of closed pipe ~a still running", + stream, nao); + } else { + uw_throwf(file_error, "strange status in when closing pipe ~a", + stream, nao); + } + } - if (f != 0) { - int result = pclose(f); - stream->co.handle = 0; - return result >= 0 ? t : nil; + return status == 0 ? t : nil; } return nil; } static struct strm_ops pipe_ops = { { common_equal, - cobj_print_op, - common_destroy, - 0 }, + stdio_stream_print, + stdio_stream_destroy, + stdio_stream_mark }, stdio_put_string, stdio_put_char, stdio_get_line, @@ -412,7 +488,7 @@ static obj_t *dir_get_line(obj_t *stream) } } -static obj_t *dir_close(obj_t *stream) +static obj_t *dir_close(obj_t *stream, obj_t *throw_on_error) { if (stream->co.handle != 0) { closedir((DIR *) stream->co.handle); @@ -438,14 +514,20 @@ static struct strm_ops dir_ops = { }; -obj_t *make_stdio_stream(FILE *handle, obj_t *input, obj_t *output) +obj_t *make_stdio_stream(FILE *f, obj_t *descr, obj_t *input, obj_t *output) { - return cobj((void *) handle, stream_t, &stdio_ops.cobj_ops); + struct stdio_handle *h = (struct stdio_handle *) chk_malloc(sizeof *h); + h->f = f; + h->descr = descr; + return cobj((void *) h, stream_t, &stdio_ops.cobj_ops); } -obj_t *make_pipe_stream(FILE *handle, obj_t *input, obj_t *output) +obj_t *make_pipe_stream(FILE *f, obj_t *descr, obj_t *input, obj_t *output) { - return cobj((void *) handle, stream_t, &pipe_ops.cobj_ops); + struct stdio_handle *h = (struct stdio_handle *) chk_malloc(sizeof *h); + h->f = f; + h->descr = descr; + return cobj((void *) h, stream_t, &pipe_ops.cobj_ops); } obj_t *make_string_input_stream(obj_t *string) @@ -455,9 +537,9 @@ obj_t *make_string_input_stream(obj_t *string) obj_t *make_string_output_stream(void) { - struct string_output *so = chk_malloc(sizeof *so); + struct string_output *so = (struct string_output *) chk_malloc(sizeof *so); so->size = 128; - so->buf = chk_malloc(so->size); + so->buf = (char *) chk_malloc(so->size); so->fill = 0; so->buf[0] = 0; return cobj((void *) so, stream_t, &string_out_ops.cobj_ops); @@ -494,14 +576,14 @@ obj_t *make_dir_stream(DIR *dir) return cobj((void *) dir, stream_t, &dir_ops.cobj_ops); } -obj_t *close_stream(obj_t *stream) +obj_t *close_stream(obj_t *stream, obj_t *throw_on_error) { type_check (stream, COBJ); type_assert (stream->co.cls == stream_t, ("~a is not a stream", stream)); { struct strm_ops *ops = (struct strm_ops *) stream->co.ops; - return ops->close ? ops->close(stream) : nil; + return ops->close ? ops->close(stream, throw_on_error) : nil; } } @@ -635,7 +717,7 @@ obj_t *put_line(obj_t *stream, obj_t *string) void stream_init(void) { protect(&std_input, &std_output, &std_error, (obj_t **) 0); - std_input = make_stdio_stream(stdin, t, nil); - std_output = make_stdio_stream(stdout, nil, t); - std_error = make_stdio_stream(stderr, nil, t); + std_input = make_stdio_stream(stdin, string("stdin"), t, nil); + std_output = make_stdio_stream(stdout, string("stdout"), nil, t); + std_error = make_stdio_stream(stderr, string("stderr"), nil, t); } @@ -26,13 +26,13 @@ extern obj_t *std_input, *std_output, *std_error; -obj_t *make_stdio_stream(FILE *, obj_t *input, obj_t *output); -obj_t *make_pipe_stream(FILE *, obj_t *input, obj_t *output); +obj_t *make_stdio_stream(FILE *, obj_t *descr, obj_t *input, obj_t *output); +obj_t *make_pipe_stream(FILE *, obj_t *descr, obj_t *input, obj_t *output); obj_t *make_string_input_stream(obj_t *); obj_t *make_string_output_stream(void); obj_t *get_string_from_stream(obj_t *); obj_t *make_dir_stream(DIR *); -obj_t *close_stream(obj_t *); +obj_t *close_stream(obj_t *stream, obj_t *throw_on_error); obj_t *get_line(obj_t *); obj_t *get_char(obj_t *); obj_t *vformat(obj_t *stream, const char *string, va_list); /* nao-terminated */ @@ -309,7 +309,7 @@ static int txr_main(int argc, char **argv) FILE *in = fopen(c_str(spec_file_str), "r"); if (in == 0) uw_throwcf(file_error, "unable to open %s", c_str(spec_file_str)); - yyin_stream = make_stdio_stream(in, t, nil); + yyin_stream = make_stdio_stream(in, spec_file_str, t, nil); } else { spec_file = "stdin"; } @@ -323,7 +323,7 @@ static int txr_main(int argc, char **argv) FILE *in = fopen(*argv, "r"); if (in == 0) uw_throwcf(file_error, "unable to open %s", *argv); - yyin_stream = make_stdio_stream(in, t, nil); + yyin_stream = make_stdio_stream(in, string(*argv), t, nil); spec_file = *argv; } else { spec_file = "stdin"; @@ -369,4 +369,5 @@ void uw_init(void) uw_register_subtype(range_err, error); uw_register_subtype(query_error, error); uw_register_subtype(file_error, error); + uw_register_subtype(process_error, error); } |