summaryrefslogtreecommitdiff
path: root/libexec/tftpd/tftpd.c
diff options
context:
space:
mode:
Diffstat (limited to 'libexec/tftpd/tftpd.c')
-rw-r--r--libexec/tftpd/tftpd.c232
1 files changed, 115 insertions, 117 deletions
diff --git a/libexec/tftpd/tftpd.c b/libexec/tftpd/tftpd.c
index d6cd982ed33..3c69625ba8b 100644
--- a/libexec/tftpd/tftpd.c
+++ b/libexec/tftpd/tftpd.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: tftpd.c,v 1.44 2006/06/16 22:40:35 beck Exp $ */
+/* $OpenBSD: tftpd.c,v 1.45 2006/07/14 22:48:13 mglocker Exp $ */
/*
* Copyright (c) 1983 Regents of the University of California.
@@ -37,7 +37,7 @@ char copyright[] =
#ifndef lint
/*static char sccsid[] = "from: @(#)tftpd.c 5.13 (Berkeley) 2/26/91";*/
-static char rcsid[] = "$OpenBSD: tftpd.c,v 1.44 2006/06/16 22:40:35 beck Exp $";
+static char rcsid[] = "$OpenBSD: tftpd.c,v 1.45 2006/07/14 22:48:13 mglocker Exp $";
#endif /* not lint */
/*
@@ -46,82 +46,91 @@ static char rcsid[] = "$OpenBSD: tftpd.c,v 1.44 2006/06/16 22:40:35 beck Exp $";
* This version includes many modifications by Jim Guyton <guyton@rand-unix>
*/
-#include <sys/param.h>
#include <sys/ioctl.h>
+#include <sys/param.h>
+#include <sys/socket.h>
#include <sys/stat.h>
#include <sys/uio.h>
-#include <unistd.h>
-#include <fcntl.h>
-#include <vis.h>
-#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/tftp.h>
#include <netdb.h>
+#include <ctype.h>
+#include <errno.h>
+#include <fcntl.h>
#include <poll.h>
-#include <syslog.h>
+#include <pwd.h>
#include <stdio.h>
-#include <errno.h>
-#include <ctype.h>
-#include <string.h>
#include <stdlib.h>
-#include <pwd.h>
+#include <string.h>
+#include <syslog.h>
+#include <unistd.h>
+#include <vis.h>
#define TIMEOUT 5
#define MAX_TIMEOUTS 5
-
-extern char *__progname;
-struct sockaddr_storage s_in;
-int peer;
-int rexmtval = TIMEOUT;
-int max_rexmtval = 2*TIMEOUT;
-
-#define PKTSIZE SEGSIZE+4
-char buf[PKTSIZE];
-char ackbuf[PKTSIZE];
-struct sockaddr_storage from;
-
-int ndirs;
-char **dirs;
-
-int secure;
-int cancreate;
-
-struct formats;
-int validate_access(char *filename, int mode);
-int recvfile(struct formats *pf);
-int sendfile(struct formats *pf);
-
-FILE *file;
+#define PKTSIZE SEGSIZE + 4
+
+struct formats;
+
+int readit(FILE *, struct tftphdr **, int);
+void read_ahead(FILE *, int);
+int writeit(FILE *, struct tftphdr **, int, int);
+int write_behind(FILE *, int);
+int synchnet(int);
+
+static void usage(void);
+void tftp(struct tftphdr *, int);
+int validate_access(char *, int);
+int sendfile(struct formats *);
+int recvfile(struct formats *);
+void nak(int);
+void oack(void);
+
+FILE *file;
+extern char *__progname;
+struct sockaddr_storage s_in;
+int peer;
+int rexmtval = TIMEOUT;
+int max_rexmtval = 2 * TIMEOUT;
+char buf[PKTSIZE];
+char ackbuf[PKTSIZE];
+struct sockaddr_storage from;
+int ndirs;
+char **dirs;
+int secure;
+int cancreate;
struct formats {
const char *f_mode;
- int (*f_validate)(char *, int);
- int (*f_send)(struct formats *);
- int (*f_recv)(struct formats *);
- int f_convert;
+ int (*f_validate)(char *, int);
+ int (*f_send)(struct formats *);
+ int (*f_recv)(struct formats *);
+ int f_convert;
} formats[] = {
{ "netascii", validate_access, sendfile, recvfile, 1 },
{ "octet", validate_access, sendfile, recvfile, 0 },
{ NULL, NULL, NULL, NULL, 0 }
};
+
struct options {
const char *o_type;
char *o_request;
- int o_reply; /* turn into union if need be */
+ int o_reply; /* turn into union if need be */
} options[] = {
{ "tsize", NULL, 0 }, /* OPT_TSIZE */
{ "timeout", NULL, 0 }, /* OPT_TIMEOUT */
{ NULL, NULL, 0 }
};
+
enum opt_enum {
OPT_TSIZE = 0,
OPT_TIMEOUT
};
struct errmsg {
- int e_code;
+ int e_code;
const char *e_msg;
} errmsgs[] = {
{ EUNDEF, "Undefined error code" },
@@ -136,17 +145,6 @@ struct errmsg {
{ -1, NULL }
};
-int validate_access(char *filename, int mode);
-void tftp(struct tftphdr *tp, int size);
-void nak(int error);
-void oack(void);
-
-int readit(FILE *file, struct tftphdr **dpp, int convert);
-void read_ahead(FILE *file, int convert);
-int writeit(FILE *file, struct tftphdr **dpp, int ct, int convert);
-int write_behind(FILE *file, int convert);
-int synchnet(int f);
-
static void
usage(void)
{
@@ -157,19 +155,19 @@ usage(void)
int
main(int argc, char *argv[])
{
- int n = 0, on = 1, fd = 0, i, c;
- struct tftphdr *tp;
- struct passwd *pw;
- char cbuf[CMSG_SPACE(sizeof(struct sockaddr_storage))];
- struct cmsghdr *cmsg;
- struct msghdr msg;
- struct iovec iov;
- pid_t pid = 0;
- socklen_t j;
-
- openlog(__progname, LOG_PID | LOG_NDELAY, LOG_DAEMON);
-
- while ((c = getopt(argc, argv, "cs")) != -1)
+ int n = 0, on = 1, fd = 0, i, c;
+ struct tftphdr *tp;
+ struct passwd *pw;
+ char cbuf[CMSG_SPACE(sizeof(struct sockaddr_storage))];
+ struct cmsghdr *cmsg;
+ struct msghdr msg;
+ struct iovec iov;
+ pid_t pid = 0;
+ socklen_t j;
+
+ openlog(__progname, LOG_PID|LOG_NDELAY, LOG_DAEMON);
+
+ while ((c = getopt(argc, argv, "cs")) != -1) {
switch (c) {
case 'c':
cancreate = 1;
@@ -179,13 +177,14 @@ main(int argc, char *argv[])
break;
default:
usage();
- break;
+ /* NOTREACHED */
}
+ }
for (; optind != argc; optind++) {
char **d;
- d = realloc(dirs, (ndirs+2) * sizeof (char *));
+ d = realloc(dirs, (ndirs + 2) * sizeof(char *));
if (d == NULL) {
syslog(LOG_ERR, "malloc: %m");
exit(1);
@@ -197,7 +196,7 @@ main(int argc, char *argv[])
}
pw = getpwnam("_tftpd");
- if (!pw) {
+ if (pw == NULL) {
syslog(LOG_ERR, "no _tftpd: %m");
exit(1);
}
@@ -290,7 +289,7 @@ main(int argc, char *argv[])
if (pid < 0) {
sleep(i);
/*
- * flush out to most recently sent request.
+ * Flush out to most recently sent request.
*
* This may drop some requests, but those
* will be resent by the clients when
@@ -311,9 +310,8 @@ main(int argc, char *argv[])
CMSG_LEN(sizeof(struct sockaddr_storage));
i = recvmsg(fd, &msg, 0);
- if (i > 0) {
+ if (i > 0)
n = i;
- }
} else
break;
}
@@ -373,11 +371,11 @@ main(int argc, char *argv[])
void
tftp(struct tftphdr *tp, int size)
{
- char *cp;
- int i, first = 1, has_options = 0, ecode;
- struct formats *pf;
- char *filename, *mode = NULL, *option, *ccp;
- char fnbuf[MAXPATHLEN], nicebuf[MAXPATHLEN];
+ char *cp;
+ int i, first = 1, has_options = 0, ecode;
+ struct formats *pf;
+ char *filename, *mode = NULL, *option, *ccp;
+ char fnbuf[MAXPATHLEN], nicebuf[MAXPATHLEN];
cp = tp->th_stuff;
again:
@@ -435,7 +433,7 @@ again:
options[i].o_request = ++cp;
has_options = 1;
}
- cp = ccp-1;
+ cp = ccp - 1;
}
option_fail:
@@ -450,7 +448,7 @@ option_fail:
options[OPT_TIMEOUT].o_request = NULL;
}
- (void) strnvis(nicebuf, filename, MAXPATHLEN, VIS_SAFE|VIS_OCTAL);
+ (void)strnvis(nicebuf, filename, MAXPATHLEN, VIS_SAFE|VIS_OCTAL);
ecode = (*pf->f_validate)(filename, tp->th_opcode);
if (has_options)
oack();
@@ -484,24 +482,24 @@ option_fail:
int
validate_access(char *filename, int mode)
{
- struct stat stbuf;
- char *cp, **dirp;
- int fd, wmode;
+ struct stat stbuf;
+ char *cp, **dirp;
+ int fd, wmode;
if (!secure) {
if (*filename != '/')
return (EACCESS);
/*
- * prevent tricksters from getting around the directory
- * restrictions
+ * Prevent tricksters from getting around the directory
+ * restrictions.
*/
for (cp = filename + 1; *cp; cp++)
- if (*cp == '.' && strncmp(cp-1, "/../", 4) == 0)
- return(EACCESS);
+ if (*cp == '.' && strncmp(cp - 1, "/../", 4) == 0)
+ return (EACCESS);
for (dirp = dirs; *dirp; dirp++)
if (strncmp(filename, *dirp, strlen(*dirp)) == 0)
break;
- if (*dirp==0 && dirp!=dirs)
+ if (*dirp == 0 && dirp != dirs)
return (EACCESS);
}
@@ -517,14 +515,14 @@ validate_access(char *filename, int mode)
if ((errno == ENOENT) && (mode != RRQ))
wmode |= O_CREAT;
else
- return(EACCESS);
+ return (EACCESS);
}
} else {
if (mode == RRQ) {
- if ((stbuf.st_mode&(S_IREAD >> 6)) == 0)
+ if ((stbuf.st_mode & (S_IREAD >> 6)) == 0)
return (EACCESS);
} else {
- if ((stbuf.st_mode&(S_IWRITE >> 6)) == 0)
+ if ((stbuf.st_mode & (S_IWRITE >> 6)) == 0)
return (EACCESS);
}
}
@@ -532,9 +530,9 @@ validate_access(char *filename, int mode)
if (mode == RRQ)
options[OPT_TSIZE].o_reply = stbuf.st_size;
else
- /* XXX Allows writes of all sizes. */
+ /* XXX allows writes of all sizes */
options[OPT_TSIZE].o_reply =
- atoi(options[OPT_TSIZE].o_request);
+ atoi(options[OPT_TSIZE].o_request);
}
fd = open(filename, mode == RRQ ? O_RDONLY : (O_WRONLY|wmode), 0666);
if (fd < 0)
@@ -550,7 +548,7 @@ validate_access(char *filename, int mode)
return (serrno + 100);
}
- file = fdopen(fd, (mode == RRQ)? "r":"w");
+ file = fdopen(fd, mode == RRQ ? "r" : "w");
if (file == NULL) {
close(fd);
return (errno + 100);
@@ -564,11 +562,11 @@ validate_access(char *filename, int mode)
int
sendfile(struct formats *pf)
{
- struct tftphdr *dp, *r_init(void);
- struct tftphdr *ap; /* ack packet */
- struct pollfd pfd[1];
- volatile unsigned short block = 1;
- int n, size, nfds, error, timeouts;
+ struct tftphdr *dp, *r_init(void);
+ struct tftphdr *ap; /* ack packet */
+ struct pollfd pfd[1];
+ volatile unsigned short block = 1;
+ int n, size, nfds, error, timeouts;
dp = r_init();
ap = (struct tftphdr *)ackbuf;
@@ -628,7 +626,7 @@ sendfile(struct formats *pf)
if (ap->th_block == block)
break;
/* re-synchronize with the other side */
- (void) synchnet(peer);
+ (void)synchnet(peer);
if (ap->th_block == (block - 1))
continue;
}
@@ -649,11 +647,11 @@ abort:
int
recvfile(struct formats *pf)
{
- struct tftphdr *dp, *w_init(void);
- struct tftphdr *ap; /* ack buffer */
- struct pollfd pfd[1];
- volatile unsigned short block = 0;
- int n, size, nfds, error, timeouts;
+ struct tftphdr *dp, *w_init(void);
+ struct tftphdr *ap; /* ack buffer */
+ struct pollfd pfd[1];
+ volatile unsigned short block = 0;
+ int n, size, nfds, error, timeouts;
dp = w_init();
ap = (struct tftphdr *)ackbuf;
@@ -709,7 +707,7 @@ recvfile(struct formats *pf)
if (dp->th_block == block)
break;
/* re-synchronize with the other side */
- (void) synchnet(peer);
+ (void)synchnet(peer);
if (dp->th_block == (block - 1))
continue;
}
@@ -729,12 +727,12 @@ recvfile(struct formats *pf)
/* close data file */
write_behind(file, pf->f_convert);
- (void) fclose(file);
+ (void)fclose(file);
/* send final ack */
ap->th_opcode = htons((u_short)ACK);
ap->th_block = htons((u_short)(block));
- (void) send(peer, ackbuf, 4, 0);
+ (void)send(peer, ackbuf, 4, 0);
/* just quit on timeout */
pfd[0].fd = peer;
@@ -744,11 +742,11 @@ recvfile(struct formats *pf)
exit(1);
n = recv(peer, buf, sizeof(buf), 0);
/*
- * if read some data and got a data block then my last ack was lost
- * resend final ack
+ * If read some data and got a data block then my last ack was lost
+ * resend final ack.
*/
if (n >= 4 && dp->th_opcode == DATA && block == dp->th_block)
- (void) send(peer, ackbuf, 4, 0);
+ (void)send(peer, ackbuf, 4, 0);
abort:
return (1);
@@ -763,9 +761,9 @@ abort:
void
nak(int error)
{
- struct tftphdr *tp;
- struct errmsg *pe;
- int length;
+ struct tftphdr *tp;
+ struct errmsg *pe;
+ int length;
tp = (struct tftphdr *)buf;
tp->th_opcode = htons((u_short)ERROR);
@@ -790,10 +788,10 @@ nak(int error)
void
oack(void)
{
- struct tftphdr *tp, *ap;
- struct pollfd pfd[1];
- char *bp;
- int i, n, size, nfds, error, timeouts;
+ struct tftphdr *tp, *ap;
+ struct pollfd pfd[1];
+ char *bp;
+ int i, n, size, nfds, error, timeouts;
tp = (struct tftphdr *)buf;
bp = buf + 2;