/* vi: set sw=4 ts=4: */ /* nc: mini-netcat - built from the ground up for LRP Copyright (C) 1998 Charles P. Wright 0.0.1 6K It works. 0.0.2 5K Smaller and you can also check the exit condition if you wish. 0.0.3 Uses select() 19980918 Busy Boxed! Dave Cinege 19990512 Uses Select. Charles P. Wright 19990513 Fixes stdin stupidity and uses buffers. Charles P. Wright Licensed under GPLv2 or later, see file LICENSE in this tarball for details. */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <signal.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <netdb.h> #include <sys/ioctl.h> #include "busybox.h" static void timeout(int signum) { bb_error_msg_and_die("Timed out"); } int nc_main(int argc, char **argv) { int do_listen = 0, lport = 0, delay = 0, wsecs = 0, tmpfd, opt, sfd, x; #ifdef CONFIG_NC_GAPING_SECURITY_HOLE char *pr00gie = NULL; #endif struct sockaddr_in address; struct hostent *hostinfo; fd_set readfds, testfds; while ((opt = getopt(argc, argv, "lp:i:e:w:")) > 0) { switch (opt) { case 'l': do_listen++; break; case 'p': lport = bb_lookup_port(optarg, "tcp", 0); break; case 'i': delay = atoi(optarg); break; #ifdef CONFIG_NC_GAPING_SECURITY_HOLE case 'e': pr00gie = optarg; break; #endif case 'w': wsecs = atoi(optarg); break; default: bb_show_usage(); } } if ((do_listen && optind != argc) || (!do_listen && optind + 2 != argc)) bb_show_usage(); sfd = bb_xsocket(AF_INET, SOCK_STREAM, 0); x = 1; if (setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &x, sizeof (x)) == -1) bb_perror_msg_and_die("reuseaddr"); address.sin_family = AF_INET; if (wsecs) { signal(SIGALRM, timeout); alarm(wsecs); } if (lport != 0) { memset(&address.sin_addr, 0, sizeof(address.sin_addr)); address.sin_port = lport; bb_xbind(sfd, (struct sockaddr *) &address, sizeof(address)); } if (do_listen) { socklen_t addrlen = sizeof(address); bb_xlisten(sfd, 1); if ((tmpfd = accept(sfd, (struct sockaddr *) &address, &addrlen)) < 0) bb_perror_msg_and_die("accept"); close(sfd); sfd = tmpfd; } else { hostinfo = xgethostbyname(argv[optind]); address.sin_addr = *(struct in_addr *) *hostinfo->h_addr_list; address.sin_port = bb_lookup_port(argv[optind+1], "tcp", 0); if (connect(sfd, (struct sockaddr *) &address, sizeof(address)) < 0) bb_perror_msg_and_die("connect"); } if (wsecs) { alarm(0); signal(SIGALRM, SIG_DFL); } #ifdef CONFIG_NC_GAPING_SECURITY_HOLE /* -e given? */ if (pr00gie) { dup2(sfd, 0); close(sfd); dup2(0, 1); dup2(0, 2); execl(pr00gie, pr00gie, NULL); /* Don't print stuff or it will go over the wire.... */ _exit(-1); } #endif /* CONFIG_NC_GAPING_SECURITY_HOLE */ FD_ZERO(&readfds); FD_SET(sfd, &readfds); FD_SET(STDIN_FILENO, &readfds); while (1) { int fd; int ofd; int nread; testfds = readfds; if (select(FD_SETSIZE, &testfds, NULL, NULL, NULL) < 0) bb_perror_msg_and_die("select"); for (fd = 0; fd < FD_SETSIZE; fd++) { if (FD_ISSET(fd, &testfds)) { if ((nread = safe_read(fd, bb_common_bufsiz1, sizeof(bb_common_bufsiz1))) < 0) { bb_perror_msg_and_die(bb_msg_read_error); } if (fd == sfd) { if (nread == 0) exit(0); ofd = STDOUT_FILENO; } else { if (nread <= 0) { shutdown(sfd, 1 /* send */ ); close(STDIN_FILENO); FD_CLR(STDIN_FILENO, &readfds); } ofd = sfd; } if (bb_full_write(ofd, bb_common_bufsiz1, nread) < 0) bb_perror_msg_and_die(bb_msg_write_error); if (delay > 0) { sleep(delay); } } } } }