6a73e85ad1
. only good for obsolete K&R support . also remove a stray ansi.h and the proto cmd
579 lines
11 KiB
C
579 lines
11 KiB
C
/* net.c Copyright 1992-2000 by Michael Temari All Rights Reserved
|
|
*
|
|
* This file is part of ftp.
|
|
*
|
|
*
|
|
* 01/25/96 Initial Release Michael Temari, <Michael@TemWare.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 "xfer.h"
|
|
#include "net.h"
|
|
|
|
void donothing(int sig);
|
|
|
|
int ftpcomm_fd;
|
|
static ipaddr_t myip;
|
|
static ipaddr_t hostip;
|
|
static char host[256];
|
|
static int lpid;
|
|
|
|
int 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");
|
|
return(-1);
|
|
}
|
|
s = ioctl(tcp_fd, NWIOGTCPCONF, &nwio_tcpconf);
|
|
if(s < 0) {
|
|
perror("ftp: Could not get tcp configuration");
|
|
return(-1);
|
|
}
|
|
|
|
myip = nwio_tcpconf.nwtc_locaddr;
|
|
|
|
close(tcp_fd);
|
|
|
|
return(0);
|
|
}
|
|
|
|
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) {
|
|
if(readline("Host: ", host, sizeof(host)) < 0)
|
|
return(-1);
|
|
} 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");
|
|
port = htons(21);
|
|
} else
|
|
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 */
|
|
#ifdef __NBSD_LIBC
|
|
if((hostip & ntohl(0xFF000000)) == inet_addr("127.0.0.0"))
|
|
#else
|
|
if((hostip & NTOHL(0xFF000000)) == inet_addr("127.0.0.0"))
|
|
#endif
|
|
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(-1);
|
|
}
|
|
|
|
tcpcopt.nwtcl_flags = 0;
|
|
|
|
s = ioctl(ftpcomm_fd, NWIOTCPCONN, &tcpcopt);
|
|
if(s < 0) {
|
|
perror("ftp: ioctl error on NWIOTCPCONN");
|
|
close(ftpcomm_fd);
|
|
return(-1);
|
|
}
|
|
|
|
s = ioctl(ftpcomm_fd, NWIOGTCPCONF, &tcpconf);
|
|
if(s < 0) {
|
|
perror("ftp: ioctl error on NWIOGTCPCONF");
|
|
close(ftpcomm_fd);
|
|
return(-1);
|
|
}
|
|
|
|
s = DOgetreply();
|
|
|
|
if(s < 0) {
|
|
close(ftpcomm_fd);
|
|
return(s);
|
|
}
|
|
|
|
if(s != 220) {
|
|
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);
|
|
}
|
|
|
|
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;
|
|
static int ftpdata_fd = -1;
|
|
char *buff;
|
|
ipaddr_t ripaddr;
|
|
tcpport_t rport;
|
|
#ifdef __NBSD_LIBC
|
|
static tcpport_t lport = htons(0xF000);
|
|
#else
|
|
static tcpport_t lport = HTONS(0xF000);
|
|
#endif
|
|
int s;
|
|
int i;
|
|
int wpid;
|
|
int cs;
|
|
int pfd[2];
|
|
char dummy;
|
|
char port[32];
|
|
int wasopen;
|
|
|
|
#ifdef DEBUG
|
|
printf("DOdata %s %s %d %d\n", datacom, file, direction, fd);
|
|
#endif
|
|
|
|
ripaddr = hostip;
|
|
#ifdef __NBSD_LIBC
|
|
rport = htons(2);
|
|
#else
|
|
rport = HTONS(20);
|
|
#endif
|
|
|
|
/* 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 >= 0 && mode != MODE_B) {
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
}
|
|
|
|
wasopen = (ftpdata_fd >= 0);
|
|
|
|
#ifdef DEBUG
|
|
printf("wasopen = %d\n", wasopen);
|
|
#endif
|
|
|
|
if(wasopen)
|
|
goto WASOPEN;
|
|
|
|
#ifdef DEBUG
|
|
printf("b4 open = %d\n", ftpdata_fd);
|
|
#endif
|
|
|
|
if(ftpdata_fd == -1)
|
|
if((ftpdata_fd = open(tcp_device, O_RDWR)) < 0) {
|
|
perror("ftp: open error on tcp device");
|
|
return(0);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
printf("at open = %d\n", ftpdata_fd);
|
|
#endif
|
|
|
|
if(passive) {
|
|
#ifdef DEBUG
|
|
printf("b4 PASV command\n");
|
|
#endif
|
|
s = DOcommand("PASV", "");
|
|
#ifdef DEBUG
|
|
printf("PASV command returned %d\n", s);
|
|
#endif
|
|
if(s != 227) {
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
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(0);
|
|
}
|
|
buff++;
|
|
}
|
|
rport = (tcpport_t)atoi(buff);
|
|
if((buff = strchr(buff, ',')) == (char *)0) {
|
|
printf("Could not parse PASV reply\n");
|
|
return(0);
|
|
}
|
|
buff++;
|
|
rport = (rport << 8) + (tcpport_t)atoi(buff);
|
|
ripaddr = ntohl(ripaddr);
|
|
rport = ntohs(rport);
|
|
#ifdef DEBUG
|
|
printf("PASV %08x %04x\n", ripaddr, rport);
|
|
#endif
|
|
}
|
|
|
|
while(1) {
|
|
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;
|
|
|
|
#ifdef DEBUG
|
|
printf("b4 STCPCONF locport = %d\n", lport);
|
|
#endif
|
|
|
|
s = ioctl(ftpdata_fd, NWIOSTCPCONF, &tcpconf);
|
|
#ifdef DEBUG
|
|
printf("at STCPCONF %d %d\n", s, errno);
|
|
#endif
|
|
if(s < 0) {
|
|
if(errno == EADDRINUSE) continue;
|
|
perror("ftp: ioctl error on NWIOSTCPCONF");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(0);
|
|
}
|
|
break;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
printf("b4 GTCPCONF\n");
|
|
#endif
|
|
s = ioctl(ftpdata_fd, NWIOGTCPCONF, &tcpconf);
|
|
#ifdef DEBUG
|
|
printf("at GTCPCONF %d %d\n", s, errno);
|
|
#endif
|
|
if(s < 0) {
|
|
perror("ftp: ioctl error on NWIOGTCPCONF");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(0);
|
|
}
|
|
lport = tcpconf.nwtc_locport;
|
|
|
|
#ifdef DEBUG
|
|
printf("lport = %04x\n", lport);
|
|
#endif
|
|
|
|
if(passive) {
|
|
/* passive mode we connect to them */
|
|
tcpcopt.nwtcl_flags = 0;
|
|
#ifdef DEBUG
|
|
printf("Doing TCPCONN\n");
|
|
#endif
|
|
s = ioctl(ftpdata_fd, NWIOTCPCONN, &tcpcopt);
|
|
#ifdef DEBUG
|
|
printf("TCPCONN %d %d\n", s, errno);
|
|
#endif
|
|
if(s < 0) {
|
|
perror("ftp: error on ioctl NWIOTCPCONN");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(0);
|
|
}
|
|
#ifdef DEBUG
|
|
printf("GTCPCONF\n");
|
|
#endif
|
|
s = ioctl(ftpdata_fd, NWIOGTCPCONF, &tcpconf);
|
|
#ifdef DEBUG
|
|
printf("GTCPCONF %d %d\n", s, errno);
|
|
#endif
|
|
if(s < 0) {
|
|
perror("ftp: error on ioctl NWIOGTCPCONF");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(0);
|
|
}
|
|
} else {
|
|
/* we listen for them */
|
|
tcplopt.nwtcl_flags = 0;
|
|
#ifdef DEBUG
|
|
printf("Listen\n");
|
|
#endif
|
|
|
|
if (pipe(pfd) < 0) {
|
|
perror("ftp: could not create a pipe");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(0);
|
|
}
|
|
lpid = fork();
|
|
if(lpid < 0) {
|
|
perror("ftp: could not fork listener");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
close(pfd[0]);
|
|
close(pfd[1]);
|
|
return(0);
|
|
} else if(lpid == 0) {
|
|
#ifdef DEBUG
|
|
printf("Child here\n");
|
|
#endif
|
|
close(pfd[0]);
|
|
signal(SIGALRM, donothing);
|
|
alarm(15);
|
|
close(pfd[1]);
|
|
#ifdef DEBUG
|
|
printf("child TCPLISTEN\n");
|
|
#endif
|
|
s = ioctl(ftpdata_fd, NWIOTCPLISTEN, &tcplopt);
|
|
alarm(0);
|
|
#ifdef DEBUG
|
|
printf("listen %d %d\n", s, errno);
|
|
#endif
|
|
if(s < 0)
|
|
exit(errno); /* error */
|
|
else
|
|
exit(0); /* connection made */
|
|
}
|
|
#ifdef DEBUG
|
|
printf("Fork = %d\n", lpid);
|
|
#endif
|
|
/* 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) {
|
|
wpid = waitpid(lpid, &cs, WNOHANG);
|
|
#ifdef DEBUG
|
|
printf("waitpid %d %d\n", wpid, cs);
|
|
printf("GTCPCONF loop\n");
|
|
#endif
|
|
if(wpid != 0) break;
|
|
signal(SIGALRM, donothing);
|
|
alarm(1);
|
|
s = ioctl(ftpdata_fd, NWIOGTCPCONF, &tcpconf);
|
|
alarm(0);
|
|
#ifdef DEBUG
|
|
printf("GTCPCONF loop %d %d\n", s, errno);
|
|
#endif
|
|
if(s == -1) break;
|
|
sleep(1);
|
|
}
|
|
#ifdef DEBUG
|
|
printf("GTCPCONF = %d\n", s);
|
|
#endif
|
|
}
|
|
|
|
#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) {
|
|
if(wpid != 0) {
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
cs = (cs >> 8) & 0x00ff;
|
|
printf("Child listener error %s\n", strerror(cs));
|
|
return(0);
|
|
}
|
|
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)));
|
|
#ifdef DEBUG
|
|
printf("sending port command %s\n", port);
|
|
#endif
|
|
s = DOcommand("PORT", port);
|
|
#ifdef DEBUG
|
|
printf("port command = %d\n", s);
|
|
#endif
|
|
if(s != 200) {
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
kill(lpid, SIGKILL);
|
|
(void) wait(&cs);
|
|
return(s);
|
|
}
|
|
}
|
|
|
|
WASOPEN:
|
|
|
|
#ifdef DEBUG
|
|
printf("doing data command %s %s\n", datacom, file);
|
|
#endif
|
|
s = DOcommand(datacom, file);
|
|
#ifdef DEBUG
|
|
printf("do command reply %d\n", s);
|
|
#endif
|
|
if(s == 125 || s == 150) {
|
|
if(!passive && !wasopen) {
|
|
while(1) {
|
|
#ifdef DEBUG
|
|
printf("Waiting for child %d\n", lpid);
|
|
#endif
|
|
s = wait(&cs);
|
|
#ifdef DEBUG
|
|
printf("Wait returned %d cs=%d errno=%d\n", s, cs, errno);
|
|
#endif
|
|
if(s < 0 || s == lpid)
|
|
break;
|
|
}
|
|
if(s < 0) {
|
|
perror("wait error:");
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
kill(lpid, SIGKILL);
|
|
(void) wait(&cs);
|
|
return(s);
|
|
}
|
|
if((cs & 0x00ff)) {
|
|
printf("Child listener failed %04x\n", cs);
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(-1);
|
|
}
|
|
cs = (cs >> 8) & 0x00ff;
|
|
if(cs) {
|
|
printf("Child listener error %s\n", strerror(cs));
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
return(DOgetreply());
|
|
}
|
|
}
|
|
#ifdef DEBUG
|
|
printf("Before recvfile/sendfile call\n");
|
|
#endif
|
|
switch(direction) {
|
|
case RETR:
|
|
s = recvfile(fd, ftpdata_fd);
|
|
break;
|
|
case STOR:
|
|
s = sendfile(fd, ftpdata_fd);
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
printf("send/recieve %d\n", s);
|
|
#endif
|
|
if(mode != MODE_B) {
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
}
|
|
|
|
s = DOgetreply();
|
|
#ifdef DEBUG
|
|
printf("send/recieve reply %d\n", s);
|
|
#endif
|
|
if(mode == MODE_B && s == 226) {
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
}
|
|
} else {
|
|
if(!passive) {
|
|
kill(lpid, SIGKILL);
|
|
(void) wait(&cs);
|
|
}
|
|
close(ftpdata_fd);
|
|
ftpdata_fd = -1;
|
|
}
|
|
|
|
return(s);
|
|
}
|