422 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			422 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
| /* net.c
 | |
|  *
 | |
|  * This file is part of ftp.
 | |
|  *
 | |
|  *
 | |
|  * 01/25/96 Initial Release	Michael Temari, <temari@ix.netcom.com>
 | |
|  */
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <sys/ioctl.h>
 | |
| #include <sys/wait.h>
 | |
| #include <stdio.h>
 | |
| #include <unistd.h>
 | |
| #include <string.h>
 | |
| #include <unistd.h>
 | |
| #include <stdlib.h>
 | |
| #include <fcntl.h>
 | |
| #include <signal.h>
 | |
| #include <errno.h>
 | |
| #include <net/netlib.h>
 | |
| #include <net/hton.h>
 | |
| #include <net/gen/netdb.h>
 | |
| #include <net/gen/in.h>
 | |
| #include <net/gen/inet.h>
 | |
| #include <net/gen/tcp.h>
 | |
| #include <net/gen/tcp_io.h>
 | |
| 
 | |
| #include "ftp.h"
 | |
| #include "file.h"
 | |
| #include "net.h"
 | |
| 
 | |
| _PROTOTYPE(void donothing, (int sig));
 | |
| 
 | |
| static int ftpcomm_fd;
 | |
| static ipaddr_t myip;
 | |
| static ipaddr_t hostip;
 | |
| static char host[256];
 | |
| static int lpid;
 | |
| 
 | |
| void NETinit()
 | |
| {
 | |
| int s;
 | |
| char *tcp_device;
 | |
| int tcp_fd;
 | |
| nwio_tcpconf_t nwio_tcpconf;
 | |
| 
 | |
|    /* All this just to get our ip address */
 | |
| 
 | |
|    if((tcp_device = getenv("TCP_DEVICE")) == (char *)NULL)
 | |
| 	tcp_device = TCP_DEVICE;
 | |
| 
 | |
|    tcp_fd = open(tcp_device, O_RDWR);
 | |
|    if(tcp_fd < 0) {
 | |
| 	perror("ftp: Could not open tcp_device");
 | |
| 	exit(-1);
 | |
|    }
 | |
|    s = ioctl(tcp_fd, NWIOGTCPCONF, &nwio_tcpconf);
 | |
|    if(s < 0) {
 | |
| 	perror("ftp: Could not get tcp configuration");
 | |
| 	exit(-1);
 | |
|    }
 | |
| 
 | |
|    myip = nwio_tcpconf.nwtc_locaddr;
 | |
| 
 | |
|    close(tcp_fd);
 | |
| }
 | |
| 
 | |
| int DOopen()
 | |
| {
 | |
| nwio_tcpconf_t tcpconf;
 | |
| nwio_tcpcl_t tcpcopt;
 | |
| char *tcp_device;
 | |
| tcpport_t port;
 | |
| int s;
 | |
| struct hostent *hp;
 | |
| struct servent *servent;
 | |
| 
 | |
|    if(linkopen) {
 | |
| 	printf("Use \"CLOSE\" to close the connection first.\n");
 | |
| 	return(0);
 | |
|    }
 | |
| 
 | |
|    if(cmdargc < 2)
 | |
| 	readline("Host: ", host, sizeof(host));
 | |
|    else
 | |
| 	strncpy(host, cmdargv[1], sizeof(host));
 | |
| 
 | |
|    if((servent = getservbyname("ftp", "tcp")) == (struct servent *)NULL) {
 | |
|    	fprintf(stderr, "ftp: Could not find ftp tcp service\n");
 | |
|    	return(-1);
 | |
|    }
 | |
|    port = (tcpport_t)servent->s_port;
 | |
| 
 | |
|    hp = gethostbyname(host);
 | |
|    if (hp == (struct hostent *)NULL) {
 | |
| 	hostip = (ipaddr_t)0;
 | |
| 	printf("Unresolved host %s\n", host);
 | |
| 	return(0);
 | |
|    } else
 | |
| 	memcpy((char *) &hostip, (char *) hp->h_addr, hp->h_length);
 | |
| 
 | |
|    /* This HACK allows the server to establish data connections correctly */
 | |
|    /* when using the loopback device to talk to ourselves */
 | |
|    if(hostip == inet_addr("127.0.0.1"))
 | |
| 	hostip = myip;
 | |
| 
 | |
|    if((tcp_device = getenv("TCP_DEVICE")) == NULL)
 | |
| 	tcp_device = "/dev/tcp";
 | |
| 
 | |
|    if((ftpcomm_fd = open(tcp_device, O_RDWR)) < 0) {
 | |
| 	perror("ftp: open error on tcp device");
 | |
| 	return(-1);
 | |
|    }
 | |
| 
 | |
|    tcpconf.nwtc_flags = NWTC_LP_SEL | NWTC_SET_RA | NWTC_SET_RP;
 | |
|    tcpconf.nwtc_remaddr = hostip;
 | |
|    tcpconf.nwtc_remport = port;
 | |
| 
 | |
|    s = ioctl(ftpcomm_fd, NWIOSTCPCONF, &tcpconf);
 | |
|    if(s < 0) {
 | |
| 	perror("ftp: ioctl error on NWIOSTCPCONF");
 | |
| 	close(ftpcomm_fd);
 | |
| 	return(s);
 | |
|    }
 | |
| 
 | |
|    tcpcopt.nwtcl_flags = 0;
 | |
| 
 | |
|    s = ioctl(ftpcomm_fd, NWIOTCPCONN, &tcpcopt);
 | |
|    if(s < 0) {
 | |
| 	perror("ftp: ioctl error on NWIOTCPCONN");
 | |
| 	close(ftpcomm_fd);
 | |
| 	return(s);
 | |
|    }
 | |
| 
 | |
|    s = ioctl(ftpcomm_fd, NWIOGTCPCONF, &tcpconf);
 | |
|    if(s < 0) {
 | |
| 	perror("ftp: ioctl error on NWIOGTCPCONF");
 | |
| 	close(ftpcomm_fd);
 | |
| 	return(s);
 | |
|    }
 | |
| 
 | |
|    fpcommin  = fdopen(ftpcomm_fd, "r");
 | |
|    fpcommout = fdopen(ftpcomm_fd, "w");
 | |
| 
 | |
|    s = DOgetreply();
 | |
| 
 | |
|    if(s < 0) {
 | |
| 	fclose(fpcommin);
 | |
| 	fclose(fpcommout);
 | |
| 	close(ftpcomm_fd);
 | |
| 	return(s);
 | |
|    }
 | |
| 
 | |
|    if(s != 220) {
 | |
| 	fclose(fpcommin);
 | |
| 	fclose(fpcommout);
 | |
| 	close(ftpcomm_fd);
 | |
| 	return(0);
 | |
|    }
 | |
| 
 | |
|    linkopen = 1;
 | |
| 
 | |
|    return(s);
 | |
| }
 | |
| 
 | |
| int DOclose()
 | |
| {
 | |
|    if(!linkopen) {
 | |
| 	printf("You can't close a connection that isn't open.\n");
 | |
| 	return(0);
 | |
|    }
 | |
| 
 | |
|    fclose(fpcommin);
 | |
|    fclose(fpcommout);
 | |
|    close(ftpcomm_fd);
 | |
| 
 | |
|    linkopen = 0;
 | |
|    loggedin = 0;
 | |
| 
 | |
|    return(0);
 | |
| }
 | |
| 
 | |
| int DOquit()
 | |
| {
 | |
| int s;
 | |
| 
 | |
|    if(linkopen) {
 | |
| 	s = DOcommand("QUIT", "");
 | |
| 	s = DOclose();
 | |
|    }
 | |
| 
 | |
|    printf("FTP done.\n");
 | |
| 
 | |
|    exit(0);
 | |
| }
 | |
| 
 | |
| void donothing(sig)
 | |
| int sig;
 | |
| {
 | |
| }
 | |
| 
 | |
| int DOdata(datacom, file, direction, fd)
 | |
| char *datacom;
 | |
| char *file;
 | |
| int direction;  /* RETR or STOR */
 | |
| int fd;
 | |
| {
 | |
| nwio_tcpconf_t tcpconf;
 | |
| nwio_tcpcl_t tcplopt, tcpcopt;
 | |
| char *tcp_device;
 | |
| int ftpdata_fd;
 | |
| char *buff;
 | |
| ipaddr_t ripaddr;
 | |
| tcpport_t rport;
 | |
| static tcpport_t lport = HTONS(0xF000);
 | |
| int s;
 | |
| int i;
 | |
| int cs;
 | |
| int pfd[2];
 | |
| char dummy;
 | |
| char port[32];
 | |
| 
 | |
|    ripaddr = hostip;
 | |
|    rport = HTONS(20);
 | |
| 
 | |
|    /* here we set up a connection to listen on if not passive mode */
 | |
|    /* otherwise we use this to connect for passive mode */
 | |
| 
 | |
|    if((tcp_device = getenv("TCP_DEVICE")) == NULL)
 | |
| 	tcp_device = "/dev/tcp";
 | |
| 
 | |
|    if((ftpdata_fd = open(tcp_device, O_RDWR)) < 0) {
 | |
| 	perror("ftp: open error on tcp device");
 | |
| 	return(-1);
 | |
|    }
 | |
| 
 | |
|    if(passive) {
 | |
| 	s = DOcommand("PASV", "");
 | |
| 	if(s != 227) {
 | |
| 		close(ftpdata_fd);
 | |
| 		return(s);
 | |
| 	}
 | |
| 	/* decode host and port */
 | |
| 	buff = reply;
 | |
| 	while(*buff && (*buff != '(')) buff++;
 | |
| 	buff++;
 | |
| 	ripaddr = (ipaddr_t)0;
 | |
| 	for(i = 0; i < 4; i++) {
 | |
| 		ripaddr = (ripaddr << 8) + (ipaddr_t)atoi(buff);
 | |
| 		if((buff = strchr(buff, ',')) == (char *)0) {
 | |
| 			printf("Could not parse PASV reply\n");
 | |
| 			return(-1);
 | |
| 		}
 | |
| 		buff++;
 | |
| 	}
 | |
| 	rport = (tcpport_t)atoi(buff);
 | |
| 	if((buff = strchr(buff, ',')) == (char *)0) {
 | |
| 		printf("Could not parse PASV reply\n");
 | |
| 		return(-1);
 | |
| 	}
 | |
| 	buff++;
 | |
| 	rport = (rport << 8) + (tcpport_t)atoi(buff);
 | |
| 	ripaddr = ntohl(ripaddr);
 | |
| 	rport = ntohs(rport);
 | |
|    }
 | |
| 
 | |
|    for (;;) {
 | |
| 	tcpconf.nwtc_flags = NWTC_SET_RA | NWTC_SET_RP;
 | |
| 	if (passive || ntohs(lport) >= 0xF000) {
 | |
| 		tcpconf.nwtc_flags |= NWTC_LP_SEL;
 | |
| 	} else {
 | |
| 		/* For no good reason Sun hosts don't like it if they have to
 | |
| 		 * connect to the same port twice in a short time...
 | |
| 		 */
 | |
| 		lport = htons(ntohs(lport) + 1);
 | |
| 		tcpconf.nwtc_flags |= NWTC_LP_SET;
 | |
| 		tcpconf.nwtc_locport = lport;
 | |
| 	}
 | |
| 
 | |
| 	tcpconf.nwtc_remaddr = ripaddr;
 | |
| 	tcpconf.nwtc_remport = rport;
 | |
| 
 | |
| 	s = ioctl(ftpdata_fd, NWIOSTCPCONF, &tcpconf);
 | |
| 	if(s < 0) {
 | |
| 		if (errno == EADDRINUSE) continue;
 | |
| 		perror("ftp: ioctl error on NWIOSTCPCONF");
 | |
| 		close(ftpdata_fd);
 | |
| 		return(s);
 | |
| 	}
 | |
| 	break;
 | |
|    }
 | |
| 
 | |
|    s = ioctl(ftpdata_fd, NWIOGTCPCONF, &tcpconf);
 | |
|    if(s < 0) {
 | |
| 	perror("ftp: ioctl error on NWIOGTCPCONF");
 | |
| 	close(ftpdata_fd);
 | |
| 	return(s);
 | |
|    }
 | |
|    lport = tcpconf.nwtc_locport;
 | |
| 
 | |
|    if(passive) {
 | |
| 	tcplopt.nwtcl_flags = 0;
 | |
| 	s = ioctl(ftpdata_fd, NWIOTCPCONN, &tcpcopt);
 | |
| 	if(s < 0) {
 | |
| 		perror("ftp: error on ioctl NWIOTCPCONN");
 | |
| 		close(ftpdata_fd);
 | |
| 		return(0);
 | |
| 	}
 | |
| 	s = ioctl(ftpdata_fd, NWIOGTCPCONF, &tcpconf);
 | |
| 	if(s < 0) {
 | |
| 		perror("ftp: error on ioctl NWIOGTCPCONF");
 | |
| 		close(ftpdata_fd);
 | |
| 		return(0);
 | |
| 	}
 | |
|    } else {
 | |
| 	tcplopt.nwtcl_flags = 0;
 | |
| 
 | |
| 	if (pipe(pfd) < 0) {
 | |
| 		perror("ftp: could not create a pipe");
 | |
| 		return(s);
 | |
| 	}
 | |
| 	lpid = fork();
 | |
| 	if(lpid < 0) {
 | |
| 		perror("ftp: could not fork listener");
 | |
| 		close(ftpdata_fd);
 | |
| 		close(pfd[0]);
 | |
| 		close(pfd[1]);
 | |
| 		return(s);
 | |
| 	} else if(lpid == 0) {
 | |
| 		close(pfd[0]);
 | |
| 		signal(SIGALRM, donothing);
 | |
| 		alarm(15);
 | |
| 		close(pfd[1]);
 | |
| 		s = ioctl(ftpdata_fd, NWIOTCPLISTEN, &tcplopt);
 | |
| 		alarm(0);
 | |
| 		if(s < 0)
 | |
| 			if(errno == EINTR)
 | |
| 				exit(1);	/* timed out */
 | |
| 			else
 | |
| 				exit(-1);	/* error */
 | |
| 		else
 | |
| 			exit(0);		/* connection made */
 | |
| 	}
 | |
| 	/* Wait for the pipe to close, then the listener is ready (almost). */
 | |
| 	close(pfd[1]);
 | |
| 	(void) read(pfd[0], &dummy, 1);
 | |
| 	close(pfd[0]);
 | |
| 	while(1) {
 | |
| 		signal(SIGALRM, donothing);
 | |
| 		alarm(1);
 | |
| 		s = ioctl(ftpdata_fd, NWIOGTCPCONF, &tcpconf);
 | |
| 		alarm(0);
 | |
| 		if(s == -1) break;
 | |
| 	}
 | |
|    }
 | |
| 
 | |
| #define hiword(x)       ((u16_t)((x) >> 16))
 | |
| #define loword(x)       ((u16_t)(x & 0xffff)) 
 | |
| #define hibyte(x)       (((x) >> 8) & 0xff)
 | |
| #define lobyte(x)       ((x) & 0xff)
 | |
| 
 | |
|    if(!passive) {
 | |
| 	sprintf(port, "%u,%u,%u,%u,%u,%u",
 | |
| 		hibyte(hiword(ntohl(myip))), lobyte(hiword(ntohl(myip))),
 | |
| 		hibyte(loword(ntohl(myip))), lobyte(loword(ntohl(myip))),
 | |
| 		hibyte(ntohs(lport)), lobyte(ntohs(lport)));
 | |
| 	s = DOcommand("PORT", port);
 | |
| 	if(s != 200) {
 | |
| 		close(ftpdata_fd);
 | |
| 		kill(lpid, SIGKILL);
 | |
| 		return(s);
 | |
| 	}
 | |
|    }
 | |
| 
 | |
|    s = DOcommand(datacom, file);
 | |
|    if(s == 125 || s == 150) {
 | |
| 	if(!passive) {
 | |
| 		while(1) {
 | |
| 			s = wait(&cs);
 | |
| 			if(s < 0 || s == lpid)
 | |
| 				break;
 | |
| 		}
 | |
| 		if(s < 0) {
 | |
| 			perror("wait error:");
 | |
| 			close(ftpdata_fd);
 | |
| 			kill(lpid, SIGKILL);
 | |
| 			return(s);
 | |
| 		}
 | |
| 		if((cs & 0x00ff)) {
 | |
| 			printf("Child listener failed %04x\n", cs);
 | |
| 			close(ftpdata_fd);
 | |
| 			return(-1);
 | |
| 		}
 | |
| 		cs = (cs >> 8) & 0x00ff;
 | |
| 		if(cs == 1) {
 | |
| 			printf("Child listener timed out\n");
 | |
| 			return(DOgetreply());
 | |
| 		} else if(cs) {
 | |
| 			printf("Child listener returned %02x\n", cs);
 | |
| 			close(ftpdata_fd);
 | |
| 			return(-1);
 | |
| 		}
 | |
| 	}
 | |
| 	switch(direction) {
 | |
| 		case RETR:
 | |
| 			s = recvfile(fd, ftpdata_fd);
 | |
| 			break;
 | |
| 		case STOR:
 | |
| 			s = sendfile(fd, ftpdata_fd);
 | |
| 			break;
 | |
| 	}
 | |
| 	close(ftpdata_fd);
 | |
| 	s = DOgetreply();
 | |
|    } else {
 | |
| 	if(!passive)
 | |
| 		kill(lpid, SIGKILL);
 | |
| 	close(ftpdata_fd);
 | |
|    }
 | |
| 
 | |
|    return(s);
 | |
| }
 | 
