diff options
author | 2016-05-25 09:23:49 +0000 | |
---|---|---|
committer | 2016-05-25 09:23:49 +0000 | |
commit | eea31d84e082354d33a160ec9772da6b8e1279be (patch) | |
tree | f65b3859e5b2a6254fcfff4fcce77595c50c80c2 /lib/libedit/read.c | |
parent | Assert we're not freeing buffers we didn't allocate (diff) | |
download | wireguard-openbsd-eea31d84e082354d33a160ec9772da6b8e1279be.tar.xz wireguard-openbsd-eea31d84e082354d33a160ec9772da6b8e1279be.zip |
Saving errno in el_errno is only needed for one purpose:
Restoring the original errno found in el_wgetc() after
el_wgets() did some cleanup that may have changed errno.
Improve clarity and robustness of the code by not setting and
inspecting el_errno where it isn't needed; in particular, let
keymacro_get() properly report read failure to read_getcmd().
Move el_errno to el_read_t because it's only used in read.c.
Never set errno back to zero.
Checked with a test program installing a USR1 signal handler
without SA_RESTART, for the cases read_getcmd(), ed_quoted_insert(),
keymacro_get(), ed_command(), and EL_EDITMODE=0.
OK czarkoff@
Diffstat (limited to 'lib/libedit/read.c')
-rw-r--r-- | lib/libedit/read.c | 46 |
1 files changed, 20 insertions, 26 deletions
diff --git a/lib/libedit/read.c b/lib/libedit/read.c index d26760010be..dc288825980 100644 --- a/lib/libedit/read.c +++ b/lib/libedit/read.c @@ -1,4 +1,4 @@ -/* $OpenBSD: read.c,v 1.42 2016/05/24 18:06:30 schwarze Exp $ */ +/* $OpenBSD: read.c,v 1.43 2016/05/25 09:23:49 schwarze Exp $ */ /* $NetBSD: read.c,v 1.97 2016/05/22 19:44:26 christos Exp $ */ /*- @@ -62,6 +62,7 @@ struct macros { struct el_read_t { struct macros macros; el_rfunc_t read_char; /* Function to read a character. */ + int read_errno; }; static int read__fixio(int, int); @@ -223,12 +224,9 @@ read_getcmd(EditLine *el, el_action_t *cmdnum, wchar_t *ch) el_action_t cmd; int num; - el->el_errno = 0; do { - if ((num = el_wgetc(el, ch)) != 1) {/* if EOF or error */ - el->el_errno = num == 0 ? 0 : errno; + if ((num = el_wgetc(el, ch)) != 1) return -1; - } #ifdef KANJI if ((*ch & meta)) { @@ -255,6 +253,8 @@ read_getcmd(EditLine *el, el_action_t *cmdnum, wchar_t *ch) case XK_STR: el_wpush(el, val.str); break; + case XK_NOD: + return -1; default: EL_ABORT((el->el_errfile, "Bad XK_ type \n")); break; @@ -407,8 +407,15 @@ el_wgetc(EditLine *el, wchar_t *cp) return 0; num_read = (*el->el_read->read_char)(el, cp); + + /* + * Remember the original reason of a read failure + * such that el_wgets() can restore it after doing + * various cleanup operation that might change errno. + */ if (num_read < 0) - el->el_errno = errno; + el->el_read->read_errno = errno; + return num_read; } @@ -456,6 +463,7 @@ el_wgets(EditLine *el, int *nread) if (nread == NULL) nread = &nrb; *nread = 0; + el->el_read->read_errno = 0; if (el->el_flags & NO_TTY) { size_t idx; @@ -476,12 +484,8 @@ el_wgets(EditLine *el, int *nread) if (cp[-1] == '\r' || cp[-1] == '\n') break; } - if (num == -1) { - if (errno == EINTR) - cp = el->el_line.buffer; - el->el_errno = errno; - } - + if (num == -1 && errno == EINTR) + cp = el->el_line.buffer; goto noedit; } @@ -530,13 +534,8 @@ el_wgets(EditLine *el, int *nread) if (crlf) break; } - - if (num == -1) { - if (errno == EINTR) - cp = el->el_line.buffer; - el->el_errno = errno; - } - + if (num == -1 && errno == EINTR) + cp = el->el_line.buffer; goto noedit; } @@ -544,12 +543,6 @@ el_wgets(EditLine *el, int *nread) /* if EOF or error */ if (read_getcmd(el, &cmdnum, &ch) == -1) break; - if (el->el_errno == EINTR) { - el->el_line.buffer[0] = '\0'; - el->el_line.lastchar = - el->el_line.cursor = el->el_line.buffer; - break; - } if ((int)cmdnum >= el->el_map.nfunc) /* BUG CHECK command */ continue; /* try again */ /* now do the real command */ @@ -650,7 +643,8 @@ done: if (*nread == 0) { if (num == -1) { *nread = -1; - errno = el->el_errno; + if (el->el_read->read_errno) + errno = el->el_read->read_errno; } return NULL; } else |