summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRay Lai <ray@cvs.openbsd.org>2008-04-12 21:20:59 +0000
committerRay Lai <ray@cvs.openbsd.org>2008-04-12 21:20:59 +0000
commit07e159e5bdf3fe2d0b4ca71aaba240aca07e9fe3 (patch)
tree9d4af959dbea566e61cd023ce179613793b982d7
parent8ec925393e05b8ba91ac20034116a3647f9afaa8 (diff)
Protect errno from being trashed by other functions.
okay okan.
-rw-r--r--usr.bin/ftp/ftp.c53
1 files changed, 34 insertions, 19 deletions
diff --git a/usr.bin/ftp/ftp.c b/usr.bin/ftp/ftp.c
index 39625c78f2f..faca5d16581 100644
--- a/usr.bin/ftp/ftp.c
+++ b/usr.bin/ftp/ftp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ftp.c,v 1.68 2008/03/10 22:56:43 espie Exp $ */
+/* $OpenBSD: ftp.c,v 1.69 2008/04/12 21:20:58 ray Exp $ */
/* $NetBSD: ftp.c,v 1.27 1997/08/18 10:20:23 lukem Exp $ */
/*
@@ -60,7 +60,7 @@
*/
#if !defined(lint) && !defined(SMALL)
-static const char rcsid[] = "$OpenBSD: ftp.c,v 1.68 2008/03/10 22:56:43 espie Exp $";
+static const char rcsid[] = "$OpenBSD: ftp.c,v 1.69 2008/04/12 21:20:58 ray Exp $";
#endif /* not lint and not SMALL */
#include <sys/types.h>
@@ -541,7 +541,7 @@ sendrequest(const char *cmd, const char *local, const char *remote,
volatile off_t hashbytes;
char * volatile lmode;
char buf[BUFSIZ], *bufp;
- int oprogress;
+ int oprogress, serrno;
hashbytes = mark;
direction = "sent";
@@ -704,7 +704,7 @@ sendrequest(const char *cmd, const char *local, const char *remote,
case TYPE_I:
case TYPE_L:
- errno = d = 0;
+ d = 0;
while ((c = read(fileno(fin), buf, sizeof(buf))) > 0) {
may_send_noop_char();
bytes += c;
@@ -720,6 +720,8 @@ sendrequest(const char *cmd, const char *local, const char *remote,
(void)fflush(ttyout);
}
}
+ if (c == -1 || d == -1)
+ serrno = errno;
if (hash && (!progress || filesize < 0) && bytes > 0) {
if (bytes < mark)
(void)putc('#', ttyout);
@@ -727,10 +729,10 @@ sendrequest(const char *cmd, const char *local, const char *remote,
(void)fflush(ttyout);
}
if (c < 0)
- warn("local: %s", local);
+ warnx("local: %s: %s", local, strerror(serrno));
if (d < 0) {
- if (errno != EPIPE)
- warn("netout");
+ if (serrno != EPIPE)
+ warnx("netout: %s", strerror(serrno));
bytes = -1;
}
break;
@@ -759,6 +761,8 @@ sendrequest(const char *cmd, const char *local, const char *remote,
}
#endif
}
+ if (ferror(fin) || ferror(dout))
+ serrno = errno;
if (hash && (!progress || filesize < 0)) {
if (bytes < hashbytes)
(void)putc('#', ttyout);
@@ -766,10 +770,10 @@ sendrequest(const char *cmd, const char *local, const char *remote,
(void)fflush(ttyout);
}
if (ferror(fin))
- warn("local: %s", local);
+ warnx("local: %s: %s", local, strerror(serrno));
if (ferror(dout)) {
if (errno != EPIPE)
- warn("netout");
+ warnx("netout: %s", strerror(serrno));
bytes = -1;
}
break;
@@ -834,7 +838,7 @@ recvrequest(const char *cmd, const char * volatile local, const char *remote,
FILE * volatile fout, * volatile din;
int (* volatile closefunc)(FILE *);
volatile sig_t oldinti, oldintr, oldintp;
- int c, d;
+ int c, d, serrno;
volatile int is_retr, tcrflag, bare_lfs;
static size_t bufsize;
static char *buf;
@@ -890,7 +894,7 @@ recvrequest(const char *cmd, const char * volatile local, const char *remote,
oldinti = signal(SIGINFO, psummary);
if (ignorespecial || (strcmp(local, "-") && *local != '|')) {
if (access(local, W_OK) < 0) {
- char *dir = strrchr(local, '/');
+ char *dir;
if (errno != ENOENT && errno != EACCES) {
warn("local: %s", local);
@@ -899,6 +903,7 @@ recvrequest(const char *cmd, const char * volatile local, const char *remote,
code = -1;
return;
}
+ dir = strrchr(local, '/');
if (dir != NULL)
*dir = 0;
d = access(dir == local ? "/" : dir ? local : ".", W_OK);
@@ -1048,6 +1053,8 @@ recvrequest(const char *cmd, const char * volatile local, const char *remote,
(void)fflush(ttyout);
}
}
+ if (c == -1 || d < c)
+ serrno = errno;
if (hash && (!progress || filesize < 0) && bytes > 0) {
if (bytes < mark)
(void)putc('#', ttyout);
@@ -1055,13 +1062,13 @@ recvrequest(const char *cmd, const char * volatile local, const char *remote,
(void)fflush(ttyout);
}
if (c < 0) {
- if (errno != EPIPE)
- warn("netin");
+ if (serrno != EPIPE)
+ warnx("netin: %s", strerror(serrno));
bytes = -1;
}
if (d < c) {
if (d < 0)
- warn("local: %s", local);
+ warnx("local: %s: %s", local, strerror(serrno));
else
warnx("%s: short write", local);
}
@@ -1075,14 +1082,20 @@ recvrequest(const char *cmd, const char * volatile local, const char *remote,
goto done;
n = restart_point;
for (i = 0; i++ < n;) {
- if ((ch = fgetc(fout)) == EOF)
+ if ((ch = fgetc(fout)) == EOF) {
+ if (!ferror(fout))
+ errno = 0;
goto done;
+ }
if (ch == '\n')
i++;
}
if (fseek(fout, 0L, SEEK_CUR) < 0) {
done:
- warn("local: %s", local);
+ if (errno)
+ warn("local: %s", local);
+ else
+ warnx("local: %s", local);
progress = oprogress;
preserve = opreserve;
if (closefunc != NULL)
@@ -1119,6 +1132,8 @@ done:
contin2: ;
}
break2:
+ if (ferror(din) || ferror(fout))
+ serrno = errno;
if (bare_lfs) {
fprintf(ttyout,
"WARNING! %d bare linefeeds received in ASCII mode.\n", bare_lfs);
@@ -1132,12 +1147,12 @@ break2:
(void)fflush(ttyout);
}
if (ferror(din)) {
- if (errno != EPIPE)
- warn("netin");
+ if (serrno != EPIPE)
+ warnx("netin: %s", strerror(serrno));
bytes = -1;
}
if (ferror(fout))
- warn("local: %s", local);
+ warnx("local: %s: %s", local, strerror(serrno));
break;
}
progressmeter(1);