/* network.c
 *
 */

#include "stdinclude.h"

#include <sys/socket.h>

#include <arpa/inet.h>
#include <arpa/telnet.h>
#include <netinet/in.h>

#include <errno.h>

#include "common.h"
#include "config.h"
#include "ficsmain.h"
#include "network.h"
#include "playerdb.h"
#include "rmalloc.h"
#ifdef TIMESEAL
#include "timeseal.h"
#endif
#include "utils.h"

/* Sparse array */
PUBLIC connection con[512];

PUBLIC int	 no_file;
PUBLIC int	 max_connections;

PRIVATE int	 sockfd = 0;
PRIVATE int	 numConnections = 0;

PUBLIC int
findConnection(int fd)
{
	if (con[fd].status == NETSTAT_EMPTY)
		return -1;
	return fd;
}

PUBLIC int
net_addConnection(int fd, unsigned int fromHost)
{
	int noblock = 1;

	if (findConnection(fd) >= 0) {
		fprintf(stderr, "FICS: FD already in connection table!\n");
		return -1;
	}
	if (numConnections >= max_connections)
		return -1;
	if (ioctl(fd, FIONBIO, &noblock) == -1) {
		fprintf(stderr, "Error setting nonblocking mode errno=%d\n",
		    errno);
	}

	con[fd].fd = fd;

	if (fd != 0)
		con[fd].outFd = fd;
	else
		con[fd].outFd = 1;

	con[fd].fromHost	= fromHost;
	con[fd].status		= NETSTAT_CONNECTED;
#ifdef TIMESEAL
	con[fd].sys[0]		= '\0';
	con[fd].time		= 0;
	con[fd].timeseal	= 0;
	con[fd].user[0]		= '\0';
#endif

	con[fd].numPending	= 0;
	con[fd].outPos		= 0;
	con[fd].processed	= 0;

	if (con[fd].sndbuf == NULL) {
#ifdef DEBUG
		fprintf(stderr, "FICS: nac(%d) allocating sndbuf.\n", fd);
#endif
		con[fd].sndbufpos = 0;
		con[fd].sndbufsize = MAX_STRING_LENGTH;
		con[fd].sndbuf = rmalloc(MAX_STRING_LENGTH);
	} else {
#ifdef DEBUG
		fprintf(stderr, "FICS: nac(%d) reusing old sndbuf "
		    "size %d pos %d.\n", fd,
		    con[fd].sndbufsize, con[fd].sndbufpos);
#else
		/* empty */;
#endif
	}
	con[fd].state = 0;

	numConnections++;

#ifdef DEBUG
	fprintf(stderr, "FICS: fd: %d connections: %d  descriptors: %d \n", fd,
	    numConnections, getdtablesize()); /* sparky 3/13/95 */
#endif

	return 0;
}

PRIVATE int
remConnection(int fd)
{
	int which;

	if ((which = findConnection(fd)) < 0)
		return -1;
	numConnections--;

	con[fd].status = NETSTAT_EMPTY;

	if (con[fd].sndbuf == NULL) {
		fprintf(stderr, "FICS: remcon(%d) SNAFU, "
		    "this shouldn't happen.\n", fd);
	} else {
		if (con[fd].sndbufsize > MAX_STRING_LENGTH) {
			con[fd].sndbufsize = MAX_STRING_LENGTH;
			con[fd].sndbuf = rrealloc(con[fd].sndbuf,
			    MAX_STRING_LENGTH);
		}

		if (con[fd].sndbufpos) // didn't send everything, bummer
			con[fd].sndbufpos = 0;
	}

	return 0;
}

PRIVATE void
net_flushme(int which)
{
	int sent;

	if ((sent = send(con[which].outFd, con[which].sndbuf,
	    con[which].sndbufpos, 0)) == -1) {
		if (errno != EPIPE) { // EPIPE = they've disconnected
			fprintf(stderr, "FICS: net_flushme(%d) couldn't send, "
			    "errno=%d.\n",
			    which, errno);
		}

		con[which].sndbufpos = 0;
	} else {
		con[which].sndbufpos -= sent;

		if (con[which].sndbufpos) {
			memmove(con[which].sndbuf, con[which].sndbuf + sent,
			    con[which].sndbufpos);
		}
	}

	if (con[which].sndbufsize > MAX_STRING_LENGTH &&
	    con[which].sndbufpos < MAX_STRING_LENGTH) {
		// time to shrink the buffer...
		con[which].sndbuf = rrealloc(con[which].sndbuf,
		    MAX_STRING_LENGTH);
		con[which].sndbufsize = MAX_STRING_LENGTH;
	}
}

PRIVATE void
net_flush_all_connections(void)
{
	fd_set		 writefds;
	int		 which;
	struct timeval	 to;

	FD_ZERO(&writefds);

	for (which = 0; which < MAX_PLAYER; which++) {
		if (con[which].status == NETSTAT_CONNECTED &&
		    con[which].sndbufpos)
			FD_SET(con[which].outFd, &writefds);
	}

	to.tv_usec = 0;
	to.tv_sec = 0;

	select(no_file, NULL, &writefds, NULL, &to);

	for (which = 0; which < MAX_PLAYER; which++) {
		if (FD_ISSET(con[which].outFd, &writefds))
			net_flushme(which);
	}
}

PRIVATE void
net_flush_connection(int fd)
{
	fd_set		 writefds;
	int		 which;
	struct timeval	 to;

	if ((which = findConnection(fd)) >= 0 && con[which].sndbufpos) {
		FD_ZERO(&writefds);
		FD_SET(con[which].outFd, &writefds);

		to.tv_usec = 0;
		to.tv_sec = 0;

		select(no_file, NULL, &writefds, NULL, &to);

		if (FD_ISSET(con[which].outFd, &writefds))
			net_flushme(which);
	}
}

PRIVATE int
sendme(int which, char *str, int len)
{
	fd_set		 writefds;
	int		 i, count;
	struct timeval	 to;

	count = len;

	while ((i = ((con[which].sndbufsize - con[which].sndbufpos) < len) ?
	    (con[which].sndbufsize - con[which].sndbufpos) : len) > 0) {
		memmove(con[which].sndbuf + con[which].sndbufpos, str, i);
		con[which].sndbufpos += i;

		if (con[which].sndbufpos == con[which].sndbufsize) {
			FD_ZERO(&writefds);
			FD_SET(con[which].outFd, &writefds);

			to.tv_usec	= 0;
			to.tv_sec	= 0;

			select(no_file, NULL, &writefds, NULL, &to);

			if (FD_ISSET(con[which].outFd, &writefds)) {
				net_flushme(which);
			} else {
				// time to grow the buffer
				con[which].sndbufsize += MAX_STRING_LENGTH;
				con[which].sndbuf = rrealloc(con[which].sndbuf,
				    con[which].sndbufsize);
			}
		}

		str += i;
		len -= i;
	}

	return count;
}

/*
 * Put LF after every CR and put '\' at the end of overlength lines.
 *
 * Doesn't send anything unless the buffer fills and output waits
 * until flushed.
 *
 * '-1' for an error other than 'EWOULDBLOCK'.
 */
PUBLIC int
net_send_string(int fd, char *str, int format)
{
	int which, i, j;

	if ((which = findConnection(fd)) < 0)
		return -1;
	while (*str) {
		for (i = 0; str[i] >= ' '; i++) {
			/* null */;
		}

		if (i) {
			if (format &&
			    (i >= (j = LINE_WIDTH - con[which].outPos))) {
				// word wrap

				i = j;

				while (i > 0 && str[i - 1] != ' ')
					i--;
				while (i > 0 && str[i - 1] == ' ')
					i--;
				if (i == 0)
					i = j - 1;
				sendme(which, str, i);
				sendme(which, "\n\r\\   ", 6);
				con[which].outPos = 4;

				while (str[i] == ' ') {	// eat the leading
							// spaces after we wrap
					i++;
				}
			} else {
				sendme(which, str, i);
				con[which].outPos += i;
			}
			str += i;
		} else { // non-printable stuff handled here
			switch (*str) {
			case '\t':
				sendme(which, "        ",
				    8 - (con[which].outPos & 7));
				con[which].outPos &= ~7;
				if (con[which].outPos += 8 >= LINE_WIDTH)
					con[which].outPos = 0;
				break;
			case '\n':
				sendme(which, "\n\r", 2);
				con[which].outPos = 0;
				break;
			case '\033':
				con[which].outPos -= 3;
			default:
				sendme(which, str, 1);
			}
			str++;
		}
	}
	return 0;
}

/*
 * A) if we get a complete line (something terminated by '\n'), copy it
 * to com and return 1.
 *
 * B) if we don't get a complete line, but there is no error, return 0.
 *
 * C) if some error, return -1.
 */
PUBLIC int
readline2(char *com, int who)
{
	int		 howmany, state, fd, pending;
	unsigned char	*start, *s, *d;

	static unsigned char	ayt[] = "[Responding to AYT: Yes, I'm here.]\n";
	static unsigned char	will_sga[] = { IAC, WILL, TELOPT_SGA, '\0' };
	static unsigned char	will_tm[] = { IAC, WILL, TELOPT_TM, '\0' };

	state = con[who].state;

	if (state == 2 || state > 4) {
		fprintf(stderr, "FICS: state screwed for con[%d], "
		    "this is a bug.\n", who);
		state = 0;
	}

	s = start = con[who].inBuf;
	pending = con[who].numPending;
	fd = con[who].fd;

	if ((howmany = recv(fd, start + pending, MAX_STRING_LENGTH - 1 -
	    pending, 0)) == 0) { // error: they've disconnected
		return -1;
	} else if (howmany == -1) {
		if (errno != EWOULDBLOCK) { // some other error
			return -1;
		} else if (con[who].processed) { // nothing new and nothing old
			return 0;
		} else {	// nothing new
				// but some unprocessed old
			howmany = 0;
		}
	}

	if (con[who].processed)
		s += pending;
	else
		howmany += pending;
	d = s;

	for (; howmany-- > 0; s++) {
		switch (state) {
		case 0:	// haven't skipped over any control chars or
			// telnet commands
			if (*s == IAC) {
				d = s;
				state = 1;
			} else if (*s == '\n') {
				*s = '\0';
				sprintf(com, "%s", start);
				if (howmany)
					bcopy(s + 1, start, howmany);
				con[who].state		= 0;
				con[who].numPending	= howmany;
				con[who].processed	= 0;
				con[who].outPos		= 0;
				return 1;
			} else if (*s > (0xff - 0x20) || *s < 0x20) {
				d = s;
				state = 2;
			}
			break;
		case 1: // got telnet IAC
			if (*s == IP) {
				return -1; // ^C = logout
			} else if (*s == DO) {
				state = 4;
			} else if (*s == WILL || *s == DONT || *s == WONT) {
				state = 3;	// this is cheesy
						// but we aren't using em
			} else if (*s == AYT) {
				send(fd, (char *)ayt, strlen((char *)ayt), 0);
				state = 2;
			} else if (*s == EL) {	// erase line
				d = start;
				state = 2;
			} else {	// dunno what it is
					// so ignore it
				state = 2;
			}
			break;
		case 2:	// we've skipped over something
			// need to shuffle processed chars down
			if (*s == IAC)
				state = 1;
			else if (*s == '\n') {
				*d = '\0';
				sprintf(com, "%s", start);
				if (howmany)
					memmove(start, s + 1, howmany);
				con[who].state		= 0;
				con[who].numPending	= howmany;
				con[who].processed	= 0;
				con[who].outPos		= 0;
				return 1;
			} else if (*s >= ' ')
				*(d++) = *s;
			break;
		case 3: // some telnet junk we're ignoring
			state = 2;
			break;
		case 4: // got IAC DO
			if (*s == TELOPT_TM) {
				send(fd, (char *)will_tm,
				    strlen((char *)will_tm), 0);
			} else if (*s == TELOPT_SGA) {
				send(fd, (char *)will_sga,
				    strlen((char *)will_sga), 0);
			}
			state = 2;
			break;
		}
	}

	if (state == 0)
		d = s;
	else if (state == 2)
		state = 0;

	con[who].state = state;
	con[who].numPending = d - start;
	con[who].processed = 1;

	if (con[who].numPending == MAX_STRING_LENGTH - 1) { // buffer full
		*d = '\0';
		sprintf(com, "%s", start);
		con[who].state		= 0;
		con[who].numPending	= 0;
		con[who].processed	= 0;
		return 1;
	}
	return 0;
}

PUBLIC int
net_init(int port)
{
	int			 opt;
	struct linger		 lingeropt;
	struct sockaddr_in	 serv_addr;

	/*
	 * Although we have 256 descriptors to work with for opening
	 * files, we can only use 126 for sockets under SunOS 4.x.x
	 * socket libs. Using glibc can get you up to 256 again. Many
	 * OS's can do more than that!
	 * Sparky 9/20/95
	 */

	if ((no_file = getdtablesize()) > MAX_PLAYER + 6)
		no_file = MAX_PLAYER + 6;
	max_connections = no_file - 6;

	for (int i = 0; i < no_file; i++) {
		con[i].status = NETSTAT_EMPTY;
		con[i].sndbuf = NULL;
		con[i].sndbufsize = con[i].sndbufpos = 0;
	}

	/* Open a TCP socket (an Internet stream socket) */
	if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
		fprintf(stderr, "FICS: can't open stream socket\n");
		return -1;
	}

	/* Bind our local address so that the client can send to us */
	memset(&serv_addr, 0, sizeof serv_addr);
	serv_addr.sin_family		= AF_INET;
	serv_addr.sin_addr.s_addr	= htonl(INADDR_ANY);
	serv_addr.sin_port		= htons(port);

	/*
	 * Attempt to allow rebinding to the port...
	 */

	opt = 1;
	setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (char *) &opt, sizeof opt);

	opt = 1;
	setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, (char *) &opt, sizeof opt);

	lingeropt.l_onoff	= 0;
	lingeropt.l_linger	= 0;
	setsockopt(sockfd, SOL_SOCKET, SO_LINGER, (char *) &lingeropt,
	    sizeof(lingeropt));

	if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof serv_addr) < 0) {
		fprintf(stderr, "FICS: can't bind local address.  errno=%d\n",
		    errno);
		return -1;
	}

	opt = 1;
	ioctl(sockfd, FIONBIO, &opt);
	listen(sockfd, 5);
	return 0;
}

PUBLIC void
net_close(void)
{
	for (int i = 0; i < no_file; i++) {
		if (con[i].status != NETSTAT_EMPTY)
			net_close_connection(con[i].fd);
	}
}

PUBLIC void
net_close_connection(int fd)
{
	if (con[fd].status == NETSTAT_CONNECTED)
		net_flush_connection(fd);
	if (!remConnection(fd)) {
		if (fd > 2)
			close(fd);
	}
}

PUBLIC void
turn_echo_on(int fd)
{
	static unsigned char wont_echo[] = { IAC, WONT, TELOPT_ECHO, '\0' };
	send(fd, (char *) wont_echo, strlen((char *) wont_echo), 0);
}

PUBLIC void
turn_echo_off(int fd)
{
	static unsigned char will_echo[] = { IAC, WILL, TELOPT_ECHO, '\0' };
	send(fd, (char *) will_echo, strlen((char *) will_echo), 0);
}

PUBLIC unsigned int
net_connected_host(int fd)
{
	int which;

	if ((which = findConnection(fd)) < 0) {
		fprintf(stderr, "FICS: FD not in connection table!\n");
		return -1;
	}
	return con[which].fromHost;
}

PUBLIC void
ngc2(char *com, int timeout)
{
	fd_set			 readfds;
	int			 fd, loop, nfound, lineComplete;
	socklen_t		 cli_len = sizeof(struct sockaddr_in);
	struct sockaddr_in	 cli_addr;
	struct timeval		 to;

	while ((fd = accept(sockfd, (struct sockaddr *) &cli_addr, &cli_len)) !=
	    -1) {
		if (net_addConnection(fd, cli_addr.sin_addr.s_addr)) {
			fprintf(stderr, "FICS is full.  fd = %d.\n", fd);
			psend_raw_file(fd, mess_dir, MESS_FULL);
			close(fd);
		} else {
			process_new_connection(fd, net_connected_host(fd));
		}
	}

	if (errno != EWOULDBLOCK) {
		fprintf(stderr, "FICS: Problem with accept().  errno=%d\n",
		    errno);
	}
	net_flush_all_connections();

	FD_ZERO(&readfds);
	for (loop = 0; loop < no_file; loop++) {
		if (con[loop].status != NETSTAT_EMPTY)
			FD_SET(con[loop].fd, &readfds);
	}

	to.tv_usec	= 0;
	to.tv_sec	= timeout;

	nfound = select(no_file, &readfds, NULL, NULL, &to);

	/* XXX: unused */
	(void) nfound;

	for (loop = 0; loop < no_file; loop++) {
		if (con[loop].status != NETSTAT_EMPTY) {
			fd = con[loop].fd;

			if ((lineComplete = readline2(com, fd)) == 0) {
				// partial line: do nothing
				continue;
			}

			if (lineComplete > 0) { // complete line: process it
#ifdef TIMESEAL
				if (!parseInput(com, &con[loop]))
					continue;
#endif
				if (process_input(fd, com) != COM_LOGOUT) {
					net_flush_connection(fd);
					continue;
				}
			}

			/*
			 * Disconnect anyone who gets here
			 */
			process_disconnection(fd);
			net_close_connection(fd);
		}
	}
}

PUBLIC int
net_consize(void)
{
	int total = sizeof con;

	for (int i = 0; i < no_file; i++)
		total += con[i].sndbufsize;
	return total;
}