usr/src/cmd/cmd-inet/usr.sbin/in.tftpd.c
author dm199272
Tue, 21 Aug 2007 08:09:49 -0700
changeset 4921 6179db775944
parent 1926 d83c7a9aec2d
child 8034 10557baff919
permissions -rw-r--r--
6425577 in.tftpd -S no longer works
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
1926
d83c7a9aec2d 6301318 in.tftpd fails with files >32MB with default block size
as198278
parents: 0
diff changeset
     5
 * Common Development and Distribution License (the "License").
d83c7a9aec2d 6301318 in.tftpd fails with files >32MB with default block size
as198278
parents: 0
diff changeset
     6
 * You may not use this file except in compliance with the License.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 *
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
    21
 * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
 * Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
 * All Rights Reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
 * University Copyright- Copyright (c) 1982, 1986, 1988
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
 * The Regents of the University of California.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
 * All Rights Reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
 * University Acknowledgment- Portions of this document are derived from
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
 * software developed by the University of California, Berkeley, and its
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
 * contributors.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
 * Trivial file transfer protocol server.  A top level process runs in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
 * an infinite loop fielding new TFTP requests.  A child process,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
 * communicating via a pipe with the top level process, sends delayed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
 * NAKs for those that we can't handle.  A new child process is created
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
 * to service each request that we can handle.  The top level process
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
 * exits after a period of time during which no new requests are
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
 * received.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
#include <sys/socket.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
#include <sys/wait.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
#include <sys/stat.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
#include <sys/time.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
#include <netinet/in.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
#include <arpa/inet.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
#include <dirent.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
#include <signal.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
#include <stdio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
#include <stdlib.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
#include <unistd.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
#include <errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
#include <ctype.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
#include <netdb.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
#include <setjmp.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
#include <syslog.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
#include <sys/param.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
#include <fcntl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
#include <pwd.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
#include <string.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
#include <priv_utils.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
#include "tftpcommon.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
#define	TIMEOUT		5
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
#define	DELAY_SECS	3
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
#define	DALLYSECS 60
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
#define	SYSLOG_MSG(message) \
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
	(syslog((((errno == ENETUNREACH) || (errno == EHOSTUNREACH) || \
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
		(errno == ECONNREFUSED)) ? LOG_WARNING : LOG_ERR), message))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
static int			rexmtval = TIMEOUT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
static int			maxtimeout = 5*TIMEOUT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
static int			securetftp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
static int			debug;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
static int			disable_pnp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
static int			standalone;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
static uid_t			uid_nobody = UID_NOBODY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
static uid_t			gid_nobody = GID_NOBODY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
static int			reqsock = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
				/* file descriptor of request socket */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
static socklen_t		fromlen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
static socklen_t		fromplen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
static struct sockaddr_storage	client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
static struct sockaddr_in6 	*sin6_ptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
static struct sockaddr_in	*sin_ptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
static struct sockaddr_in6	*from6_ptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
static struct sockaddr_in	*from_ptr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
static int			addrfmly;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
static int			peer;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
static off_t			tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
static tftpbuf			ackbuf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
static struct sockaddr_storage	from;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
static boolean_t		tsize_set;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
static pid_t			child;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
				/* pid of child handling delayed replys */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
static int			delay_fd [2];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
				/* pipe for communicating with child */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
static FILE			*file;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
static char			*filename;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
static union {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
	struct tftphdr	hdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
	char		data[SEGSIZE + 4];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
} buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
static union {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
	struct tftphdr	hdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
	char		data[SEGSIZE];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
} oackbuf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
struct	delay_info {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
	long	timestamp;		/* time request received */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
	int	ecode;			/* error code to return */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
	struct	sockaddr_storage from;	/* address of client */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
int	blocksize = SEGSIZE;	/* Number of data bytes in a DATA packet */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
 * Default directory for unqualified names
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
 * Used by TFTP boot procedures
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   137
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
static char	*homedir = "/tftpboot";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
struct formats {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   141
	char	*f_mode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   142
	int	(*f_validate)(int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   143
	void	(*f_send)(struct formats *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   144
	void	(*f_recv)(struct formats *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   145
	int	f_convert;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   146
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   147
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   148
static void	delayed_responder(void);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   149
static void	tftp(struct tftphdr *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   150
static int	validate_filename(int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   151
static void	tftpd_sendfile(struct formats *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   152
static void	tftpd_recvfile(struct formats *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   153
static void	nak(int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   154
static char	*blksize_handler(int, char *, int *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   155
static char	*timeout_handler(int, char *, int *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   156
static char	*tsize_handler(int, char *, int *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   157
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   158
static struct formats formats[] = {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   159
	{ "netascii",	validate_filename, tftpd_sendfile, tftpd_recvfile, 1 },
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   160
	{ "octet",	validate_filename, tftpd_sendfile, tftpd_recvfile, 0 },
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   161
	{ NULL }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   162
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   163
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   164
struct options {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   165
	char	*opt_name;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   166
	char	*(*opt_handler)(int, char *, int *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   167
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   168
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   169
static struct options options[] = {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   170
	{ "blksize",	blksize_handler },
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   171
	{ "timeout",	timeout_handler },
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   172
	{ "tsize",	tsize_handler },
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   173
	{ NULL }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   174
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   175
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   176
static char		optbuf[MAX_OPTVAL_LEN];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   177
static int		timeout;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   178
static sigjmp_buf	timeoutbuf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   179
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   180
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   181
main(int argc, char **argv)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   182
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   183
	struct tftphdr *tp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   184
	int n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   185
	int c;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   186
	struct	passwd *pwd;		/* for "nobody" entry */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   187
	struct in_addr ipv4addr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   188
	char abuf[INET6_ADDRSTRLEN];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   189
	socklen_t addrlen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   190
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   191
	openlog("tftpd", LOG_PID, LOG_DAEMON);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   192
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   193
	pwd = getpwnam("nobody");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   194
	if (pwd != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   195
		uid_nobody = pwd->pw_uid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   196
		gid_nobody = pwd->pw_gid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   197
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   198
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   199
	(void) __init_daemon_priv(
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   200
	    PU_LIMITPRIVS,
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   201
	    uid_nobody, gid_nobody,
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   202
	    PRIV_PROC_FORK, PRIV_PROC_CHROOT, PRIV_NET_PRIVADDR, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   203
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   204
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   205
	 *  Limit set is still "all."  Trim it down to just what we need:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   206
	 *  fork and chroot.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   207
	 */
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   208
	(void) priv_set(PRIV_SET, PRIV_ALLSETS,
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   209
	    PRIV_PROC_FORK, PRIV_PROC_CHROOT, PRIV_NET_PRIVADDR, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   210
	(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   211
	(void) priv_set(PRIV_SET, PRIV_INHERITABLE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   212
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   213
	while ((c = getopt(argc, argv, "dspS")) != EOF)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   214
		switch (c) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   215
		case 'd':		/* enable debug */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   216
			debug++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   217
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   218
		case 's':		/* secure daemon */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   219
			securetftp = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   220
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   221
		case 'p':		/* disable name pnp mapping */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   222
			disable_pnp = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   223
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   224
		case 'S':
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   225
			standalone = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   226
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   227
		case '?':
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   228
		default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   229
usage:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   230
			(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   231
			    "usage:  %s [-spd] [home-directory]\n", argv[0]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   232
			for (; optind < argc; optind++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   233
				syslog(LOG_ERR, "bad argument %s",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   234
				    argv[optind]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   235
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   236
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   237
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   238
	if (optind < argc)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   239
		if (optind == argc - 1 && *argv [optind] == '/')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   240
			homedir = argv [optind];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   241
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   242
			goto usage;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   243
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   244
	if (pipe(delay_fd) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   245
		syslog(LOG_ERR, "pipe (main): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   246
		exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   247
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   248
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   249
	(void) sigset(SIGCHLD, SIG_IGN); /* no zombies please */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   250
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   251
	if (standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   252
		socklen_t clientlen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   253
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   254
		sin6_ptr = (struct sockaddr_in6 *)&client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   255
		clientlen = sizeof (struct sockaddr_in6);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   256
		reqsock = socket(AF_INET6, SOCK_DGRAM, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   257
		if (reqsock == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   258
			perror("socket");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   259
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   260
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   261
		(void) memset(&client, 0, clientlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   262
		sin6_ptr->sin6_family = AF_INET6;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   263
		sin6_ptr->sin6_port = htons(IPPORT_TFTP);
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   264
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   265
		/* Enable privilege as tftp port is < 1024 */
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   266
		(void) priv_set(PRIV_SET,
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   267
		    PRIV_EFFECTIVE, PRIV_NET_PRIVADDR, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   268
		if (bind(reqsock, (struct sockaddr *)&client,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   269
		    clientlen) == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   270
			perror("bind");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   271
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   272
		}
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   273
		(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   274
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   275
		if (debug)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   276
			(void) puts("running in standalone mode...");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   277
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   278
		/* request socket passed on fd 0 by inetd */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   279
		reqsock = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   280
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   281
	if (debug) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   282
		int on = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   283
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   284
		(void) setsockopt(reqsock, SOL_SOCKET, SO_DEBUG,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   285
		    (char *)&on, sizeof (on));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   286
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   287
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   288
	(void) chdir(homedir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   289
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   290
	(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, PRIV_PROC_FORK, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   291
	if ((child = fork()) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   292
		syslog(LOG_ERR, "fork (main): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   293
		exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   294
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   295
	(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   296
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   297
	if (child == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   298
		(void) priv_set(PRIV_SET, PRIV_ALLSETS, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   299
		delayed_responder();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   300
	} /* child */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   301
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   302
	/* close read side of pipe */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   303
	(void) close(delay_fd[0]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   304
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   306
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   307
	 * Top level handling of incomming tftp requests.  Read a request
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   308
	 * and pass it off to be handled.  If request is valid, handling
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   309
	 * forks off and parent returns to this loop.  If no new requests
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   310
	 * are received for DALLYSECS, exit and return to inetd.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   311
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   312
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   313
	for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   314
		fd_set readfds;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   315
		struct timeval dally;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   316
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   317
		FD_ZERO(&readfds);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   318
		FD_SET(reqsock, &readfds);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   319
		dally.tv_sec = DALLYSECS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   320
		dally.tv_usec = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   321
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   322
		n = select(reqsock + 1, &readfds, NULL, NULL, &dally);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   323
		if (n < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   324
			if (errno == EINTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   325
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   326
			syslog(LOG_ERR, "select: %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   327
			(void) kill(child, SIGKILL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   328
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   329
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   330
		if (n == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   331
			/* Select timed out.  Its time to die. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   332
			if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   333
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   334
			else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   335
				(void) kill(child, SIGKILL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   336
				exit(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   337
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   338
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   339
		addrlen = sizeof (from);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   340
		if (getsockname(reqsock, (struct sockaddr  *)&from,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   341
		    &addrlen) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   342
			syslog(LOG_ERR, "getsockname: %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   343
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   344
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   345
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   346
		switch (from.ss_family) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   347
		case AF_INET:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   348
			fromlen = (socklen_t)sizeof (struct sockaddr_in);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   349
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   350
		case AF_INET6:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   351
			fromlen = (socklen_t)sizeof (struct sockaddr_in6);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   352
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   353
		default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   354
			syslog(LOG_ERR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   355
			    "Unknown address Family on peer connection %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   356
			    from.ss_family);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   357
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   358
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   359
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   360
		n = recvfrom(reqsock, &buf, sizeof (buf), 0,
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
   361
		    (struct sockaddr *)&from, &fromlen);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   362
		if (n < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   363
			if (errno == EINTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   364
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   365
			if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   366
				perror("recvfrom");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   367
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   368
				syslog(LOG_ERR, "recvfrom: %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   369
			(void) kill(child, SIGKILL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   370
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   371
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   372
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   373
		(void) alarm(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   374
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   375
		switch (from.ss_family) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   376
		case AF_INET:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   377
			addrfmly = AF_INET;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   378
			fromplen = sizeof (struct sockaddr_in);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   379
			sin_ptr = (struct sockaddr_in *)&client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   380
			(void) memset(&client, 0, fromplen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   381
			sin_ptr->sin_family = AF_INET;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   382
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   383
		case AF_INET6:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   384
			addrfmly = AF_INET6;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   385
			fromplen = sizeof (struct sockaddr_in6);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   386
			sin6_ptr = (struct sockaddr_in6 *)&client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   387
			(void) memset(&client, 0, fromplen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   388
			sin6_ptr->sin6_family = AF_INET6;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   389
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   390
		default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   391
			syslog(LOG_ERR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   392
			    "Unknown address Family on peer connection");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   393
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   394
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   395
		peer = socket(addrfmly, SOCK_DGRAM, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   396
		if (peer < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   397
			if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   398
				perror("socket (main)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   399
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   400
				syslog(LOG_ERR, "socket (main): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   401
			(void) kill(child, SIGKILL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   402
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   403
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   404
		if (debug) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   405
			int on = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   406
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   407
			(void) setsockopt(peer, SOL_SOCKET, SO_DEBUG,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   408
			    (char *)&on, sizeof (on));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   409
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   410
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   411
		if (bind(peer, (struct sockaddr *)&client, fromplen) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   412
			if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   413
				perror("bind (main)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   414
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   415
				syslog(LOG_ERR, "bind (main): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   416
			(void) kill(child, SIGKILL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   417
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   418
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   419
		if (standalone && debug) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   420
			sin6_ptr = (struct sockaddr_in6 *)&client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   421
			from6_ptr = (struct sockaddr_in6 *)&from;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   422
			if (IN6_IS_ADDR_V4MAPPED(&from6_ptr->sin6_addr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   423
				IN6_V4MAPPED_TO_INADDR(&from6_ptr->sin6_addr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   424
				    &ipv4addr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   425
				(void) inet_ntop(AF_INET, &ipv4addr, abuf,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   426
				    sizeof (abuf));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   427
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   428
				(void) inet_ntop(AF_INET6,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   429
				    &from6_ptr->sin6_addr, abuf,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   430
				    sizeof (abuf));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   431
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   432
			/* get local port */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   433
			if (getsockname(peer, (struct sockaddr *)&client,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   434
			    &fromplen) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   435
				perror("getsockname (main)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   436
			(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   437
			    "request from %s port %d; local port %d\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   438
			    abuf, from6_ptr->sin6_port, sin6_ptr->sin6_port);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   439
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   440
		tp = &buf.hdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   441
		tp->th_opcode = ntohs((ushort_t)tp->th_opcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   442
		if (tp->th_opcode == RRQ || tp->th_opcode == WRQ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   443
			tftp(tp, n);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   444
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   445
		(void) close(peer);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   446
		(void) fclose(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   447
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   448
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   449
	/*NOTREACHED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   450
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   451
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   452
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   453
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   454
delayed_responder(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   455
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   456
	struct delay_info dinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   457
	long now;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   458
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   459
	/* we don't use the descriptors passed in to the parent */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   460
	(void) close(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   461
	(void) close(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   462
	if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   463
		(void) close(reqsock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   464
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   465
	/* close write side of pipe */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   466
	(void) close(delay_fd[1]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   467
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   468
	for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   469
		int n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   470
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   471
		if ((n = read(delay_fd[0], &dinfo,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   472
		    sizeof (dinfo))) != sizeof (dinfo)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   473
			if (n < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   474
				if (errno == EINTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   475
					continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   476
				if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   477
					perror("read from pipe "
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   478
					    "(delayed responder)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   479
				else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   480
					syslog(LOG_ERR, "read from pipe: %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   481
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   482
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   483
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   484
		switch (dinfo.from.ss_family) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   485
		case AF_INET:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   486
			addrfmly = AF_INET;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   487
			fromplen = sizeof (struct sockaddr_in);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   488
			sin_ptr = (struct sockaddr_in *)&client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   489
			(void) memset(&client, 0, fromplen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   490
			sin_ptr->sin_family = AF_INET;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   491
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   492
		case AF_INET6:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   493
			addrfmly = AF_INET6;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   494
			fromplen = sizeof (struct sockaddr_in6);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   495
			sin6_ptr = (struct sockaddr_in6 *)&client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   496
			(void) memset(&client, 0, fromplen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   497
			sin6_ptr->sin6_family = AF_INET6;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   498
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   499
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   500
		peer = socket(addrfmly, SOCK_DGRAM, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   501
		if (peer == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   502
			if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   503
				perror("socket (delayed responder)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   504
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   505
				syslog(LOG_ERR, "socket (delay): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   506
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   507
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   508
		if (debug) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   509
			int on = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   510
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   511
			(void) setsockopt(peer, SOL_SOCKET, SO_DEBUG,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   512
			    (char *)&on, sizeof (on));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   513
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   514
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   515
		if (bind(peer, (struct sockaddr *)&client, fromplen) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   516
			if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   517
				perror("bind (delayed responder)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   518
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   519
				syslog(LOG_ERR, "bind (delay): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   520
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   521
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   522
		if (client.ss_family == AF_INET) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   523
			from_ptr = (struct sockaddr_in *)&dinfo.from;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   524
			from_ptr->sin_family = AF_INET;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   525
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   526
			from6_ptr = (struct sockaddr_in6 *)&dinfo.from;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   527
			from6_ptr->sin6_family = AF_INET6;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   528
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   529
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   530
		 * Since a request hasn't been received from the client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   531
		 * before the delayed responder process is forked, the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   532
		 * from variable is uninitialized.  So set it to contain
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   533
		 * the client address.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   534
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   535
		from = dinfo.from;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   536
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   537
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   538
		 * only sleep if DELAY_SECS has not elapsed since
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   539
		 * original request was received.  Ensure that `now'
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   540
		 * is not earlier than `dinfo.timestamp'
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   541
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   542
		now = time(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   543
		if ((uint_t)(now - dinfo.timestamp) < DELAY_SECS)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   544
			(void) sleep(DELAY_SECS - (now - dinfo.timestamp));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   545
		nak(dinfo.ecode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   546
		(void) close(peer);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   547
	} /* for */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   548
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   549
	/* NOTREACHED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   550
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   551
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   552
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   553
 * Handle the Blocksize option.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   554
 * Return the blksize option value string to include in the OACK reply.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   555
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   556
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   557
static char *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   558
blksize_handler(int opcode, char *optval, int *errcode)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   559
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   560
	char *endp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   561
	int value;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   562
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   563
	*errcode = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   564
	errno = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   565
	value = (int)strtol(optval, &endp, 10);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   566
	if (errno != 0 || value < MIN_BLKSIZE || *endp != '\0')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   567
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   568
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   569
	 * As the blksize value in the OACK reply can be less than the value
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   570
	 * requested, to support broken clients if the value requested is larger
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   571
	 * than allowed in the RFC, reply with the maximum value permitted.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   572
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   573
	if (value > MAX_BLKSIZE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   574
		value = MAX_BLKSIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   575
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   576
	blocksize = value;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   577
	(void) snprintf(optbuf, sizeof (optbuf), "%d", blocksize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   578
	return (optbuf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   579
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   580
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   581
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   582
 * Handle the Timeout Interval option.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   583
 * Return the timeout option value string to include in the OACK reply.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   584
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   585
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   586
static char *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   587
timeout_handler(int opcode, char *optval, int *errcode)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   588
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   589
	char *endp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   590
	int value;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   591
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   592
	*errcode = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   593
	errno = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   594
	value = (int)strtol(optval, &endp, 10);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   595
	if (errno != 0 || *endp != '\0')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   596
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   597
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   598
	 * The timeout value in the OACK reply must match the value specified
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   599
	 * by the client, so if an invalid timeout is requested don't include
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   600
	 * the timeout option in the OACK reply.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   601
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   602
	if (value < MIN_TIMEOUT || value > MAX_TIMEOUT)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   603
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   604
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   605
	rexmtval = value;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   606
	maxtimeout = 5 * rexmtval;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   607
	(void) snprintf(optbuf, sizeof (optbuf), "%d", rexmtval);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   608
	return (optbuf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   609
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   610
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   611
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   612
 * Handle the Transfer Size option.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   613
 * Return the tsize option value string to include in the OACK reply.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   614
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   615
static char *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   616
tsize_handler(int opcode, char *optval, int *errcode)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   617
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   618
	char *endp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   619
	longlong_t value;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   620
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   621
	*errcode = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   622
	errno = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   623
	value = strtoll(optval, &endp, 10);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   624
	if (errno != 0 || value < 0 || *endp != '\0')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   625
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   626
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   627
	if (opcode == RRQ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   628
		if (tsize_set == B_FALSE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   629
			return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   630
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   631
		 * The tsize value should be 0 for a read request, but to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   632
		 * support broken clients we don't check that it is.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   633
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   634
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   635
#if _FILE_OFFSET_BITS == 32
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   636
		if (value > MAXOFF_T) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   637
			*errcode = ENOSPACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   638
			return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   639
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   640
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   641
		tsize = value;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   642
		tsize_set = B_TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   643
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   644
	(void) snprintf(optbuf, sizeof (optbuf), OFF_T_FMT, tsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   645
	return (optbuf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   646
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   647
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   648
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   649
 * Process any options included by the client in the request packet.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   650
 * Return the size of the OACK reply packet built or 0 for no OACK reply.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   651
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   652
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   653
process_options(int opcode, char *opts, char *endopts)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   654
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   655
	char *cp, *optname, *optval, *ostr, *oackend;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   656
	struct tftphdr *oackp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   657
	int i, errcode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   658
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   659
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   660
	 * To continue to interoperate with broken TFTP clients, ignore
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   661
	 * null padding appended to requests which don't include options.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   662
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   663
	cp = opts;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   664
	while ((cp < endopts) && (*cp == '\0'))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   665
		cp++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   666
	if (cp == endopts)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   667
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   668
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   669
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   670
	 * Construct an Option ACKnowledgement packet if any requested option
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   671
	 * is recognized.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   672
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   673
	oackp = &oackbuf.hdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   674
	oackend = oackbuf.data + sizeof (oackbuf.data);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   675
	oackp->th_opcode = htons((ushort_t)OACK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   676
	cp = (char *)&oackp->th_stuff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   677
	while (opts < endopts) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   678
		optname = opts;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   679
		if ((optval = next_field(optname, endopts)) == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   680
			nak(EOPTNEG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   681
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   682
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   683
		if ((opts = next_field(optval, endopts)) == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   684
			nak(EOPTNEG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   685
			exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   686
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   687
		for (i = 0; options[i].opt_name != NULL; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   688
			if (strcasecmp(optname, options[i].opt_name) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   689
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   690
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   691
		if (options[i].opt_name != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   692
			ostr = options[i].opt_handler(opcode, optval, &errcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   693
			if (ostr != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   694
				cp += strlcpy(cp, options[i].opt_name,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   695
				    oackend - cp) + 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   696
				if (cp <= oackend)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   697
					cp += strlcpy(cp, ostr, oackend - cp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   698
					    + 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   699
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   700
				if (cp > oackend) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   701
					nak(EOPTNEG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   702
					exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   703
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   704
			} else if (errcode >= 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   705
				nak(errcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   706
				exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   707
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   708
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   709
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   710
	if (cp != (char *)&oackp->th_stuff)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   711
		return (cp - oackbuf.data);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   712
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   713
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   714
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   715
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   716
 * Handle access errors caused by client requests.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   717
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   718
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   719
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   720
delay_exit(int ecode)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   721
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   722
	struct delay_info dinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   723
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   724
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   725
	 * The most likely cause of an error here is that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   726
	 * someone has broadcast an RRQ packet because s/he's
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   727
	 * trying to boot and doesn't know who the server is.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   728
	 * Rather then sending an ERROR packet immediately, we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   729
	 * wait a while so that the real server has a better chance
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   730
	 * of getting through (in case client has lousy Ethernet
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   731
	 * interface).  We write to a child that handles delayed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   732
	 * ERROR packets to avoid delaying service to new
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   733
	 * requests.  Of course, we would rather just not answer
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   734
	 * RRQ packets that are broadcasted, but there's no way
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   735
	 * for a user process to determine this.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   736
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   737
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   738
	dinfo.timestamp = time(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   739
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   740
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   741
	 * If running in secure mode, we map all errors to EACCESS
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   742
	 * so that the client gets no information about which files
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   743
	 * or directories exist.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   744
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   745
	if (securetftp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   746
		dinfo.ecode = EACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   747
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   748
		dinfo.ecode = ecode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   749
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   750
	dinfo.from = from;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   751
	if (write(delay_fd[1], &dinfo, sizeof (dinfo)) !=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   752
	    sizeof (dinfo)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   753
		syslog(LOG_ERR, "delayed write failed.");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   754
		(void) kill(child, SIGKILL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   755
		exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   756
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   757
	exit(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   758
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   759
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   760
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   761
 * Handle initial connection protocol.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   762
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   763
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   764
tftp(struct tftphdr *tp, int size)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   765
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   766
	char *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   767
	int readmode, ecode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   768
	struct formats *pf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   769
	char *mode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   770
	int fd;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   771
	static boolean_t firsttime = B_TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   772
	int oacklen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   773
	struct stat statb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   774
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   775
	readmode = (tp->th_opcode == RRQ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   776
	filename = (char *)&tp->th_stuff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   777
	mode = next_field(filename, &buf.data[size]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   778
	cp = (mode != NULL) ? next_field(mode, &buf.data[size]) : NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   779
	if (cp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   780
		nak(EBADOP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   781
		exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   782
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   783
	if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   784
		(void) fprintf(stderr, "%s for %s %s ",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   785
		    readmode ? "RRQ" : "WRQ", filename, mode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   786
		print_options(stderr, cp, size + buf.data - cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   787
		(void) putc('\n', stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   788
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   789
	for (pf = formats; pf->f_mode != NULL; pf++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   790
		if (strcasecmp(pf->f_mode, mode) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   791
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   792
	if (pf->f_mode == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   793
		nak(EBADOP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   794
		exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   795
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   796
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   797
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   798
	 * XXX fork a new process to handle this request before
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   799
	 * chroot(), otherwise the parent won't be able to create a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   800
	 * new socket as that requires library access to system files
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   801
	 * and devices.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   802
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   803
	(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, PRIV_PROC_FORK, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   804
	switch (fork()) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   805
	case -1:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   806
		syslog(LOG_ERR, "fork (tftp): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   807
		(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   808
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   809
	case 0:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   810
		(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   811
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   812
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   813
		(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   814
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   815
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   816
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   817
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   818
	 * Try to see if we can access the file.  The access can still
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   819
	 * fail later if we are running in secure mode because of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   820
	 * the chroot() call.  We only want to execute the chroot()  once.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   821
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   822
	if (securetftp && firsttime) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   823
		(void) priv_set(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   824
		    PRIV_SET, PRIV_EFFECTIVE, PRIV_PROC_CHROOT, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   825
		if (chroot(homedir) == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   826
			syslog(LOG_ERR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   827
			    "tftpd: cannot chroot to directory %s: %m\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   828
			    homedir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   829
			delay_exit(EACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   830
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   831
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   832
		{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   833
			firsttime = B_FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   834
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   835
		(void) priv_set(PRIV_SET, PRIV_EFFECTIVE, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   836
		(void) chdir("/");  /* cd to  new root */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   837
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   838
	(void) priv_set(PRIV_SET, PRIV_ALLSETS, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   839
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   840
	ecode = (*pf->f_validate)(tp->th_opcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   841
	if (ecode != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   842
		delay_exit(ecode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   843
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   844
	/* we don't use the descriptors passed in to the parent */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   845
	(void) close(STDIN_FILENO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   846
	(void) close(STDOUT_FILENO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   847
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   848
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   849
	 * Try to open file as low-priv setuid/setgid.  Note that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   850
	 * a chroot() has already been done.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   851
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   852
	fd = open(filename,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   853
	    (readmode ? O_RDONLY : (O_WRONLY|O_TRUNC)) | O_NONBLOCK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   854
	if ((fd < 0) || (fstat(fd, &statb) < 0))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   855
		delay_exit((errno == ENOENT) ? ENOTFOUND : EACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   856
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   857
	if (((statb.st_mode & ((readmode) ? S_IROTH : S_IWOTH)) == 0) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   858
	    ((statb.st_mode & S_IFMT) != S_IFREG))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   859
		delay_exit(EACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   860
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   861
	file = fdopen(fd, readmode ? "r" : "w");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   862
	if (file == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   863
		delay_exit(errno + 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   864
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   865
	/* Don't know the size of transfers which involve conversion */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   866
	tsize_set = (readmode && (pf->f_convert == 0));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   867
	if (tsize_set)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   868
		tsize = statb.st_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   869
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   870
	/* Deal with any options sent by the client */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   871
	oacklen = process_options(tp->th_opcode, cp, buf.data + size);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   872
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   873
	if (tp->th_opcode == WRQ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   874
		(*pf->f_recv)(pf, oacklen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   875
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   876
		(*pf->f_send)(pf, oacklen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   877
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   878
	exit(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   879
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   880
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   881
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   882
 *	Maybe map filename into another one.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   883
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   884
 *	For PNP, we get TFTP boot requests for filenames like
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   885
 *	<Unknown Hex IP Addr>.<Architecture Name>.   We must
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   886
 *	map these to 'pnp.<Architecture Name>'.  Note that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   887
 *	uppercase is mapped to lowercase in the architecture names.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   888
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   889
 *	For names <Hex IP Addr> there are two cases.  First,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   890
 *	it may be a buggy prom that omits the architecture code.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   891
 *	So first check if <Hex IP Addr>.<arch> is on the filesystem.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   892
 *	Second, this is how most Sun3s work; assume <arch> is sun3.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   893
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   894
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   895
static char *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   896
pnp_check(char *origname)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   897
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   898
	static char buf [MAXNAMLEN + 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   899
	char *arch, *s, *bufend;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   900
	in_addr_t ipaddr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   901
	int len = (origname ? strlen(origname) : 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   902
	DIR *dir;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   903
	struct dirent *dp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   904
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   905
	if (securetftp || disable_pnp || len < 8 || len > 14)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   906
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   907
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   908
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   909
	 * XXX see if this cable allows pnp; if not, return NULL
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   910
	 * Requires YP support for determining this!
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   911
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   912
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   913
	ipaddr = htonl(strtol(origname, &arch, 16));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   914
	if ((arch == NULL) || (len > 8 && *arch != '.'))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   915
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   916
	if (len == 8)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   917
		arch = "SUN3";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   918
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   919
		arch++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   920
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   921
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   922
	 * Allow <Hex IP Addr>* filename request to to be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   923
	 * satisfied by <Hex IP Addr><Any Suffix> rather
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   924
	 * than enforcing this to be Sun3 systems.  Also serves
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   925
	 * to make case of suffix a don't-care.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   926
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   927
	if ((dir = opendir(homedir)) == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   928
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   929
	while ((dp = readdir(dir)) != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   930
		if (strncmp(origname, dp->d_name, 8) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   931
			(void) strlcpy(buf, dp->d_name, sizeof (buf));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   932
			(void) closedir(dir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   933
			return (buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   934
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   935
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   936
	(void) closedir(dir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   937
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   938
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   939
	 * XXX maybe call YP master for most current data iff
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   940
	 * pnp is enabled.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   941
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   942
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   943
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   944
	 * only do mapping PNP boot file name for machines that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   945
	 * are not in the hosts database.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   946
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   947
	if (gethostbyaddr((char *)&ipaddr, sizeof (ipaddr), AF_INET) != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   948
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   949
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   950
	s = buf + strlcpy(buf, "pnp.", sizeof (buf));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   951
	bufend = &buf[sizeof (buf) - 1];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   952
	while ((*arch != '\0') && (s < bufend))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   953
		*s++ = tolower (*arch++);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   954
	*s = '\0';
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   955
	return (buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   956
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   957
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   958
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   959
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   960
 * Try to validate filename. If the filename doesn't exist try PNP mapping.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   961
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   962
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   963
validate_filename(int mode)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   964
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   965
	struct stat stbuf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   966
	char *origfile;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   967
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   968
	if (stat(filename, &stbuf) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   969
		if (errno != ENOENT)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   970
			return (EACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   971
		if (mode == WRQ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   972
			return (ENOTFOUND);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   973
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   974
		/* try to map requested filename into a pnp filename */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   975
		origfile = filename;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   976
		filename = pnp_check(origfile);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   977
		if (filename == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   978
			return (ENOTFOUND);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   979
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   980
		if (stat(filename, &stbuf) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   981
			return (errno == ENOENT ? ENOTFOUND : EACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   982
		syslog(LOG_NOTICE, "%s -> %s\n", origfile, filename);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   983
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   984
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   985
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   986
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   987
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   988
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   989
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   990
timer(int signum)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   991
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   992
	timeout += rexmtval;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   993
	if (timeout >= maxtimeout)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   994
		exit(1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   995
	siglongjmp(timeoutbuf, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   996
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   997
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   998
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   999
 * Send the requested file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1000
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1001
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1002
tftpd_sendfile(struct formats *pf, int oacklen)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1003
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1004
	struct tftphdr *dp;
1926
d83c7a9aec2d 6301318 in.tftpd fails with files >32MB with default block size
as198278
parents: 0
diff changeset
  1005
	volatile ushort_t block = 1;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1006
	int size, n, serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1007
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1008
	if (oacklen != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1009
		(void) sigset(SIGALRM, timer);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1010
		timeout = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1011
		(void) sigsetjmp(timeoutbuf, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1012
		if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1013
			(void) fputs("Sending OACK ", stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1014
			print_options(stderr, (char *)&oackbuf.hdr.th_stuff,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1015
			    oacklen - 2);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1016
			(void) putc('\n', stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1017
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1018
		if (sendto(peer, &oackbuf, oacklen, 0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1019
		    (struct sockaddr *)&from, fromplen) != oacklen) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1020
			if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1021
				serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1022
				perror("sendto (oack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1023
				errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1024
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1025
			SYSLOG_MSG("sendto (oack): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1026
			goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1027
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1028
		(void) alarm(rexmtval); /* read the ack */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1029
		for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1030
			(void) sigrelse(SIGALRM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1031
			n = recv(peer, &ackbuf, sizeof (ackbuf), 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1032
			(void) sighold(SIGALRM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1033
			if (n < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1034
				if (errno == EINTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1035
					continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1036
				serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1037
				SYSLOG_MSG("recv (ack): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1038
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1039
					errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1040
					perror("recv (ack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1041
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1042
				goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1043
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1044
			ackbuf.tb_hdr.th_opcode =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1045
			    ntohs((ushort_t)ackbuf.tb_hdr.th_opcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1046
			ackbuf.tb_hdr.th_block =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1047
			    ntohs((ushort_t)ackbuf.tb_hdr.th_block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1048
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1049
			if (ackbuf.tb_hdr.th_opcode == ERROR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1050
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1051
					(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1052
					    "received ERROR %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1053
					    ackbuf.tb_hdr.th_code);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1054
					if (n > 4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1055
						(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1056
						    " %.*s", n - 4,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1057
						    ackbuf.tb_hdr.th_msg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1058
					(void) putc('\n', stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1059
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1060
				goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1061
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1062
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1063
			if (ackbuf.tb_hdr.th_opcode == ACK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1064
				if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1065
					(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1066
					    "received ACK for block %d\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1067
					    ackbuf.tb_hdr.th_block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1068
				if (ackbuf.tb_hdr.th_block == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1069
					break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1070
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1071
				 * Don't resend the OACK, avoids getting stuck
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1072
				 * in an OACK/ACK loop if the client keeps
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1073
				 * replying with a bad ACK. Client will either
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1074
				 * send a good ACK or timeout sending bad ones.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1075
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1076
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1077
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1078
		cancel_alarm();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1079
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1080
	dp = r_init();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1081
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1082
		(void) sigset(SIGALRM, timer);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1083
		size = readit(file, &dp, pf->f_convert);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1084
		if (size < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1085
			nak(errno + 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1086
			goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1087
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1088
		dp->th_opcode = htons((ushort_t)DATA);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1089
		dp->th_block = htons((ushort_t)block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1090
		timeout = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1091
		(void) sigsetjmp(timeoutbuf, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1092
		if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1093
			(void) fprintf(stderr, "Sending DATA block %d\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1094
			    block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1095
		if (sendto(peer, dp, size + 4, 0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1096
		    (struct sockaddr *)&from,  fromplen) != size + 4) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1097
			if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1098
				serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1099
				perror("sendto (data)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1100
				errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1101
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1102
			SYSLOG_MSG("sendto (data): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1103
			goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1104
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1105
		read_ahead(file, pf->f_convert);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1106
		(void) alarm(rexmtval); /* read the ack */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1107
		for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1108
			(void) sigrelse(SIGALRM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1109
			n = recv(peer, &ackbuf, sizeof (ackbuf), 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1110
			(void) sighold(SIGALRM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1111
			if (n < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1112
				if (errno == EINTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1113
					continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1114
				serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1115
				SYSLOG_MSG("recv (ack): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1116
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1117
					errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1118
					perror("recv (ack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1119
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1120
				goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1121
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1122
			ackbuf.tb_hdr.th_opcode =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1123
			    ntohs((ushort_t)ackbuf.tb_hdr.th_opcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1124
			ackbuf.tb_hdr.th_block =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1125
			    ntohs((ushort_t)ackbuf.tb_hdr.th_block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1126
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1127
			if (ackbuf.tb_hdr.th_opcode == ERROR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1128
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1129
					(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1130
					    "received ERROR %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1131
					    ackbuf.tb_hdr.th_code);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1132
					if (n > 4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1133
						(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1134
						    " %.*s", n - 4,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1135
						    ackbuf.tb_hdr.th_msg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1136
					(void) putc('\n', stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1137
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1138
				goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1139
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1140
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1141
			if (ackbuf.tb_hdr.th_opcode == ACK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1142
				if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1143
					(void) fprintf(stderr,
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
  1144
					    "received ACK for block %d\n",
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
  1145
					    ackbuf.tb_hdr.th_block);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1146
				if (ackbuf.tb_hdr.th_block == block) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1147
					break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1148
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1149
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1150
				 * Never resend the current DATA packet on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1151
				 * receipt of a duplicate ACK, doing so would
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1152
				 * cause the "Sorcerer's Apprentice Syndrome".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1153
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1154
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1155
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1156
		cancel_alarm();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1157
		block++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1158
	} while (size == blocksize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1159
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1160
abort:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1161
	cancel_alarm();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1162
	(void) fclose(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1163
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1164
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1165
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1166
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1167
justquit(int signum)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1168
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1169
	exit(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1170
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1171
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1172
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1173
 * Receive a file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1174
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1175
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1176
tftpd_recvfile(struct formats *pf, int oacklen)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1177
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1178
	struct tftphdr *dp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1179
	struct tftphdr *ap;    /* ack buffer */
1926
d83c7a9aec2d 6301318 in.tftpd fails with files >32MB with default block size
as198278
parents: 0
diff changeset
  1180
	ushort_t block = 0;
d83c7a9aec2d 6301318 in.tftpd fails with files >32MB with default block size
as198278
parents: 0
diff changeset
  1181
	int n, size, acklen, serrno;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1182
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1183
	dp = w_init();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1184
	ap = &ackbuf.tb_hdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1185
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1186
		(void) sigset(SIGALRM, timer);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1187
		timeout = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1188
		if (oacklen == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1189
			ap->th_opcode = htons((ushort_t)ACK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1190
			ap->th_block = htons((ushort_t)block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1191
			acklen = 4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1192
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1193
			/* copy OACK packet to the ack buffer ready to send */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1194
			(void) memcpy(&ackbuf, &oackbuf, oacklen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1195
			acklen = oacklen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1196
			oacklen = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1197
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1198
		block++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1199
		(void) sigsetjmp(timeoutbuf, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1200
send_ack:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1201
		if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1202
			if (ap->th_opcode == htons((ushort_t)ACK)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1203
				(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1204
				    "Sending ACK for block %d\n", block - 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1205
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1206
				(void) fprintf(stderr, "Sending OACK ");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1207
				print_options(stderr, (char *)&ap->th_stuff,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1208
				    acklen - 2);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1209
				(void) putc('\n', stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1210
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1211
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1212
		if (sendto(peer, &ackbuf, acklen, 0, (struct sockaddr *)&from,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1213
		    fromplen) != acklen) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1214
			if (ap->th_opcode == htons((ushort_t)ACK)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1215
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1216
					serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1217
					perror("sendto (ack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1218
					errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1219
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1220
				syslog(LOG_ERR, "sendto (ack): %m\n");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1221
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1222
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1223
					serrno = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1224
					perror("sendto (oack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1225
					errno = serrno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1226
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1227
				syslog(LOG_ERR, "sendto (oack): %m\n");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1228
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1229
			goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1230
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1231
		if (write_behind(file, pf->f_convert) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1232
			nak(errno + 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1233
			goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1234
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1235
		(void) alarm(rexmtval);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1236
		for (;;) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1237
			(void) sigrelse(SIGALRM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1238
			n = recv(peer, dp, blocksize + 4, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1239
			(void) sighold(SIGALRM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1240
			if (n < 0) { /* really? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1241
				if (errno == EINTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1242
					continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1243
				syslog(LOG_ERR, "recv (data): %m");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1244
				goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1245
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1246
			dp->th_opcode = ntohs((ushort_t)dp->th_opcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1247
			dp->th_block = ntohs((ushort_t)dp->th_block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1248
			if (dp->th_opcode == ERROR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1249
				cancel_alarm();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1250
				if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1251
					(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1252
					    "received ERROR %d", dp->th_code);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1253
					if (n > 4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1254
						(void) fprintf(stderr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1255
						    " %.*s", n - 4, dp->th_msg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1256
					(void) putc('\n', stderr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1257
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1258
				return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1259
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1260
			if (dp->th_opcode == DATA) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1261
				if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1262
					(void) fprintf(stderr,
4921
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
  1263
					    "Received DATA block %d\n",
6179db775944 6425577 in.tftpd -S no longer works
dm199272
parents: 1926
diff changeset
  1264
					    dp->th_block);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1265
				if (dp->th_block == block) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1266
					break;   /* normal */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1267
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1268
				/* Re-synchronize with the other side */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1269
				if (synchnet(peer) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1270
					nak(errno + 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1271
					goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1272
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1273
				if (dp->th_block == (block-1))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1274
					goto send_ack; /* rexmit */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1275
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1276
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1277
		cancel_alarm();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1278
		/*  size = write(file, dp->th_data, n - 4); */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1279
		size = writeit(file, &dp, n - 4, pf->f_convert);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1280
		if (size != (n - 4)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1281
			nak((size < 0) ? (errno + 100) : ENOSPACE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1282
			goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1283
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1284
	} while (size == blocksize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1285
	if (write_behind(file, pf->f_convert) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1286
		nak(errno + 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1287
		goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1288
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1289
	n = fclose(file);	/* close data file */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1290
	file = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1291
	if (n == EOF) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1292
		nak(errno + 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1293
		goto abort;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1294
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1295
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1296
	ap->th_opcode = htons((ushort_t)ACK);    /* send the "final" ack */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1297
	ap->th_block = htons((ushort_t)(block));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1298
	if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1299
		(void) fprintf(stderr, "Sending ACK for block %d\n", block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1300
	if (sendto(peer, &ackbuf, 4, 0, (struct sockaddr *)&from,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1301
	    fromplen) == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1302
		if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1303
			perror("sendto (ack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1304
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1305
	(void) sigset(SIGALRM, justquit); /* just quit on timeout */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1306
	(void) alarm(rexmtval);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1307
	/* normally times out and quits */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1308
	n = recv(peer, dp, blocksize + 4, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1309
	(void) alarm(0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1310
	dp->th_opcode = ntohs((ushort_t)dp->th_opcode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1311
	dp->th_block = ntohs((ushort_t)dp->th_block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1312
	if (n >= 4 &&		/* if read some data */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1313
	    dp->th_opcode == DATA && /* and got a data block */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1314
	    block == dp->th_block) {	/* then my last ack was lost */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1315
		if (debug && standalone) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1316
			(void) fprintf(stderr, "Sending ACK for block %d\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1317
			    block);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1318
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1319
		/* resend final ack */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1320
		if (sendto(peer, &ackbuf, 4, 0, (struct sockaddr *)&from,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1321
		    fromplen) == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1322
			if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1323
				perror("sendto (last ack)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1324
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1325
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1326
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1327
abort:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1328
	cancel_alarm();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1329
	if (file != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1330
		(void) fclose(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1331
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1332
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1333
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1334
 * Send a nak packet (error message).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1335
 * Error code passed in is one of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1336
 * standard TFTP codes, or a UNIX errno
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1337
 * offset by 100.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1338
 * Handles connected as well as unconnected peer.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1339
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1340
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1341
nak(int error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1342
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1343
	struct tftphdr *tp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1344
	int length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1345
	struct errmsg *pe;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1346
	int ret;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1347
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1348
	tp = &buf.hdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1349
	tp->th_opcode = htons((ushort_t)ERROR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1350
	tp->th_code = htons((ushort_t)error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1351
	for (pe = errmsgs; pe->e_code >= 0; pe++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1352
		if (pe->e_code == error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1353
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1354
	if (pe->e_code < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1355
		pe->e_msg = strerror(error - 100);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1356
		tp->th_code = EUNDEF;   /* set 'undef' errorcode */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1357
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1358
	(void) strlcpy(tp->th_msg, (pe->e_msg != NULL) ? pe->e_msg : "UNKNOWN",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1359
	    sizeof (buf) - sizeof (struct tftphdr));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1360
	length = strlen(tp->th_msg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1361
	length += sizeof (struct tftphdr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1362
	if (debug && standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1363
		(void) fprintf(stderr, "Sending NAK: %s\n", tp->th_msg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1364
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1365
	ret = sendto(peer, &buf, length, 0, (struct sockaddr *)&from,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1366
	    fromplen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1367
	if (ret == -1 && errno == EISCONN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1368
		/* Try without an address */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1369
		ret = send(peer, &buf, length, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1370
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1371
	if (ret == -1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1372
		if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1373
			perror("sendto (nak)");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1374
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1375
			syslog(LOG_ERR, "tftpd: nak: %m\n");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1376
	} else if (ret != length) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1377
		if (standalone)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1378
			perror("sendto (nak) lost data");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1379
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1380
			syslog(LOG_ERR, "tftpd: nak: %d lost\n", length - ret);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1381
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1382
}