 cc47c8fa7b
			
		
	
	cc47c8fa7b
	
	
	
		
			
			We turn on a few extra warnings and fix the fallout that occurs when building with --enable-developer. Note that we get different warnings on different machines and so this will be a work in progress. So far, we have built on NetBSD/amd64 5.99.64 (which uses gcc 4.5.3) and Ubuntu 10.04.3 LTS (which uses gcc 4.4.3). Notably, we fixed 1. a lot of missing structure initialisers, 2. unchecked return values for functions that glibc marks as __attribute__((warn-unused-result)), 3. made minor modifications to slc and asn1_compile which can generate code which generates warnings, and 4. a few stragglers here and there. We turned off the extended warnings for many programs in appl/ as they are nearing the end of their useful lifetime, e.g. rsh, rcp, popper, ftp and telnet. Interestingly, glibc's strncmp() macro needed to be worked around whereas the function calls did not. We have not yet tried this on 32 bit platforms, so there will be a few more warnings when we do.
		
			
				
	
	
		
			792 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			792 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 1983, 1990, 1992, 1993
 | |
|  *	The Regents of the University of California.  All rights reserved.
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without
 | |
|  * modification, are permitted provided that the following conditions
 | |
|  * are met:
 | |
|  * 1. Redistributions of source code must retain the above copyright
 | |
|  *    notice, this list of conditions and the following disclaimer.
 | |
|  * 2. Redistributions in binary form must reproduce the above copyright
 | |
|  *    notice, this list of conditions and the following disclaimer in the
 | |
|  *    documentation and/or other materials provided with the distribution.
 | |
|  * 3. Neither the name of the University nor the names of its contributors
 | |
|  *    may be used to endorse or promote products derived from this software
 | |
|  *    without specific prior written permission.
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 | |
|  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | |
|  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 | |
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | |
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 | |
|  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | |
|  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 | |
|  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 | |
|  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 | |
|  * SUCH DAMAGE.
 | |
|  */
 | |
| 
 | |
| #include "rcp_locl.h"
 | |
| #include <getarg.h>
 | |
| 
 | |
| #define RSH_PROGRAM "rsh"
 | |
| 
 | |
| struct  passwd *pwd;
 | |
| uid_t	userid;
 | |
| int     errs, remin, remout;
 | |
| int     pflag, iamremote, iamrecursive, targetshouldbedirectory;
 | |
| int     doencrypt, noencrypt;
 | |
| int     usebroken, usekrb4, usekrb5, forwardtkt;
 | |
| char    *port;
 | |
| int     eflag = 0;
 | |
| 
 | |
| #define	CMDNEEDS	64
 | |
| char cmd[CMDNEEDS];		/* must hold "rcp -r -p -d\0" */
 | |
| 
 | |
| int	 response (void);
 | |
| void	 rsource (char *, struct stat *);
 | |
| void	 sink (int, char *[]);
 | |
| void	 source (int, char *[]);
 | |
| void	 tolocal (int, char *[]);
 | |
| void	 toremote (char *, int, char *[]);
 | |
| 
 | |
| int      do_cmd(char *host, char *remuser, char *cmd, int *fdin, int *fdout);
 | |
| 
 | |
| static int fflag, tflag;
 | |
| 
 | |
| static int version_flag, help_flag;
 | |
| 
 | |
| struct getargs args[] = {
 | |
|     { NULL,	'4', arg_flag,	 &usekrb4,	"use Kerberos 4 authentication",      NULL },
 | |
|     { NULL,	'5', arg_flag,	 &usekrb5,	"use Kerberos 5 authentication",      NULL },
 | |
|     { NULL,	'F', arg_flag,	 &forwardtkt,	"forward credentials", NULL },
 | |
|     { NULL,	'K', arg_flag,	 &usebroken,	"use BSD authentication",
 | |
|       NULL },
 | |
|     { NULL,	'P', arg_string, &port,		"non-default port", "port" },
 | |
|     { NULL,	'p', arg_flag,	 &pflag,	"preserve file permissions",
 | |
|       NULL },
 | |
|     { NULL,	'r', arg_flag,	 &iamrecursive,	"recursive mode", NULL },
 | |
|     { NULL,	'x', arg_flag,	 &doencrypt,	"use encryption", NULL },
 | |
|     { NULL,	'z', arg_flag,	 &noencrypt,	"don't encrypt", NULL },
 | |
|     { NULL,	'd', arg_flag,	 &targetshouldbedirectory, NULL, NULL },
 | |
|     { NULL,	'e', arg_flag,	 &eflag,	"passed to rsh", NULL },
 | |
|     { NULL,	'f', arg_flag,	 &fflag,	NULL, NULL },
 | |
|     { NULL,	't', arg_flag,	 &tflag,	NULL, NULL },
 | |
|     { "version", 0,  arg_flag,	 &version_flag,	NULL, NULL },
 | |
|     { "help",	 0,  arg_flag,	 &help_flag,	NULL, NULL }
 | |
| };
 | |
| 
 | |
| static void
 | |
| usage (int ret)
 | |
| {
 | |
|     arg_printusage (args,
 | |
| 		    sizeof(args) / sizeof(args[0]),
 | |
| 		    NULL,
 | |
| 		    "file1 file2|file... directory");
 | |
|     exit (ret);
 | |
| }
 | |
| 
 | |
| int
 | |
| main(int argc, char **argv)
 | |
| {
 | |
| 	char *targ;
 | |
| 	int optind = 0;
 | |
| 
 | |
| 	setprogname(argv[0]);
 | |
| 	if (getarg (args, sizeof(args) / sizeof(args[0]), argc, argv,
 | |
| 		    &optind))
 | |
| 	    usage (1);
 | |
| 	if(help_flag)
 | |
| 	    usage(0);
 | |
| 	if (version_flag) {
 | |
| 	    print_version (NULL);
 | |
| 	    return 0;
 | |
| 	}
 | |
| 
 | |
| 	iamremote = (fflag || tflag);
 | |
| 
 | |
| 	argc -= optind;
 | |
| 	argv += optind;
 | |
| 
 | |
| 	if ((pwd = getpwuid(userid = getuid())) == NULL)
 | |
| 		errx(1, "unknown user %d", (int)userid);
 | |
| 
 | |
| 	remin = STDIN_FILENO;		/* XXX */
 | |
| 	remout = STDOUT_FILENO;
 | |
| 
 | |
| 	if (fflag) {			/* Follow "protocol", send data. */
 | |
| 		(void)response();
 | |
| 		source(argc, argv);
 | |
| 		exit(errs);
 | |
| 	}
 | |
| 
 | |
| 	if (tflag) {			/* Receive data. */
 | |
| 		sink(argc, argv);
 | |
| 		exit(errs);
 | |
| 	}
 | |
| 
 | |
| 	if (argc < 2)
 | |
| 	    usage(1);
 | |
| 	if (argc > 2)
 | |
| 		targetshouldbedirectory = 1;
 | |
| 
 | |
| 	remin = remout = -1;
 | |
| 	/* Command to be executed on remote system using "rsh". */
 | |
| 	snprintf(cmd, sizeof(cmd),
 | |
| 		 "rcp%s%s%s", iamrecursive ? " -r" : "",
 | |
| 		 pflag ? " -p" : "", targetshouldbedirectory ? " -d" : "");
 | |
| 
 | |
| 	signal(SIGPIPE, lostconn);
 | |
| 
 | |
| 	if ((targ = colon(argv[argc - 1])))	/* Dest is remote host. */
 | |
| 		toremote(targ, argc, argv);
 | |
| 	else {
 | |
| 		tolocal(argc, argv);		/* Dest is local host. */
 | |
| 		if (targetshouldbedirectory)
 | |
| 			verifydir(argv[argc - 1]);
 | |
| 	}
 | |
| 	exit(errs);
 | |
| }
 | |
| 
 | |
| void
 | |
| toremote(char *targ, int argc, char **argv)
 | |
| {
 | |
| 	int i;
 | |
| 	char *bp, *host, *src, *suser, *thost, *tuser;
 | |
| 
 | |
| 	*targ++ = 0;
 | |
| 	if (*targ == 0)
 | |
| 		targ = ".";
 | |
| 
 | |
| 	if ((thost = strchr(argv[argc - 1], '@')) != NULL) {
 | |
| 		/* user@host */
 | |
| 		*thost++ = 0;
 | |
| 		tuser = argv[argc - 1];
 | |
| 		if (*tuser == '\0')
 | |
| 			tuser = NULL;
 | |
| 		else if (!okname(tuser))
 | |
| 			exit(1);
 | |
| 	} else {
 | |
| 		thost = argv[argc - 1];
 | |
| 		tuser = NULL;
 | |
| 	}
 | |
| 	thost = unbracket(thost);
 | |
| 
 | |
| 	for (i = 0; i < argc - 1; i++) {
 | |
| 		src = colon(argv[i]);
 | |
| 		if (src) {			/* remote to remote */
 | |
| 			int ret;
 | |
| 			*src++ = 0;
 | |
| 			if (*src == 0)
 | |
| 				src = ".";
 | |
| 			host = strchr(argv[i], '@');
 | |
| 			if (host) {
 | |
| 				*host++ = '\0';
 | |
| 				host = unbracket(host);
 | |
| 				suser = argv[i];
 | |
| 				if (*suser == '\0')
 | |
| 					suser = pwd->pw_name;
 | |
| 				else if (!okname(suser))
 | |
| 					continue;
 | |
| 				ret = asprintf(&bp,
 | |
| 				    "%s%s %s -l %s -n %s %s '%s%s%s:%s'",
 | |
| 					 _PATH_RSH, eflag ? " -e" : "",
 | |
| 					 host, suser, cmd, src,
 | |
| 				    tuser ? tuser : "", tuser ? "@" : "",
 | |
| 				    thost, targ);
 | |
| 			} else {
 | |
| 				host = unbracket(argv[i]);
 | |
| 				ret = asprintf(&bp,
 | |
| 					 "exec %s%s %s -n %s %s '%s%s%s:%s'",
 | |
| 					 _PATH_RSH, eflag ? " -e" : "",
 | |
| 					 host, cmd, src,
 | |
| 					 tuser ? tuser : "", tuser ? "@" : "",
 | |
| 					 thost, targ);
 | |
| 			}
 | |
| 			if (ret == -1)
 | |
| 				err (1, "malloc");
 | |
| 			susystem(bp);
 | |
| 			free(bp);
 | |
| 		} else {			/* local to remote */
 | |
| 			if (remin == -1) {
 | |
| 				if (asprintf(&bp, "%s -t %s", cmd, targ) == -1)
 | |
| 					err (1, "malloc");
 | |
| 				host = thost;
 | |
| 
 | |
| 				if (do_cmd(host, tuser, bp, &remin, &remout) < 0)
 | |
| 					exit(1);
 | |
| 
 | |
| 				if (response() < 0)
 | |
| 					exit(1);
 | |
| 				free(bp);
 | |
| 			}
 | |
| 			source(1, argv+i);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| tolocal(int argc, char **argv)
 | |
| {
 | |
| 	int i;
 | |
| 	char *bp, *host, *src, *suser;
 | |
| 
 | |
| 	for (i = 0; i < argc - 1; i++) {
 | |
| 		int ret;
 | |
| 
 | |
| 		if (!(src = colon(argv[i]))) {		/* Local to local. */
 | |
| 			ret = asprintf(&bp, "exec %s%s%s %s %s", _PATH_CP,
 | |
| 			    iamrecursive ? " -PR" : "", pflag ? " -p" : "",
 | |
| 			    argv[i], argv[argc - 1]);
 | |
| 			if (ret == -1)
 | |
| 				err (1, "malloc");
 | |
| 			if (susystem(bp))
 | |
| 				++errs;
 | |
| 			free(bp);
 | |
| 			continue;
 | |
| 		}
 | |
| 		*src++ = 0;
 | |
| 		if (*src == 0)
 | |
| 			src = ".";
 | |
| 		if ((host = strchr(argv[i], '@')) == NULL) {
 | |
| 			host = argv[i];
 | |
| 			suser = pwd->pw_name;
 | |
| 		} else {
 | |
| 			*host++ = 0;
 | |
| 			suser = argv[i];
 | |
| 			if (*suser == '\0')
 | |
| 				suser = pwd->pw_name;
 | |
| 			else if (!okname(suser))
 | |
| 				continue;
 | |
| 		}
 | |
| 		ret = asprintf(&bp, "%s -f %s", cmd, src);
 | |
| 		if (ret == -1)
 | |
| 			err (1, "malloc");
 | |
| 		if (do_cmd(host, suser, bp, &remin, &remout) < 0) {
 | |
| 			free(bp);
 | |
| 			++errs;
 | |
| 			continue;
 | |
| 		}
 | |
| 		free(bp);
 | |
| 		sink(1, argv + argc - 1);
 | |
| 		close(remin);
 | |
| 		remin = remout = -1;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| source(int argc, char **argv)
 | |
| {
 | |
| 	struct stat stb;
 | |
| 	static BUF buffer;
 | |
| 	BUF *bp;
 | |
| 	off_t i;
 | |
| 	off_t amt;
 | |
| 	int fd, haderr, indx, result;
 | |
| 	char *last, *name, buf[BUFSIZ];
 | |
| 
 | |
| 	for (indx = 0; indx < argc; ++indx) {
 | |
|                 name = argv[indx];
 | |
| 		if ((fd = open(name, O_RDONLY, 0)) < 0)
 | |
| 			goto syserr;
 | |
| 		if (fstat(fd, &stb)) {
 | |
| syserr:			run_err("%s: %s", name, strerror(errno));
 | |
| 			goto next;
 | |
| 		}
 | |
| 		if (S_ISDIR(stb.st_mode) && iamrecursive) {
 | |
| 			rsource(name, &stb);
 | |
| 			goto next;
 | |
| 		} else if (!S_ISREG(stb.st_mode)) {
 | |
| 			run_err("%s: not a regular file", name);
 | |
| 			goto next;
 | |
| 		}
 | |
| 		if ((last = strrchr(name, '/')) == NULL)
 | |
| 			last = name;
 | |
| 		else
 | |
| 			++last;
 | |
| 		if (pflag) {
 | |
| 			/*
 | |
| 			 * Make it compatible with possible future
 | |
| 			 * versions expecting microseconds.
 | |
| 			 */
 | |
| 			snprintf(buf, sizeof(buf), "T%ld 0 %ld 0\n",
 | |
| 			    (long)stb.st_mtime,
 | |
| 			    (long)stb.st_atime);
 | |
| 			write(remout, buf, strlen(buf));
 | |
| 			if (response() < 0)
 | |
| 				goto next;
 | |
| 		}
 | |
| #undef MODEMASK
 | |
| #define	MODEMASK	(S_ISUID|S_ISGID|S_ISVTX|S_IRWXU|S_IRWXG|S_IRWXO)
 | |
| 		snprintf(buf, sizeof(buf), "C%04o %lu %s\n",
 | |
| 			 (unsigned int)(stb.st_mode & MODEMASK),
 | |
| 			 (unsigned long)stb.st_size,
 | |
| 			 last);
 | |
| 		write(remout, buf, strlen(buf));
 | |
| 		if (response() < 0)
 | |
| 			goto next;
 | |
| 		if ((bp = allocbuf(&buffer, fd, BUFSIZ)) == NULL) {
 | |
| next:			close(fd);
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		/* Keep writing after an error so that we stay sync'd up. */
 | |
| 		for (haderr = i = 0; i < stb.st_size; i += bp->cnt) {
 | |
| 			amt = bp->cnt;
 | |
| 			if (i + amt > stb.st_size)
 | |
| 				amt = stb.st_size - i;
 | |
| 			if (!haderr) {
 | |
| 			        result = read(fd, bp->buf, (size_t)amt);
 | |
| 				if (result != amt)
 | |
| 					haderr = result >= 0 ? EIO : errno;
 | |
| 			}
 | |
| 			if (haderr)
 | |
| 				write(remout, bp->buf, amt);
 | |
| 			else {
 | |
| 			        result = write(remout, bp->buf, (size_t)amt);
 | |
| 				if (result != amt)
 | |
| 					haderr = result >= 0 ? EIO : errno;
 | |
| 			}
 | |
| 		}
 | |
| 		if (close(fd) && !haderr)
 | |
| 			haderr = errno;
 | |
| 		if (!haderr)
 | |
| 			write(remout, "", 1);
 | |
| 		else
 | |
| 			run_err("%s: %s", name, strerror(haderr));
 | |
| 		response();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void
 | |
| rsource(char *name, struct stat *statp)
 | |
| {
 | |
| 	DIR *dirp;
 | |
| 	struct dirent *dp;
 | |
| 	char *last, *vect[1], path[MAXPATHLEN];
 | |
| 
 | |
| 	if (!(dirp = opendir(name))) {
 | |
| 		run_err("%s: %s", name, strerror(errno));
 | |
| 		return;
 | |
| 	}
 | |
| 	last = strrchr(name, '/');
 | |
| 	if (last == 0)
 | |
| 		last = name;
 | |
| 	else
 | |
| 		last++;
 | |
| 	if (pflag) {
 | |
| 		snprintf(path, sizeof(path), "T%ld 0 %ld 0\n",
 | |
| 		    (long)statp->st_mtime,
 | |
| 		    (long)statp->st_atime);
 | |
| 		write(remout, path, strlen(path));
 | |
| 		if (response() < 0) {
 | |
| 			closedir(dirp);
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 	snprintf(path, sizeof(path),
 | |
| 		 "D%04o %d %s\n",
 | |
| 		 (unsigned int)(statp->st_mode & MODEMASK), 0, last);
 | |
| 	write(remout, path, strlen(path));
 | |
| 	if (response() < 0) {
 | |
| 		closedir(dirp);
 | |
| 		return;
 | |
| 	}
 | |
| 	while ((dp = readdir(dirp)) != NULL) {
 | |
| 		if (dp->d_ino == 0)
 | |
| 			continue;
 | |
| 		if (!strcmp(dp->d_name, ".") || !strcmp(dp->d_name, ".."))
 | |
| 			continue;
 | |
| 		if (strlen(name) + 1 + strlen(dp->d_name) >= MAXPATHLEN - 1) {
 | |
| 			run_err("%s/%s: name too long", name, dp->d_name);
 | |
| 			continue;
 | |
| 		}
 | |
| 		snprintf(path, sizeof(path), "%s/%s", name, dp->d_name);
 | |
| 		vect[0] = path;
 | |
| 		source(1, vect);
 | |
| 	}
 | |
| 	closedir(dirp);
 | |
| 	write(remout, "E\n", 2);
 | |
| 	response();
 | |
| }
 | |
| 
 | |
| void
 | |
| sink(int argc, char **argv)
 | |
| {
 | |
| 	static BUF buffer;
 | |
| 	struct stat stb;
 | |
| 	struct timeval tv[2];
 | |
| 	enum { YES, NO, DISPLAYED } wrerr;
 | |
| 	BUF *bp;
 | |
| 	off_t i, j, size;
 | |
| 	int amt, count, exists, first, mask, mode, ofd, omode;
 | |
| 	int setimes, targisdir, wrerrno = 0;
 | |
| 	char ch, *cp, *np, *targ, *why, *vect[1], buf[BUFSIZ];
 | |
| 
 | |
| #define	atime	tv[0]
 | |
| #define	mtime	tv[1]
 | |
| #define	SCREWUP(str)	{ why = str; goto screwup; }
 | |
| 
 | |
| 	setimes = targisdir = 0;
 | |
| 	mask = umask(0);
 | |
| 	if (!pflag)
 | |
| 		umask(mask);
 | |
| 	if (argc != 1) {
 | |
| 		run_err("ambiguous target");
 | |
| 		exit(1);
 | |
| 	}
 | |
| 	targ = *argv;
 | |
| 	if (targetshouldbedirectory)
 | |
| 		verifydir(targ);
 | |
| 	write(remout, "", 1);
 | |
| 	if (stat(targ, &stb) == 0 && S_ISDIR(stb.st_mode))
 | |
| 		targisdir = 1;
 | |
| 	for (first = 1;; first = 0) {
 | |
| 		cp = buf;
 | |
| 		if (read(remin, cp, 1) <= 0)
 | |
| 			return;
 | |
| 		if (*cp++ == '\n')
 | |
| 			SCREWUP("unexpected <newline>");
 | |
| 		do {
 | |
| 			if (read(remin, &ch, sizeof(ch)) != sizeof(ch))
 | |
| 				SCREWUP("lost connection");
 | |
| 			*cp++ = ch;
 | |
| 		} while (cp < &buf[BUFSIZ - 1] && ch != '\n');
 | |
| 		*cp = 0;
 | |
| 
 | |
| 		if (buf[0] == '\01' || buf[0] == '\02') {
 | |
| 			if (iamremote == 0)
 | |
| 				write(STDERR_FILENO,
 | |
| 				    buf + 1, strlen(buf + 1));
 | |
| 			if (buf[0] == '\02')
 | |
| 				exit(1);
 | |
| 			++errs;
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (buf[0] == 'E') {
 | |
| 			write(remout, "", 1);
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (ch == '\n')
 | |
| 			*--cp = 0;
 | |
| 
 | |
| 		cp = buf;
 | |
| 		if (*cp == 'T') {
 | |
| 			setimes++;
 | |
| 			cp++;
 | |
| 			mtime.tv_sec = strtol(cp, &cp, 10);
 | |
| 			if (!cp || *cp++ != ' ')
 | |
| 				SCREWUP("mtime.sec not delimited");
 | |
| 			mtime.tv_usec = strtol(cp, &cp, 10);
 | |
| 			if (!cp || *cp++ != ' ')
 | |
| 				SCREWUP("mtime.usec not delimited");
 | |
| 			atime.tv_sec = strtol(cp, &cp, 10);
 | |
| 			if (!cp || *cp++ != ' ')
 | |
| 				SCREWUP("atime.sec not delimited");
 | |
| 			atime.tv_usec = strtol(cp, &cp, 10);
 | |
| 			if (!cp || *cp++ != '\0')
 | |
| 				SCREWUP("atime.usec not delimited");
 | |
| 			write(remout, "", 1);
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (*cp != 'C' && *cp != 'D') {
 | |
| 			/*
 | |
| 			 * Check for the case "rcp remote:foo\* local:bar".
 | |
| 			 * In this case, the line "No match." can be returned
 | |
| 			 * by the shell before the rcp command on the remote is
 | |
| 			 * executed so the ^Aerror_message convention isn't
 | |
| 			 * followed.
 | |
| 			 */
 | |
| 			if (first) {
 | |
| 				run_err("%s", cp);
 | |
| 				exit(1);
 | |
| 			}
 | |
| 			SCREWUP("expected control record");
 | |
| 		}
 | |
| 		mode = 0;
 | |
| 		for (++cp; cp < buf + 5; cp++) {
 | |
| 			if (*cp < '0' || *cp > '7')
 | |
| 				SCREWUP("bad mode");
 | |
| 			mode = (mode << 3) | (*cp - '0');
 | |
| 		}
 | |
| 		if (*cp++ != ' ')
 | |
| 			SCREWUP("mode not delimited");
 | |
| 
 | |
| 		for (size = 0; isdigit((unsigned char)*cp);)
 | |
| 			size = size * 10 + (*cp++ - '0');
 | |
| 		if (*cp++ != ' ')
 | |
| 			SCREWUP("size not delimited");
 | |
| 		if (targisdir) {
 | |
| 			static char *namebuf;
 | |
| 			static int cursize;
 | |
| 			size_t need;
 | |
| 
 | |
| 			need = strlen(targ) + strlen(cp) + 250;
 | |
| 			if (need > cursize) {
 | |
| 				if (!(namebuf = malloc(need)))
 | |
| 					run_err("%s", strerror(errno));
 | |
| 			}
 | |
| 			snprintf(namebuf, need, "%s%s%s", targ,
 | |
| 			    *targ ? "/" : "", cp);
 | |
| 			np = namebuf;
 | |
| 		} else
 | |
| 			np = targ;
 | |
| 		exists = stat(np, &stb) == 0;
 | |
| 		if (buf[0] == 'D') {
 | |
| 			int mod_flag = pflag;
 | |
| 			if (exists) {
 | |
| 				if (!S_ISDIR(stb.st_mode)) {
 | |
| 					errno = ENOTDIR;
 | |
| 					goto bad;
 | |
| 				}
 | |
| 				if (pflag)
 | |
| 					chmod(np, mode);
 | |
| 			} else {
 | |
| 				/* Handle copying from a read-only directory */
 | |
| 				mod_flag = 1;
 | |
| 				if (mkdir(np, mode | S_IRWXU) < 0)
 | |
| 					goto bad;
 | |
| 			}
 | |
| 			vect[0] = np;
 | |
| 			sink(1, vect);
 | |
| 			if (setimes) {
 | |
| 				setimes = 0;
 | |
| 				if (utimes(np, tv) < 0)
 | |
| 				    run_err("%s: set times: %s",
 | |
| 					np, strerror(errno));
 | |
| 			}
 | |
| 			if (mod_flag)
 | |
| 				chmod(np, mode);
 | |
| 			continue;
 | |
| 		}
 | |
| 		omode = mode;
 | |
| 		mode |= S_IWRITE;
 | |
| 		if ((ofd = open(np, O_WRONLY|O_CREAT, mode)) < 0) {
 | |
| bad:			run_err("%s: %s", np, strerror(errno));
 | |
| 			continue;
 | |
| 		}
 | |
| 		write(remout, "", 1);
 | |
| 		if ((bp = allocbuf(&buffer, ofd, BUFSIZ)) == NULL) {
 | |
| 			close(ofd);
 | |
| 			continue;
 | |
| 		}
 | |
| 		cp = bp->buf;
 | |
| 		wrerr = NO;
 | |
| 		for (count = i = 0; i < size; i += BUFSIZ) {
 | |
| 			amt = BUFSIZ;
 | |
| 			if (i + amt > size)
 | |
| 				amt = size - i;
 | |
| 			count += amt;
 | |
| 			if((j = net_read(remin, cp, amt)) != amt) {
 | |
| 			    run_err("%s", j ? strerror(errno) :
 | |
| 				    "dropped connection");
 | |
| 			    exit(1);
 | |
| 			}
 | |
| 			amt -= j;
 | |
| 			cp += j;
 | |
| 			if (count == bp->cnt) {
 | |
| 				/* Keep reading so we stay sync'd up. */
 | |
| 				if (wrerr == NO) {
 | |
| 					j = write(ofd, bp->buf, (size_t)count);
 | |
| 					if (j != count) {
 | |
| 						wrerr = YES;
 | |
| 						wrerrno = j >= 0 ? EIO : errno;
 | |
| 					}
 | |
| 				}
 | |
| 				count = 0;
 | |
| 				cp = bp->buf;
 | |
| 			}
 | |
| 		}
 | |
| 		if (count != 0 && wrerr == NO &&
 | |
| 		    (j = write(ofd, bp->buf, (size_t)count)) != count) {
 | |
| 			wrerr = YES;
 | |
| 			wrerrno = j >= 0 ? EIO : errno;
 | |
| 		}
 | |
| 		if (ftruncate(ofd, size)) {
 | |
| 			run_err("%s: truncate: %s", np, strerror(errno));
 | |
| 			wrerr = DISPLAYED;
 | |
| 		}
 | |
| 		if (pflag) {
 | |
| 			if (exists || omode != mode)
 | |
| 				if (fchmod(ofd, omode))
 | |
| 					run_err("%s: set mode: %s",
 | |
| 					    np, strerror(errno));
 | |
| 		} else {
 | |
| 			if (!exists && omode != mode)
 | |
| 				if (fchmod(ofd, omode & ~mask))
 | |
| 					run_err("%s: set mode: %s",
 | |
| 					    np, strerror(errno));
 | |
| 		}
 | |
| 		close(ofd);
 | |
| 		response();
 | |
| 		if (setimes && wrerr == NO) {
 | |
| 			setimes = 0;
 | |
| 			if (utimes(np, tv) < 0) {
 | |
| 				run_err("%s: set times: %s",
 | |
| 				    np, strerror(errno));
 | |
| 				wrerr = DISPLAYED;
 | |
| 			}
 | |
| 		}
 | |
| 		switch(wrerr) {
 | |
| 		case YES:
 | |
| 			run_err("%s: %s", np, strerror(wrerrno));
 | |
| 			break;
 | |
| 		case NO:
 | |
| 			write(remout, "", 1);
 | |
| 			break;
 | |
| 		case DISPLAYED:
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| screwup:
 | |
| 	run_err("protocol error: %s", why);
 | |
| 	exit(1);
 | |
| }
 | |
| 
 | |
| int
 | |
| response(void)
 | |
| {
 | |
| 	char ch, *cp, resp, rbuf[BUFSIZ];
 | |
| 
 | |
| 	if (read(remin, &resp, sizeof(resp)) != sizeof(resp))
 | |
| 		lostconn(0);
 | |
| 
 | |
| 	cp = rbuf;
 | |
| 	switch(resp) {
 | |
| 	case 0:				/* ok */
 | |
| 		return (0);
 | |
| 	default:
 | |
| 		*cp++ = resp;
 | |
| 		/* FALLTHROUGH */
 | |
| 	case 1:				/* error, followed by error msg */
 | |
| 	case 2:				/* fatal error, "" */
 | |
| 		do {
 | |
| 			if (read(remin, &ch, sizeof(ch)) != sizeof(ch))
 | |
| 				lostconn(0);
 | |
| 			*cp++ = ch;
 | |
| 		} while (cp < &rbuf[BUFSIZ] && ch != '\n');
 | |
| 
 | |
| 		if (!iamremote)
 | |
| 			write(STDERR_FILENO, rbuf, cp - rbuf);
 | |
| 		++errs;
 | |
| 		if (resp == 1)
 | |
| 			return (-1);
 | |
| 		exit(1);
 | |
| 	}
 | |
| 	/* NOTREACHED */
 | |
| }
 | |
| 
 | |
| #include <stdarg.h>
 | |
| 
 | |
| void
 | |
| run_err(const char *fmt, ...)
 | |
| {
 | |
| 	static FILE *fp;
 | |
| 	va_list ap;
 | |
| 
 | |
| 	++errs;
 | |
| 	if (fp == NULL && !(fp = fdopen(remout, "w")))
 | |
| 		return;
 | |
| 	va_start(ap, fmt);
 | |
| 	fprintf(fp, "%c", 0x01);
 | |
| 	fprintf(fp, "rcp: ");
 | |
| 	vfprintf(fp, fmt, ap);
 | |
| 	fprintf(fp, "\n");
 | |
| 	fflush(fp);
 | |
| 	va_end(ap);
 | |
| 
 | |
| 	if (!iamremote) {
 | |
| 	    va_start(ap, fmt);
 | |
| 	    vwarnx(fmt, ap);
 | |
| 	    va_end(ap);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function executes the given command as the specified user on the
 | |
|  * given host.  This returns < 0 if execution fails, and >= 0 otherwise. This
 | |
|  * assigns the input and output file descriptors on success.
 | |
|  *
 | |
|  * If it cannot create necessary pipes it exits with error message.
 | |
|  */
 | |
| 
 | |
| int
 | |
| do_cmd(char *host, char *remuser, char *cmd, int *fdin, int *fdout)
 | |
| {
 | |
| 	int pin[2], pout[2], reserved[2];
 | |
| 
 | |
| 	/*
 | |
| 	 * Reserve two descriptors so that the real pipes won't get
 | |
| 	 * descriptors 0 and 1 because that will screw up dup2 below.
 | |
| 	 */
 | |
| 	pipe(reserved);
 | |
| 
 | |
| 	/* Create a socket pair for communicating with rsh. */
 | |
| 	if (pipe(pin) < 0) {
 | |
| 		perror("pipe");
 | |
| 		exit(255);
 | |
| 	}
 | |
| 	if (pipe(pout) < 0) {
 | |
| 		perror("pipe");
 | |
| 		exit(255);
 | |
| 	}
 | |
| 
 | |
| 	/* Free the reserved descriptors. */
 | |
| 	close(reserved[0]);
 | |
| 	close(reserved[1]);
 | |
| 
 | |
| 	/* For a child to execute the command on the remote host using rsh. */
 | |
| 	if (fork() == 0) {
 | |
| 		char *args[100];
 | |
| 		unsigned int i;
 | |
| 
 | |
| 		/* Child. */
 | |
| 		close(pin[1]);
 | |
| 		close(pout[0]);
 | |
| 		dup2(pin[0], 0);
 | |
| 		dup2(pout[1], 1);
 | |
| 		close(pin[0]);
 | |
| 		close(pout[1]);
 | |
| 
 | |
| 		i = 0;
 | |
| 		args[i++] = RSH_PROGRAM;
 | |
| 		if (usekrb4)
 | |
| 			args[i++] = "-4";
 | |
| 		if (usekrb5)
 | |
| 			args[i++] = "-5";
 | |
| 		if (usebroken)
 | |
| 			args[i++] = "-K";
 | |
| 		if (doencrypt)
 | |
| 			args[i++] = "-x";
 | |
| 		if (forwardtkt)
 | |
| 			args[i++] = "-F";
 | |
| 		if (noencrypt)
 | |
| 			args[i++] = "-z";
 | |
| 		if (port != NULL) {
 | |
| 			args[i++] = "-p";
 | |
| 			args[i++] = port;
 | |
| 		}
 | |
| 		if (eflag)
 | |
| 		    args[i++] = "-e";
 | |
| 		if (remuser != NULL) {
 | |
| 			args[i++] = "-l";
 | |
| 			args[i++] = remuser;
 | |
| 		}
 | |
| 		args[i++] = host;
 | |
| 		args[i++] = cmd;
 | |
| 		args[i++] = NULL;
 | |
| 
 | |
| 		execvp(RSH_PROGRAM, args);
 | |
| 		perror(RSH_PROGRAM);
 | |
| 		exit(1);
 | |
| 	}
 | |
| 	/* Parent.  Close the other side, and return the local side. */
 | |
| 	close(pin[0]);
 | |
| 	*fdout = pin[1];
 | |
| 	close(pout[1]);
 | |
| 	*fdin = pout[0];
 | |
| 	return 0;
 | |
| }
 |