1 /* netcat.c - Forward stdin/stdout to a file or network connection.
2 *
3 * Copyright 2007 Rob Landley <rob@landley.net>
4 *
5 * TODO: udp, ipv6, genericize for telnet/microcom/tail-f
6
7 USE_NETCAT(OLDTOY(nc, netcat, TOYFLAG_USR|TOYFLAG_BIN))
8 USE_NETCAT(NEWTOY(netcat, USE_NETCAT_LISTEN("^tlL")"w#p#s:q#f:", TOYFLAG_BIN))
9
10 config NETCAT
11 bool "netcat"
12 default y
13 help
14 usage: netcat [-u] [-wpq #] [-s addr] {IPADDR PORTNUM|-f FILENAME}
15
16 -f use FILENAME (ala /dev/ttyS0) instead of network
17 -p local port number
18 -q SECONDS quit this many seconds after EOF on stdin.
19 -s local ipv4 address
20 -w SECONDS timeout for connection
21
22 Use "stty 115200 -F /dev/ttyS0 && stty raw -echo -ctlecho" with
23 netcat -f to connect to a serial port.
24
25 config NETCAT_LISTEN
26 bool "netcat server options (-let)"
27 default y
28 depends on NETCAT
29 depends on TOYBOX_FORK
30 help
31 usage: netcat [-lL COMMAND...]
32
33 -l listen for one incoming connection.
34 -L listen for multiple incoming connections (server mode).
35
36 The command line after -l or -L is executed to handle each incoming
37 connection. If none, the connection is forwarded to stdin/stdout.
38
39 For a quick-and-dirty server, try something like:
40 netcat -s 127.0.0.1 -p 1234 -tL /bin/bash -l
41
42 config NETCAT_LISTEN_TTY
43 bool
44 default y
45 depends on NETCAT_LISTEN
46 depends on TOYBOX_FORK
47 help
48 usage: netcat [-t]
49
50 -t allocate tty (must come before -l or -L)
51 */
52
53 #define FOR_netcat
54 #include "toys.h"
55
GLOBALS(char * filename;long quit_delay;char * source_address;long port;long wait;)56 GLOBALS(
57 char *filename; // -f read from filename instead of network
58 long quit_delay; // -q Exit after EOF from stdin after # seconds.
59 char *source_address; // -s Bind to a specific source address.
60 long port; // -p Bind to a specific source port.
61 long wait; // -w Wait # seconds for a connection.
62 )
63
64 static void timeout(int signum)
65 {
66 if (TT.wait) error_exit("Timeout");
67 // This should be xexit() but would need siglongjmp()...
68 exit(0);
69 }
70
set_alarm(int seconds)71 static void set_alarm(int seconds)
72 {
73 xsignal(SIGALRM, seconds ? timeout : SIG_DFL);
74 alarm(seconds);
75 }
76
77 // Translate x.x.x.x numeric IPv4 address, or else DNS lookup an IPv4 name.
lookup_name(char * name,uint32_t * result)78 static void lookup_name(char *name, uint32_t *result)
79 {
80 struct hostent *hostbyname;
81
82 hostbyname = gethostbyname(name); // getaddrinfo
83 if (!hostbyname) error_exit("no host '%s'", name);
84 *result = *(uint32_t *)*hostbyname->h_addr_list;
85 }
86
87 // Worry about a fancy lookup later.
lookup_port(char * str,uint16_t * port)88 static void lookup_port(char *str, uint16_t *port)
89 {
90 *port = SWAP_BE16(atoi(str));
91 }
92
netcat_main(void)93 void netcat_main(void)
94 {
95 int sockfd=-1, pollcount=2;
96 struct pollfd pollfds[2];
97
98 memset(pollfds, 0, 2*sizeof(struct pollfd));
99 pollfds[0].events = pollfds[1].events = POLLIN;
100 set_alarm(TT.wait);
101
102 // The argument parsing logic can't make "<2" conditional on other
103 // arguments like -f and -l, so we do it by hand here.
104 if ((toys.optflags&FLAG_f) ? toys.optc :
105 (!(toys.optflags&(FLAG_l|FLAG_L)) && toys.optc!=2))
106 help_exit("Argument count wrong");
107
108 if (TT.filename) pollfds[0].fd = xopen(TT.filename, O_RDWR);
109 else {
110 int temp;
111 struct sockaddr_in address;
112
113 // Setup socket
114 sockfd = xsocket(AF_INET, SOCK_STREAM, 0);
115 fcntl(sockfd, F_SETFD, FD_CLOEXEC);
116 temp = 1;
117 setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &temp, sizeof(temp));
118 memset(&address, 0, sizeof(address));
119 address.sin_family = AF_INET;
120 if (TT.source_address || TT.port) {
121 address.sin_port = SWAP_BE16(TT.port);
122 if (TT.source_address)
123 lookup_name(TT.source_address, (uint32_t *)&address.sin_addr);
124 if (bind(sockfd, (struct sockaddr *)&address, sizeof(address)))
125 perror_exit("bind");
126 }
127
128 // Dial out
129
130 if (!CFG_NETCAT_LISTEN || !(toys.optflags&(FLAG_L|FLAG_l))) {
131 // Figure out where to dial out to.
132 lookup_name(*toys.optargs, (uint32_t *)&address.sin_addr);
133 lookup_port(toys.optargs[1], &address.sin_port);
134 temp = connect(sockfd, (struct sockaddr *)&address, sizeof(address));
135 if (temp<0) perror_exit("connect");
136 pollfds[0].fd = sockfd;
137
138 // Listen for incoming connections
139
140 } else {
141 socklen_t len = sizeof(address);
142
143 if (listen(sockfd, 5)) error_exit("listen");
144 if (!TT.port) {
145 getsockname(sockfd, (struct sockaddr *)&address, &len);
146 printf("%d\n", SWAP_BE16(address.sin_port));
147 fflush(stdout);
148 }
149 // Do we need to return immediately because -l has arguments?
150
151 if ((toys.optflags & FLAG_l) && toys.optc) {
152 if (CFG_TOYBOX_FORK && xfork()) goto cleanup;
153 close(0);
154 close(1);
155 close(2);
156 }
157
158 for (;;) {
159 pid_t child = 0;
160
161 // For -l, call accept from the _new_ process.
162
163 pollfds[0].fd = accept(sockfd, (struct sockaddr *)&address, &len);
164 if (pollfds[0].fd<0) perror_exit("accept");
165
166 // Do we need a tty?
167
168 if (toys.optflags&FLAG_t)
169 child = forkpty(&(pollfds[1].fd), NULL, NULL, NULL);
170
171 // Do we need to fork and/or redirect for exec?
172
173 else {
174 if (toys.optflags&FLAG_L) {
175 toys.stacktop = 0;
176 child = vfork();
177 }
178 if (!child && toys.optc) {
179 int fd = pollfds[0].fd;
180
181 dup2(fd, 0);
182 dup2(fd, 1);
183 if (toys.optflags&FLAG_L) dup2(fd, 2);
184 if (fd>2) close(fd);
185 }
186 }
187
188 if (child<0) error_msg("Fork failed\n");
189 if (child<1) break;
190 close(pollfds[0].fd);
191 }
192 }
193 }
194
195 // We have a connection. Disarm timeout.
196 // (Does not play well with -L, but what _should_ that do?)
197 set_alarm(0);
198
199 if (CFG_NETCAT_LISTEN && ((toys.optflags&(FLAG_L|FLAG_l)) && toys.optc))
200 xexec(toys.optargs);
201
202 // Poll loop copying stdin->socket and socket->stdout.
203 for (;;) {
204 int i;
205
206 if (0>poll(pollfds, pollcount, -1)) perror_exit("poll");
207
208 for (i=0; i<pollcount; i++) {
209 if (pollfds[i].revents & POLLIN) {
210 int len = read(pollfds[i].fd, toybuf, sizeof(toybuf));
211 if (len<1) goto dohupnow;
212 xwrite(i ? pollfds[0].fd : 1, toybuf, len);
213 } else if (pollfds[i].revents & POLLHUP) {
214 dohupnow:
215 // Close half-connection. This is needed for things like
216 // "echo GET / | netcat landley.net 80"
217 if (i) {
218 shutdown(pollfds[0].fd, SHUT_WR);
219 pollcount--;
220 set_alarm(TT.quit_delay);
221 } else goto cleanup;
222 }
223 }
224 }
225 cleanup:
226 if (CFG_TOYBOX_FREE) {
227 close(pollfds[0].fd);
228 close(sockfd);
229 }
230 }
231