blob: 3f512d1cb56fe11be546ca88ea11102072697263 [file] [log] [blame]
Erik Andersene49d5ec2000-02-08 19:58:47 +00001/* vi: set sw=4 ts=4: */
Erik Andersen7ab9c7e2000-05-12 19:41:47 +00002/* nc: mini-netcat - built from the ground up for LRP
Eric Andersencc8ed391999-10-05 16:24:54 +00003 Copyright (C) 1998 Charles P. Wright
4
5 0.0.1 6K It works.
6 0.0.2 5K Smaller and you can also check the exit condition if you wish.
Eric Andersen2ce1edc1999-10-12 15:42:48 +00007 0.0.3 Uses select()
Eric Andersencc8ed391999-10-05 16:24:54 +00008
9 19980918 Busy Boxed! Dave Cinege
Eric Andersen2ce1edc1999-10-12 15:42:48 +000010 19990512 Uses Select. Charles P. Wright
11 19990513 Fixes stdin stupidity and uses buffers. Charles P. Wright
Eric Andersencc8ed391999-10-05 16:24:54 +000012
13 This program is free software; you can redistribute it and/or modify
14 it under the terms of the GNU General Public License as published by
15 the Free Software Foundation; either version 2 of the License, or
16 (at your option) any later version.
17
18 This program is distributed in the hope that it will be useful,
19 but WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 GNU General Public License for more details.
22
23 You should have received a copy of the GNU General Public License
24 along with this program; if not, write to the Free Software
25 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
26
27*/
Eric Andersen67059862001-01-22 22:48:42 +000028
Eric Andersen3570a342000-09-25 21:45:58 +000029#include "busybox.h"
Eric Andersencc8ed391999-10-05 16:24:54 +000030#include <stdio.h>
31#include <stdlib.h>
32#include <unistd.h>
33
34#include <sys/types.h>
35#include <sys/socket.h>
36#include <netinet/in.h>
37#include <arpa/inet.h>
38#include <netdb.h>
39#include <sys/time.h>
40#include <sys/ioctl.h>
41
Erik Andersen7ab9c7e2000-05-12 19:41:47 +000042int nc_main(int argc, char **argv)
Eric Andersencc8ed391999-10-05 16:24:54 +000043{
Matt Kraai1d702672001-02-07 04:09:23 +000044 int do_listen = 0, lport = 0, tmpfd, opt, sfd;
Matt Kraaibfa79672000-12-15 22:34:34 +000045 char buf[BUFSIZ];
Eric Andersencc8ed391999-10-05 16:24:54 +000046
Erik Andersene49d5ec2000-02-08 19:58:47 +000047 struct sockaddr_in address;
48 struct hostent *hostinfo;
Eric Andersencc8ed391999-10-05 16:24:54 +000049
Erik Andersene49d5ec2000-02-08 19:58:47 +000050 fd_set readfds, testfds;
51
Matt Kraai1d702672001-02-07 04:09:23 +000052 while ((opt = getopt(argc, argv, "lp:")) > 0) {
53 switch (opt) {
54 case 'l':
55 do_listen++;
56 break;
57 case 'p':
58 lport = atoi(optarg);
59 break;
60 default:
61 usage(nc_usage);
62 }
Erik Andersen5e1189e2000-04-15 16:34:54 +000063 }
Eric Andersenb6a44b81999-11-13 04:47:09 +000064
Matt Kraai1d702672001-02-07 04:09:23 +000065 if ((do_listen && optind != argc) || (!do_listen && optind + 2 != argc))
66 usage(nc_usage);
67
Matt Kraaibfa79672000-12-15 22:34:34 +000068 if ((sfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
69 perror_msg_and_die("socket");
Eric Andersencc8ed391999-10-05 16:24:54 +000070
Erik Andersene49d5ec2000-02-08 19:58:47 +000071 address.sin_family = AF_INET;
Eric Andersencc8ed391999-10-05 16:24:54 +000072
Matt Kraai1d702672001-02-07 04:09:23 +000073 if (lport != 0) {
74 memset(&address.sin_addr, 0, sizeof(address.sin_addr));
75 address.sin_port = htons(lport);
76
77 if (bind(sfd, (struct sockaddr *) &address, sizeof(address)) < 0)
78 perror_msg_and_die("bind");
79 }
80
81 if (do_listen) {
82 if (listen(sfd, 1) < 0)
83 perror_msg_and_die("listen");
84
85 if ((tmpfd = accept(sfd, (struct sockaddr *) &address, &opt)) < 0)
86 perror_msg_and_die("accept");
87
88 close(sfd);
89 sfd = tmpfd;
90 } else {
91 if ((hostinfo = gethostbyname(argv[optind])) == NULL)
92 error_msg_and_die("cannot resolve %s\n", argv[optind]);
93
94 address.sin_addr = *(struct in_addr *) *hostinfo->h_addr_list;
95 address.sin_port = htons(atoi(argv[optind+1]));
96
97 if (connect(sfd, (struct sockaddr *) &address, sizeof(address)) < 0)
98 perror_msg_and_die("connect");
99 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000100
Erik Andersene49d5ec2000-02-08 19:58:47 +0000101 FD_ZERO(&readfds);
102 FD_SET(sfd, &readfds);
Matt Kraaibfa79672000-12-15 22:34:34 +0000103 FD_SET(STDIN_FILENO, &readfds);
Eric Andersencc8ed391999-10-05 16:24:54 +0000104
Erik Andersene49d5ec2000-02-08 19:58:47 +0000105 while (1) {
106 int fd;
Eric Andersen2ce1edc1999-10-12 15:42:48 +0000107 int ofd;
Erik Andersene49d5ec2000-02-08 19:58:47 +0000108 int nread;
Eric Andersencc8ed391999-10-05 16:24:54 +0000109
Erik Andersene49d5ec2000-02-08 19:58:47 +0000110 testfds = readfds;
Eric Andersencc8ed391999-10-05 16:24:54 +0000111
Matt Kraaibfa79672000-12-15 22:34:34 +0000112 if (select(FD_SETSIZE, &testfds, NULL, NULL, NULL) < 0)
113 perror_msg_and_die("select");
Eric Andersencc8ed391999-10-05 16:24:54 +0000114
Erik Andersene49d5ec2000-02-08 19:58:47 +0000115 for (fd = 0; fd < FD_SETSIZE; fd++) {
116 if (FD_ISSET(fd, &testfds)) {
Matt Kraaibfa79672000-12-15 22:34:34 +0000117 if ((nread = safe_read(fd, buf, sizeof(buf))) < 0)
118 perror_msg_and_die("read");
Eric Andersencc8ed391999-10-05 16:24:54 +0000119
Erik Andersene49d5ec2000-02-08 19:58:47 +0000120 if (fd == sfd) {
121 if (nread == 0)
122 exit(0);
Matt Kraaibfa79672000-12-15 22:34:34 +0000123 ofd = STDOUT_FILENO;
Erik Andersene49d5ec2000-02-08 19:58:47 +0000124 } else {
Matt Kraai95fa0ea2000-12-14 04:34:58 +0000125 if (nread == 0)
126 shutdown(sfd, 1);
Eric Andersen2ce1edc1999-10-12 15:42:48 +0000127 ofd = sfd;
128 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000129
Matt Kraaibfa79672000-12-15 22:34:34 +0000130 if (full_write(ofd, buf, nread) < 0)
131 perror_msg_and_die("write");
Eric Andersen2ce1edc1999-10-12 15:42:48 +0000132 }
Erik Andersene49d5ec2000-02-08 19:58:47 +0000133 }
134 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000135}