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