285 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			285 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 1983 Regents of the University of California.
 | 
						|
 * All rights reserved.
 | 
						|
 *
 | 
						|
 * Redistribution and use in source and binary forms are permitted
 | 
						|
 * provided that: (1) source distributions retain this entire copyright
 | 
						|
 * notice and comment, and (2) distributions including binaries display
 | 
						|
 * the following acknowledgement:  ``This product includes software
 | 
						|
 * developed by the University of California, Berkeley and its contributors''
 | 
						|
 * in the documentation or other materials provided with the distribution
 | 
						|
 * and in all advertising materials mentioning features or use of this
 | 
						|
 * software. 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 ``AS IS'' AND WITHOUT ANY EXPRESS OR
 | 
						|
 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
 | 
						|
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
 | 
						|
 */
 | 
						|
 | 
						|
#if defined(LIBC_SCCS) && !defined(lint)
 | 
						|
static char sccsid[] = "@(#)rcmd.c	5.22 (Berkeley) 6/1/90";
 | 
						|
#endif /* LIBC_SCCS and not lint */
 | 
						|
 | 
						|
#include <sys/types.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <sys/ioctl.h>
 | 
						|
#include <ctype.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <limits.h>
 | 
						|
#include <pwd.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <signal.h>
 | 
						|
#include <string.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <net/gen/netdb.h>
 | 
						|
#include <net/gen/in.h>
 | 
						|
#include <net/gen/tcp.h>
 | 
						|
#include <net/gen/tcp_io.h>
 | 
						|
#include <net/hton.h>
 | 
						|
#include <net/netlib.h>
 | 
						|
 | 
						|
#define MAXHOSTNAMELEN	256
 | 
						|
#define MAXPATHLEN PATH_MAX
 | 
						|
 | 
						|
#ifdef __STDC__
 | 
						|
#define CONST	const
 | 
						|
#else
 | 
						|
#define CONST
 | 
						|
#endif
 | 
						|
 | 
						|
extern	errno;
 | 
						|
 | 
						|
int rcmd(ahost, rport, locuser, remuser, cmd, fd2p)
 | 
						|
char **ahost;
 | 
						|
int rport;
 | 
						|
CONST char *locuser, *remuser, *cmd;
 | 
						|
int *fd2p;
 | 
						|
{
 | 
						|
	int fd, fd2, result;
 | 
						|
	struct hostent *hp;
 | 
						|
	int n;
 | 
						|
	static tcpport_t lport;
 | 
						|
	nwio_tcpconf_t tcpconf;
 | 
						|
	nwio_tcpcl_t tcpconnopt;
 | 
						|
	pid_t pid;
 | 
						|
	char num[8];
 | 
						|
	char c;
 | 
						|
	char *tcp_device;
 | 
						|
 | 
						|
	fd= -1;
 | 
						|
	fd2= -1;
 | 
						|
 | 
						|
	if (lport == 0) {
 | 
						|
		pid = getpid();
 | 
						|
		lport = 1;
 | 
						|
		do {
 | 
						|
			lport = (lport << 1) | (pid & 1);
 | 
						|
 | 
						|
			pid >>= 1;
 | 
						|
		} while (lport < TCPPORT_RESERVED/2);
 | 
						|
	}
 | 
						|
 | 
						|
	tcp_device= getenv("TCP_DEVICE");
 | 
						|
	if (tcp_device == NULL)
 | 
						|
		tcp_device= TCP_DEVICE;
 | 
						|
	hp= gethostbyname(*ahost);
 | 
						|
	if (!hp)
 | 
						|
	{
 | 
						|
		fprintf(stderr, "%s: unknown host\n", *ahost);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
	*ahost= hp->h_name;
 | 
						|
	n = TCPPORT_RESERVED/2;
 | 
						|
	do
 | 
						|
	{
 | 
						|
		if (--lport < TCPPORT_RESERVED/2)
 | 
						|
			lport = TCPPORT_RESERVED-1;
 | 
						|
		fd= open (tcp_device, O_RDWR);
 | 
						|
		if (fd<0)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to open %s: %s\n",
 | 
						|
				tcp_device, strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		tcpconf.nwtc_flags= NWTC_LP_SET | NWTC_SET_RA | NWTC_SET_RP |
 | 
						|
			NWTC_EXCL;
 | 
						|
		tcpconf.nwtc_locport= htons(lport);
 | 
						|
		tcpconf.nwtc_remport= rport;
 | 
						|
		tcpconf.nwtc_remaddr= *(ipaddr_t *)hp->h_addr;
 | 
						|
 | 
						|
		result= ioctl(fd, NWIOSTCPCONF, &tcpconf);
 | 
						|
		if (result<0)
 | 
						|
		{
 | 
						|
			if (errno == EADDRINUSE)
 | 
						|
			{
 | 
						|
				close(fd);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			fprintf(stderr, "unable to ioctl(NWIOSTCPCONF): %s\n",
 | 
						|
				strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		tcpconf.nwtc_flags= NWTC_SHARED;
 | 
						|
		result= ioctl(fd, NWIOSTCPCONF, &tcpconf);
 | 
						|
		if (result<0)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to ioctl(NWIOSTCPCONF): %s\n",
 | 
						|
				strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		tcpconnopt.nwtcl_flags= 0;
 | 
						|
 | 
						|
		do
 | 
						|
		{
 | 
						|
			result= ioctl (fd, NWIOTCPCONN, &tcpconnopt);
 | 
						|
			if (result<0 && errno == EAGAIN)
 | 
						|
			{
 | 
						|
				sleep(2);
 | 
						|
			}
 | 
						|
		} while (result<0 && errno == EAGAIN);
 | 
						|
		if (result<0 && errno != EADDRINUSE)
 | 
						|
		{
 | 
						|
			fprintf(stderr,
 | 
						|
				"unable to ioctl(NWIOTCPCONN): %s\n",
 | 
						|
				strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		if (result>=0)
 | 
						|
			break;
 | 
						|
	} while (--n > 0);
 | 
						|
	if (n == 0)
 | 
						|
	{
 | 
						|
		fprintf(stderr, "can't get port\n");
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
	if (!fd2p)
 | 
						|
	{
 | 
						|
		if (write(fd, "", 1) != 1)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to write: %s", strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		fd2= open (tcp_device, O_RDWR);
 | 
						|
		if (fd2<0)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to open %s: %s\n",
 | 
						|
				tcp_device, strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		tcpconf.nwtc_flags= NWTC_LP_SET | NWTC_UNSET_RA | 
 | 
						|
			NWTC_UNSET_RP | NWTC_SHARED;
 | 
						|
		tcpconf.nwtc_locport= htons(lport);
 | 
						|
 | 
						|
		result= ioctl(fd2, NWIOSTCPCONF, &tcpconf);
 | 
						|
		if (result<0)
 | 
						|
		{
 | 
						|
			fprintf(stderr,
 | 
						|
				"unable to ioctl(NWIOSTCPCONF): %s\n",
 | 
						|
				strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		pid= fork();
 | 
						|
		if (pid<0)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to fork: %s\n",
 | 
						|
				strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		if (!pid)
 | 
						|
		{
 | 
						|
			alarm(0);
 | 
						|
			signal(SIGALRM, SIG_DFL);
 | 
						|
			alarm(30); /* give up after half a minute */
 | 
						|
			tcpconnopt.nwtcl_flags= 0;
 | 
						|
 | 
						|
			do
 | 
						|
			{
 | 
						|
				result= ioctl (fd2, NWIOTCPLISTEN,
 | 
						|
					&tcpconnopt);
 | 
						|
				if (result<0 && errno == EAGAIN)
 | 
						|
				{
 | 
						|
					sleep(2);
 | 
						|
				}
 | 
						|
			} while (result<0 && errno == EAGAIN);
 | 
						|
			if (result<0 && errno != EADDRINUSE)
 | 
						|
			{
 | 
						|
				fprintf(stderr,
 | 
						|
					"unable to ioctl(NWIOTCPLISTEN): %s\n",
 | 
						|
					strerror(errno));
 | 
						|
				exit(1);
 | 
						|
			}
 | 
						|
			if (result>=0)
 | 
						|
				exit(0);
 | 
						|
			else
 | 
						|
				exit(1);
 | 
						|
		}
 | 
						|
		/*
 | 
						|
		 * This sleep is a HACK.  The command that we are starting
 | 
						|
		 * will try to connect to the fd2 port.  It seems that for
 | 
						|
		 * this to succeed the child process must have already made
 | 
						|
		 * the call to ioctl above (the NWIOTCPLISTEN) call.
 | 
						|
		 * The sleep gives the child a chance to make the call
 | 
						|
		 * before the parent sends the port number to the
 | 
						|
		 * command being started.
 | 
						|
		 */
 | 
						|
		sleep(1);
 | 
						|
 | 
						|
		sprintf(num, "%d", lport);
 | 
						|
		if (write(fd, num, strlen(num)+1) != strlen(num)+1)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to write: %s\n",
 | 
						|
				strerror(errno));
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
	write (fd, locuser, strlen(locuser)+1);
 | 
						|
	write (fd, remuser, strlen(remuser)+1);
 | 
						|
	write (fd, cmd, strlen(cmd)+1);
 | 
						|
	if (read(fd, &c, 1) != 1)
 | 
						|
	{
 | 
						|
		fprintf(stderr, "unable to read: %s\n", strerror(errno) );
 | 
						|
		goto bad;
 | 
						|
	}
 | 
						|
	if (c != 0)
 | 
						|
	{
 | 
						|
		while (read(fd, &c, 1) == 1)
 | 
						|
		{
 | 
						|
			write(2, &c, 1);
 | 
						|
			if (c == '\n')
 | 
						|
				break;
 | 
						|
		}
 | 
						|
		goto bad;
 | 
						|
	}
 | 
						|
	if (fd2p)
 | 
						|
	{
 | 
						|
		*fd2p= fd2;
 | 
						|
		result= ioctl(fd2, NWIOGTCPCONF, &tcpconf);
 | 
						|
		if (result<0)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to ioctl(NWIOGTCPCONF): %s\n",
 | 
						|
				strerror(errno) );
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
		if (ntohs(tcpconf.nwtc_remport) >= TCPPORT_RESERVED)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "unable to setup 2nd channel\n");
 | 
						|
			goto bad;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return fd;
 | 
						|
 | 
						|
bad:
 | 
						|
	if (fd>=0)
 | 
						|
		close(fd);
 | 
						|
	if (fd2>=0)
 | 
						|
		close(fd2);
 | 
						|
	return -1;
 | 
						|
}
 |