1 |
gbeauche |
1.1 |
/* |
2 |
|
|
* Copyright (c) 1982, 1986, 1988, 1990, 1993 |
3 |
|
|
* The Regents of the University of California. All rights reserved. |
4 |
|
|
* |
5 |
|
|
* Redistribution and use in source and binary forms, with or without |
6 |
|
|
* modification, are permitted provided that the following conditions |
7 |
|
|
* are met: |
8 |
|
|
* 1. Redistributions of source code must retain the above copyright |
9 |
|
|
* notice, this list of conditions and the following disclaimer. |
10 |
|
|
* 2. Redistributions in binary form must reproduce the above copyright |
11 |
|
|
* notice, this list of conditions and the following disclaimer in the |
12 |
|
|
* documentation and/or other materials provided with the distribution. |
13 |
|
|
* 3. All advertising materials mentioning features or use of this software |
14 |
|
|
* must display the following acknowledgement: |
15 |
|
|
* This product includes software developed by the University of |
16 |
|
|
* California, Berkeley and its contributors. |
17 |
|
|
* 4. Neither the name of the University nor the names of its contributors |
18 |
|
|
* may be used to endorse or promote products derived from this software |
19 |
|
|
* without specific prior written permission. |
20 |
|
|
* |
21 |
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND |
22 |
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
23 |
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
24 |
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE |
25 |
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
26 |
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS |
27 |
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) |
28 |
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
29 |
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY |
30 |
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF |
31 |
|
|
* SUCH DAMAGE. |
32 |
|
|
* |
33 |
|
|
* @(#)tcp_subr.c 8.1 (Berkeley) 6/10/93 |
34 |
|
|
* tcp_subr.c,v 1.5 1994/10/08 22:39:58 phk Exp |
35 |
|
|
*/ |
36 |
|
|
|
37 |
|
|
/* |
38 |
|
|
* Changes and additions relating to SLiRP |
39 |
|
|
* Copyright (c) 1995 Danny Gasparovski. |
40 |
|
|
* |
41 |
|
|
* Please read the file COPYRIGHT for the |
42 |
|
|
* terms and conditions of the copyright. |
43 |
|
|
*/ |
44 |
|
|
|
45 |
|
|
#define WANT_SYS_IOCTL_H |
46 |
asvitkine |
1.3 |
#include <stdlib.h> |
47 |
gbeauche |
1.1 |
#include <slirp.h> |
48 |
|
|
|
49 |
|
|
/* patchable/settable parameters for tcp */ |
50 |
|
|
int tcp_mssdflt = TCP_MSS; |
51 |
|
|
int tcp_rttdflt = TCPTV_SRTTDFLT / PR_SLOWHZ; |
52 |
|
|
int tcp_do_rfc1323 = 0; /* Don't do rfc1323 performance enhancements */ |
53 |
|
|
int tcp_rcvspace; /* You may want to change this */ |
54 |
|
|
int tcp_sndspace; /* Keep small if you have an error prone link */ |
55 |
|
|
|
56 |
|
|
/* |
57 |
|
|
* Tcp initialization |
58 |
|
|
*/ |
59 |
|
|
void |
60 |
|
|
tcp_init() |
61 |
|
|
{ |
62 |
|
|
tcp_iss = 1; /* wrong */ |
63 |
|
|
tcb.so_next = tcb.so_prev = &tcb; |
64 |
|
|
|
65 |
|
|
/* tcp_rcvspace = our Window we advertise to the remote */ |
66 |
|
|
tcp_rcvspace = TCP_RCVSPACE; |
67 |
|
|
tcp_sndspace = TCP_SNDSPACE; |
68 |
|
|
|
69 |
|
|
/* Make sure tcp_sndspace is at least 2*MSS */ |
70 |
|
|
if (tcp_sndspace < 2*(min(if_mtu, if_mru) - sizeof(struct tcpiphdr))) |
71 |
|
|
tcp_sndspace = 2*(min(if_mtu, if_mru) - sizeof(struct tcpiphdr)); |
72 |
|
|
} |
73 |
|
|
|
74 |
|
|
/* |
75 |
|
|
* Create template to be used to send tcp packets on a connection. |
76 |
|
|
* Call after host entry created, fills |
77 |
|
|
* in a skeletal tcp/ip header, minimizing the amount of work |
78 |
|
|
* necessary when the connection is used. |
79 |
|
|
*/ |
80 |
|
|
/* struct tcpiphdr * */ |
81 |
|
|
void |
82 |
|
|
tcp_template(tp) |
83 |
|
|
struct tcpcb *tp; |
84 |
|
|
{ |
85 |
|
|
struct socket *so = tp->t_socket; |
86 |
|
|
register struct tcpiphdr *n = &tp->t_template; |
87 |
|
|
|
88 |
|
|
n->ti_next = n->ti_prev = 0; |
89 |
|
|
n->ti_x1 = 0; |
90 |
|
|
n->ti_pr = IPPROTO_TCP; |
91 |
|
|
n->ti_len = htons(sizeof (struct tcpiphdr) - sizeof (struct ip)); |
92 |
|
|
n->ti_src = so->so_faddr; |
93 |
|
|
n->ti_dst = so->so_laddr; |
94 |
|
|
n->ti_sport = so->so_fport; |
95 |
|
|
n->ti_dport = so->so_lport; |
96 |
|
|
|
97 |
|
|
n->ti_seq = 0; |
98 |
|
|
n->ti_ack = 0; |
99 |
|
|
n->ti_x2 = 0; |
100 |
|
|
n->ti_off = 5; |
101 |
|
|
n->ti_flags = 0; |
102 |
|
|
n->ti_win = 0; |
103 |
|
|
n->ti_sum = 0; |
104 |
|
|
n->ti_urp = 0; |
105 |
|
|
} |
106 |
|
|
|
107 |
|
|
/* |
108 |
|
|
* Send a single message to the TCP at address specified by |
109 |
|
|
* the given TCP/IP header. If m == 0, then we make a copy |
110 |
|
|
* of the tcpiphdr at ti and send directly to the addressed host. |
111 |
|
|
* This is used to force keep alive messages out using the TCP |
112 |
|
|
* template for a connection tp->t_template. If flags are given |
113 |
|
|
* then we send a message back to the TCP which originated the |
114 |
|
|
* segment ti, and discard the mbuf containing it and any other |
115 |
|
|
* attached mbufs. |
116 |
|
|
* |
117 |
|
|
* In any case the ack and sequence number of the transmitted |
118 |
|
|
* segment are as specified by the parameters. |
119 |
|
|
*/ |
120 |
|
|
void |
121 |
|
|
tcp_respond(tp, ti, m, ack, seq, flags) |
122 |
|
|
struct tcpcb *tp; |
123 |
|
|
register struct tcpiphdr *ti; |
124 |
|
|
register struct mbuf *m; |
125 |
|
|
tcp_seq ack, seq; |
126 |
|
|
int flags; |
127 |
|
|
{ |
128 |
|
|
register int tlen; |
129 |
|
|
int win = 0; |
130 |
|
|
|
131 |
|
|
DEBUG_CALL("tcp_respond"); |
132 |
|
|
DEBUG_ARG("tp = %lx", (long)tp); |
133 |
|
|
DEBUG_ARG("ti = %lx", (long)ti); |
134 |
|
|
DEBUG_ARG("m = %lx", (long)m); |
135 |
|
|
DEBUG_ARG("ack = %u", ack); |
136 |
|
|
DEBUG_ARG("seq = %u", seq); |
137 |
|
|
DEBUG_ARG("flags = %x", flags); |
138 |
|
|
|
139 |
|
|
if (tp) |
140 |
|
|
win = sbspace(&tp->t_socket->so_rcv); |
141 |
|
|
if (m == 0) { |
142 |
|
|
if ((m = m_get()) == NULL) |
143 |
|
|
return; |
144 |
|
|
#ifdef TCP_COMPAT_42 |
145 |
|
|
tlen = 1; |
146 |
|
|
#else |
147 |
|
|
tlen = 0; |
148 |
|
|
#endif |
149 |
|
|
m->m_data += if_maxlinkhdr; |
150 |
|
|
*mtod(m, struct tcpiphdr *) = *ti; |
151 |
|
|
ti = mtod(m, struct tcpiphdr *); |
152 |
|
|
flags = TH_ACK; |
153 |
|
|
} else { |
154 |
|
|
/* |
155 |
|
|
* ti points into m so the next line is just making |
156 |
|
|
* the mbuf point to ti |
157 |
|
|
*/ |
158 |
|
|
m->m_data = (caddr_t)ti; |
159 |
|
|
|
160 |
|
|
m->m_len = sizeof (struct tcpiphdr); |
161 |
|
|
tlen = 0; |
162 |
|
|
#define xchg(a,b,type) { type t; t=a; a=b; b=t; } |
163 |
|
|
xchg(ti->ti_dst.s_addr, ti->ti_src.s_addr, u_int32_t); |
164 |
|
|
xchg(ti->ti_dport, ti->ti_sport, u_int16_t); |
165 |
|
|
#undef xchg |
166 |
|
|
} |
167 |
|
|
ti->ti_len = htons((u_short)(sizeof (struct tcphdr) + tlen)); |
168 |
|
|
tlen += sizeof (struct tcpiphdr); |
169 |
|
|
m->m_len = tlen; |
170 |
|
|
|
171 |
|
|
ti->ti_next = ti->ti_prev = 0; |
172 |
|
|
ti->ti_x1 = 0; |
173 |
|
|
ti->ti_seq = htonl(seq); |
174 |
|
|
ti->ti_ack = htonl(ack); |
175 |
|
|
ti->ti_x2 = 0; |
176 |
|
|
ti->ti_off = sizeof (struct tcphdr) >> 2; |
177 |
|
|
ti->ti_flags = flags; |
178 |
|
|
if (tp) |
179 |
|
|
ti->ti_win = htons((u_int16_t) (win >> tp->rcv_scale)); |
180 |
|
|
else |
181 |
|
|
ti->ti_win = htons((u_int16_t)win); |
182 |
|
|
ti->ti_urp = 0; |
183 |
|
|
ti->ti_sum = 0; |
184 |
|
|
ti->ti_sum = cksum(m, tlen); |
185 |
|
|
((struct ip *)ti)->ip_len = tlen; |
186 |
|
|
|
187 |
|
|
if(flags & TH_RST) |
188 |
|
|
((struct ip *)ti)->ip_ttl = MAXTTL; |
189 |
|
|
else |
190 |
|
|
((struct ip *)ti)->ip_ttl = ip_defttl; |
191 |
|
|
|
192 |
|
|
(void) ip_output((struct socket *)0, m); |
193 |
|
|
} |
194 |
|
|
|
195 |
|
|
/* |
196 |
|
|
* Create a new TCP control block, making an |
197 |
|
|
* empty reassembly queue and hooking it to the argument |
198 |
|
|
* protocol control block. |
199 |
|
|
*/ |
200 |
|
|
struct tcpcb * |
201 |
|
|
tcp_newtcpcb(so) |
202 |
|
|
struct socket *so; |
203 |
|
|
{ |
204 |
|
|
register struct tcpcb *tp; |
205 |
|
|
|
206 |
|
|
tp = (struct tcpcb *)malloc(sizeof(*tp)); |
207 |
|
|
if (tp == NULL) |
208 |
|
|
return ((struct tcpcb *)0); |
209 |
|
|
|
210 |
|
|
memset((char *) tp, 0, sizeof(struct tcpcb)); |
211 |
|
|
tp->seg_next = tp->seg_prev = (tcpiphdrp_32)tp; |
212 |
|
|
tp->t_maxseg = tcp_mssdflt; |
213 |
|
|
|
214 |
|
|
tp->t_flags = tcp_do_rfc1323 ? (TF_REQ_SCALE|TF_REQ_TSTMP) : 0; |
215 |
|
|
tp->t_socket = so; |
216 |
|
|
|
217 |
|
|
/* |
218 |
|
|
* Init srtt to TCPTV_SRTTBASE (0), so we can tell that we have no |
219 |
|
|
* rtt estimate. Set rttvar so that srtt + 2 * rttvar gives |
220 |
|
|
* reasonable initial retransmit time. |
221 |
|
|
*/ |
222 |
|
|
tp->t_srtt = TCPTV_SRTTBASE; |
223 |
|
|
tp->t_rttvar = tcp_rttdflt * PR_SLOWHZ << 2; |
224 |
|
|
tp->t_rttmin = TCPTV_MIN; |
225 |
|
|
|
226 |
|
|
TCPT_RANGESET(tp->t_rxtcur, |
227 |
|
|
((TCPTV_SRTTBASE >> 2) + (TCPTV_SRTTDFLT << 2)) >> 1, |
228 |
|
|
TCPTV_MIN, TCPTV_REXMTMAX); |
229 |
|
|
|
230 |
|
|
tp->snd_cwnd = TCP_MAXWIN << TCP_MAX_WINSHIFT; |
231 |
|
|
tp->snd_ssthresh = TCP_MAXWIN << TCP_MAX_WINSHIFT; |
232 |
|
|
tp->t_state = TCPS_CLOSED; |
233 |
|
|
|
234 |
|
|
so->so_tcpcb = tp; |
235 |
|
|
|
236 |
|
|
return (tp); |
237 |
|
|
} |
238 |
|
|
|
239 |
|
|
/* |
240 |
|
|
* Drop a TCP connection, reporting |
241 |
|
|
* the specified error. If connection is synchronized, |
242 |
|
|
* then send a RST to peer. |
243 |
|
|
*/ |
244 |
|
|
struct tcpcb *tcp_drop(struct tcpcb *tp, int err) |
245 |
|
|
{ |
246 |
|
|
/* tcp_drop(tp, errno) |
247 |
|
|
register struct tcpcb *tp; |
248 |
|
|
int errno; |
249 |
|
|
{ |
250 |
|
|
*/ |
251 |
|
|
|
252 |
|
|
DEBUG_CALL("tcp_drop"); |
253 |
|
|
DEBUG_ARG("tp = %lx", (long)tp); |
254 |
|
|
DEBUG_ARG("errno = %d", errno); |
255 |
|
|
|
256 |
|
|
if (TCPS_HAVERCVDSYN(tp->t_state)) { |
257 |
|
|
tp->t_state = TCPS_CLOSED; |
258 |
|
|
(void) tcp_output(tp); |
259 |
|
|
tcpstat.tcps_drops++; |
260 |
|
|
} else |
261 |
|
|
tcpstat.tcps_conndrops++; |
262 |
|
|
/* if (errno == ETIMEDOUT && tp->t_softerror) |
263 |
|
|
* errno = tp->t_softerror; |
264 |
|
|
*/ |
265 |
|
|
/* so->so_error = errno; */ |
266 |
|
|
return (tcp_close(tp)); |
267 |
|
|
} |
268 |
|
|
|
269 |
|
|
/* |
270 |
|
|
* Close a TCP control block: |
271 |
|
|
* discard all space held by the tcp |
272 |
|
|
* discard internet protocol block |
273 |
|
|
* wake up any sleepers |
274 |
|
|
*/ |
275 |
|
|
struct tcpcb * |
276 |
|
|
tcp_close(tp) |
277 |
|
|
register struct tcpcb *tp; |
278 |
|
|
{ |
279 |
|
|
register struct tcpiphdr *t; |
280 |
|
|
struct socket *so = tp->t_socket; |
281 |
|
|
register struct mbuf *m; |
282 |
|
|
|
283 |
|
|
DEBUG_CALL("tcp_close"); |
284 |
|
|
DEBUG_ARG("tp = %lx", (long )tp); |
285 |
|
|
|
286 |
|
|
/* free the reassembly queue, if any */ |
287 |
|
|
t = (struct tcpiphdr *) tp->seg_next; |
288 |
|
|
while (t != (struct tcpiphdr *)tp) { |
289 |
|
|
t = (struct tcpiphdr *)t->ti_next; |
290 |
|
|
m = (struct mbuf *) REASS_MBUF((struct tcpiphdr *)t->ti_prev); |
291 |
|
|
remque_32((struct tcpiphdr *) t->ti_prev); |
292 |
|
|
m_freem(m); |
293 |
|
|
} |
294 |
|
|
/* It's static */ |
295 |
|
|
/* if (tp->t_template) |
296 |
|
|
* (void) m_free(dtom(tp->t_template)); |
297 |
|
|
*/ |
298 |
|
|
/* free(tp, M_PCB); */ |
299 |
|
|
free(tp); |
300 |
|
|
so->so_tcpcb = 0; |
301 |
|
|
soisfdisconnected(so); |
302 |
|
|
/* clobber input socket cache if we're closing the cached connection */ |
303 |
|
|
if (so == tcp_last_so) |
304 |
|
|
tcp_last_so = &tcb; |
305 |
|
|
closesocket(so->s); |
306 |
|
|
sbfree(&so->so_rcv); |
307 |
|
|
sbfree(&so->so_snd); |
308 |
|
|
sofree(so); |
309 |
|
|
tcpstat.tcps_closed++; |
310 |
|
|
return ((struct tcpcb *)0); |
311 |
|
|
} |
312 |
|
|
|
313 |
|
|
void |
314 |
|
|
tcp_drain() |
315 |
|
|
{ |
316 |
|
|
/* XXX */ |
317 |
|
|
} |
318 |
|
|
|
319 |
|
|
/* |
320 |
|
|
* When a source quench is received, close congestion window |
321 |
|
|
* to one segment. We will gradually open it again as we proceed. |
322 |
|
|
*/ |
323 |
|
|
|
324 |
|
|
#ifdef notdef |
325 |
|
|
|
326 |
|
|
void |
327 |
|
|
tcp_quench(i, errno) |
328 |
|
|
|
329 |
|
|
int errno; |
330 |
|
|
{ |
331 |
|
|
struct tcpcb *tp = intotcpcb(inp); |
332 |
|
|
|
333 |
|
|
if (tp) |
334 |
|
|
tp->snd_cwnd = tp->t_maxseg; |
335 |
|
|
} |
336 |
|
|
|
337 |
|
|
#endif /* notdef */ |
338 |
|
|
|
339 |
|
|
/* |
340 |
|
|
* TCP protocol interface to socket abstraction. |
341 |
|
|
*/ |
342 |
|
|
|
343 |
|
|
/* |
344 |
|
|
* User issued close, and wish to trail through shutdown states: |
345 |
|
|
* if never received SYN, just forget it. If got a SYN from peer, |
346 |
|
|
* but haven't sent FIN, then go to FIN_WAIT_1 state to send peer a FIN. |
347 |
|
|
* If already got a FIN from peer, then almost done; go to LAST_ACK |
348 |
|
|
* state. In all other cases, have already sent FIN to peer (e.g. |
349 |
|
|
* after PRU_SHUTDOWN), and just have to play tedious game waiting |
350 |
|
|
* for peer to send FIN or not respond to keep-alives, etc. |
351 |
|
|
* We can let the user exit from the close as soon as the FIN is acked. |
352 |
|
|
*/ |
353 |
|
|
void |
354 |
|
|
tcp_sockclosed(tp) |
355 |
|
|
struct tcpcb *tp; |
356 |
|
|
{ |
357 |
|
|
|
358 |
|
|
DEBUG_CALL("tcp_sockclosed"); |
359 |
|
|
DEBUG_ARG("tp = %lx", (long)tp); |
360 |
|
|
|
361 |
|
|
switch (tp->t_state) { |
362 |
|
|
|
363 |
|
|
case TCPS_CLOSED: |
364 |
|
|
case TCPS_LISTEN: |
365 |
|
|
case TCPS_SYN_SENT: |
366 |
|
|
tp->t_state = TCPS_CLOSED; |
367 |
|
|
tp = tcp_close(tp); |
368 |
|
|
break; |
369 |
|
|
|
370 |
|
|
case TCPS_SYN_RECEIVED: |
371 |
|
|
case TCPS_ESTABLISHED: |
372 |
|
|
tp->t_state = TCPS_FIN_WAIT_1; |
373 |
|
|
break; |
374 |
|
|
|
375 |
|
|
case TCPS_CLOSE_WAIT: |
376 |
|
|
tp->t_state = TCPS_LAST_ACK; |
377 |
|
|
break; |
378 |
|
|
} |
379 |
|
|
/* soisfdisconnecting(tp->t_socket); */ |
380 |
|
|
if (tp && tp->t_state >= TCPS_FIN_WAIT_2) |
381 |
|
|
soisfdisconnected(tp->t_socket); |
382 |
|
|
if (tp) |
383 |
|
|
tcp_output(tp); |
384 |
|
|
} |
385 |
|
|
|
386 |
|
|
/* |
387 |
|
|
* Connect to a host on the Internet |
388 |
|
|
* Called by tcp_input |
389 |
|
|
* Only do a connect, the tcp fields will be set in tcp_input |
390 |
|
|
* return 0 if there's a result of the connect, |
391 |
|
|
* else return -1 means we're still connecting |
392 |
|
|
* The return value is almost always -1 since the socket is |
393 |
|
|
* nonblocking. Connect returns after the SYN is sent, and does |
394 |
|
|
* not wait for ACK+SYN. |
395 |
|
|
*/ |
396 |
|
|
int tcp_fconnect(so) |
397 |
|
|
struct socket *so; |
398 |
|
|
{ |
399 |
|
|
int ret=0; |
400 |
|
|
|
401 |
|
|
DEBUG_CALL("tcp_fconnect"); |
402 |
|
|
DEBUG_ARG("so = %lx", (long )so); |
403 |
|
|
|
404 |
|
|
if( (ret=so->s=socket(AF_INET,SOCK_STREAM,0)) >= 0) { |
405 |
|
|
int opt, s=so->s; |
406 |
|
|
struct sockaddr_in addr; |
407 |
|
|
|
408 |
|
|
fd_nonblock(s); |
409 |
|
|
opt = 1; |
410 |
|
|
setsockopt(s,SOL_SOCKET,SO_REUSEADDR,(char *)&opt,sizeof(opt )); |
411 |
|
|
opt = 1; |
412 |
|
|
setsockopt(s,SOL_SOCKET,SO_OOBINLINE,(char *)&opt,sizeof(opt )); |
413 |
|
|
|
414 |
|
|
addr.sin_family = AF_INET; |
415 |
|
|
if ((so->so_faddr.s_addr & htonl(0xffffff00)) == special_addr.s_addr) { |
416 |
|
|
/* It's an alias */ |
417 |
|
|
switch(ntohl(so->so_faddr.s_addr) & 0xff) { |
418 |
|
|
case CTL_DNS: |
419 |
|
|
addr.sin_addr = dns_addr; |
420 |
|
|
break; |
421 |
|
|
case CTL_ALIAS: |
422 |
|
|
default: |
423 |
|
|
addr.sin_addr = loopback_addr; |
424 |
|
|
break; |
425 |
|
|
} |
426 |
|
|
} else |
427 |
|
|
addr.sin_addr = so->so_faddr; |
428 |
|
|
addr.sin_port = so->so_fport; |
429 |
|
|
|
430 |
|
|
DEBUG_MISC((dfd, " connect()ing, addr.sin_port=%d, " |
431 |
|
|
"addr.sin_addr.s_addr=%.16s\n", |
432 |
|
|
ntohs(addr.sin_port), inet_ntoa(addr.sin_addr))); |
433 |
|
|
/* We don't care what port we get */ |
434 |
|
|
ret = connect(s,(struct sockaddr *)&addr,sizeof (addr)); |
435 |
|
|
|
436 |
|
|
/* |
437 |
|
|
* If it's not in progress, it failed, so we just return 0, |
438 |
|
|
* without clearing SS_NOFDREF |
439 |
|
|
*/ |
440 |
|
|
soisfconnecting(so); |
441 |
|
|
} |
442 |
|
|
|
443 |
|
|
return(ret); |
444 |
|
|
} |
445 |
|
|
|
446 |
|
|
/* |
447 |
|
|
* Accept the socket and connect to the local-host |
448 |
|
|
* |
449 |
|
|
* We have a problem. The correct thing to do would be |
450 |
|
|
* to first connect to the local-host, and only if the |
451 |
|
|
* connection is accepted, then do an accept() here. |
452 |
|
|
* But, a) we need to know who's trying to connect |
453 |
|
|
* to the socket to be able to SYN the local-host, and |
454 |
|
|
* b) we are already connected to the foreign host by |
455 |
|
|
* the time it gets to accept(), so... We simply accept |
456 |
|
|
* here and SYN the local-host. |
457 |
|
|
*/ |
458 |
|
|
void |
459 |
|
|
tcp_connect(inso) |
460 |
|
|
struct socket *inso; |
461 |
|
|
{ |
462 |
|
|
struct socket *so; |
463 |
|
|
struct sockaddr_in addr; |
464 |
cebix |
1.2 |
socklen_t addrlen = sizeof(struct sockaddr_in); |
465 |
gbeauche |
1.1 |
struct tcpcb *tp; |
466 |
|
|
int s, opt; |
467 |
|
|
|
468 |
|
|
DEBUG_CALL("tcp_connect"); |
469 |
|
|
DEBUG_ARG("inso = %lx", (long)inso); |
470 |
|
|
|
471 |
|
|
/* |
472 |
|
|
* If it's an SS_ACCEPTONCE socket, no need to socreate() |
473 |
|
|
* another socket, just use the accept() socket. |
474 |
|
|
*/ |
475 |
|
|
if (inso->so_state & SS_FACCEPTONCE) { |
476 |
|
|
/* FACCEPTONCE already have a tcpcb */ |
477 |
|
|
so = inso; |
478 |
|
|
} else { |
479 |
|
|
if ((so = socreate()) == NULL) { |
480 |
|
|
/* If it failed, get rid of the pending connection */ |
481 |
|
|
closesocket(accept(inso->s,(struct sockaddr *)&addr,&addrlen)); |
482 |
|
|
return; |
483 |
|
|
} |
484 |
|
|
if (tcp_attach(so) < 0) { |
485 |
|
|
free(so); /* NOT sofree */ |
486 |
|
|
return; |
487 |
|
|
} |
488 |
|
|
so->so_laddr = inso->so_laddr; |
489 |
|
|
so->so_lport = inso->so_lport; |
490 |
|
|
} |
491 |
|
|
|
492 |
|
|
(void) tcp_mss(sototcpcb(so), 0); |
493 |
|
|
|
494 |
|
|
if ((s = accept(inso->s,(struct sockaddr *)&addr,&addrlen)) < 0) { |
495 |
|
|
tcp_close(sototcpcb(so)); /* This will sofree() as well */ |
496 |
|
|
return; |
497 |
|
|
} |
498 |
|
|
fd_nonblock(s); |
499 |
|
|
opt = 1; |
500 |
|
|
setsockopt(s,SOL_SOCKET,SO_REUSEADDR,(char *)&opt,sizeof(int)); |
501 |
|
|
opt = 1; |
502 |
|
|
setsockopt(s,SOL_SOCKET,SO_OOBINLINE,(char *)&opt,sizeof(int)); |
503 |
gbeauche |
1.5 |
opt = 1; |
504 |
|
|
setsockopt(s,IPPROTO_TCP,TCP_NODELAY,(char *)&opt,sizeof(int)); |
505 |
gbeauche |
1.1 |
|
506 |
|
|
so->so_fport = addr.sin_port; |
507 |
|
|
so->so_faddr = addr.sin_addr; |
508 |
|
|
/* Translate connections from localhost to the real hostname */ |
509 |
|
|
if (so->so_faddr.s_addr == 0 || so->so_faddr.s_addr == loopback_addr.s_addr) |
510 |
gbeauche |
1.4 |
so->so_faddr = alias_addr; |
511 |
gbeauche |
1.1 |
|
512 |
|
|
/* Close the accept() socket, set right state */ |
513 |
|
|
if (inso->so_state & SS_FACCEPTONCE) { |
514 |
|
|
closesocket(so->s); /* If we only accept once, close the accept() socket */ |
515 |
|
|
so->so_state = SS_NOFDREF; /* Don't select it yet, even though we have an FD */ |
516 |
|
|
/* if it's not FACCEPTONCE, it's already NOFDREF */ |
517 |
|
|
} |
518 |
|
|
so->s = s; |
519 |
|
|
|
520 |
|
|
so->so_iptos = tcp_tos(so); |
521 |
|
|
tp = sototcpcb(so); |
522 |
|
|
|
523 |
|
|
tcp_template(tp); |
524 |
|
|
|
525 |
|
|
/* Compute window scaling to request. */ |
526 |
|
|
/* while (tp->request_r_scale < TCP_MAX_WINSHIFT && |
527 |
|
|
* (TCP_MAXWIN << tp->request_r_scale) < so->so_rcv.sb_hiwat) |
528 |
|
|
* tp->request_r_scale++; |
529 |
|
|
*/ |
530 |
|
|
|
531 |
|
|
/* soisconnecting(so); */ /* NOFDREF used instead */ |
532 |
|
|
tcpstat.tcps_connattempt++; |
533 |
|
|
|
534 |
|
|
tp->t_state = TCPS_SYN_SENT; |
535 |
|
|
tp->t_timer[TCPT_KEEP] = TCPTV_KEEP_INIT; |
536 |
|
|
tp->iss = tcp_iss; |
537 |
|
|
tcp_iss += TCP_ISSINCR/2; |
538 |
|
|
tcp_sendseqinit(tp); |
539 |
|
|
tcp_output(tp); |
540 |
|
|
} |
541 |
|
|
|
542 |
|
|
/* |
543 |
|
|
* Attach a TCPCB to a socket. |
544 |
|
|
*/ |
545 |
|
|
int |
546 |
|
|
tcp_attach(so) |
547 |
|
|
struct socket *so; |
548 |
|
|
{ |
549 |
|
|
if ((so->so_tcpcb = tcp_newtcpcb(so)) == NULL) |
550 |
|
|
return -1; |
551 |
|
|
|
552 |
|
|
insque(so, &tcb); |
553 |
|
|
|
554 |
|
|
return 0; |
555 |
|
|
} |
556 |
|
|
|
557 |
|
|
/* |
558 |
|
|
* Set the socket's type of service field |
559 |
|
|
*/ |
560 |
|
|
struct tos_t tcptos[] = { |
561 |
|
|
{0, 20, IPTOS_THROUGHPUT, 0}, /* ftp data */ |
562 |
|
|
{21, 21, IPTOS_LOWDELAY, EMU_FTP}, /* ftp control */ |
563 |
|
|
{0, 23, IPTOS_LOWDELAY, 0}, /* telnet */ |
564 |
|
|
{0, 80, IPTOS_THROUGHPUT, 0}, /* WWW */ |
565 |
|
|
{0, 513, IPTOS_LOWDELAY, EMU_RLOGIN|EMU_NOCONNECT}, /* rlogin */ |
566 |
|
|
{0, 514, IPTOS_LOWDELAY, EMU_RSH|EMU_NOCONNECT}, /* shell */ |
567 |
|
|
{0, 544, IPTOS_LOWDELAY, EMU_KSH}, /* kshell */ |
568 |
|
|
{0, 543, IPTOS_LOWDELAY, 0}, /* klogin */ |
569 |
|
|
{0, 6667, IPTOS_THROUGHPUT, EMU_IRC}, /* IRC */ |
570 |
|
|
{0, 6668, IPTOS_THROUGHPUT, EMU_IRC}, /* IRC undernet */ |
571 |
|
|
{0, 7070, IPTOS_LOWDELAY, EMU_REALAUDIO }, /* RealAudio control */ |
572 |
|
|
{0, 113, IPTOS_LOWDELAY, EMU_IDENT }, /* identd protocol */ |
573 |
|
|
{0, 0, 0, 0} |
574 |
|
|
}; |
575 |
|
|
|
576 |
|
|
struct emu_t *tcpemu = 0; |
577 |
|
|
|
578 |
|
|
/* |
579 |
|
|
* Return TOS according to the above table |
580 |
|
|
*/ |
581 |
|
|
u_int8_t |
582 |
|
|
tcp_tos(so) |
583 |
|
|
struct socket *so; |
584 |
|
|
{ |
585 |
|
|
int i = 0; |
586 |
|
|
struct emu_t *emup; |
587 |
|
|
|
588 |
|
|
while(tcptos[i].tos) { |
589 |
|
|
if ((tcptos[i].fport && (ntohs(so->so_fport) == tcptos[i].fport)) || |
590 |
|
|
(tcptos[i].lport && (ntohs(so->so_lport) == tcptos[i].lport))) { |
591 |
|
|
so->so_emu = tcptos[i].emu; |
592 |
|
|
return tcptos[i].tos; |
593 |
|
|
} |
594 |
|
|
i++; |
595 |
|
|
} |
596 |
|
|
|
597 |
|
|
/* Nope, lets see if there's a user-added one */ |
598 |
|
|
for (emup = tcpemu; emup; emup = emup->next) { |
599 |
|
|
if ((emup->fport && (ntohs(so->so_fport) == emup->fport)) || |
600 |
|
|
(emup->lport && (ntohs(so->so_lport) == emup->lport))) { |
601 |
|
|
so->so_emu = emup->emu; |
602 |
|
|
return emup->tos; |
603 |
|
|
} |
604 |
|
|
} |
605 |
|
|
|
606 |
|
|
return 0; |
607 |
|
|
} |
608 |
|
|
|
609 |
|
|
int do_echo = -1; |
610 |
|
|
|
611 |
|
|
/* |
612 |
|
|
* Emulate programs that try and connect to us |
613 |
|
|
* This includes ftp (the data connection is |
614 |
|
|
* initiated by the server) and IRC (DCC CHAT and |
615 |
|
|
* DCC SEND) for now |
616 |
|
|
* |
617 |
|
|
* NOTE: It's possible to crash SLiRP by sending it |
618 |
|
|
* unstandard strings to emulate... if this is a problem, |
619 |
|
|
* more checks are needed here |
620 |
|
|
* |
621 |
|
|
* XXX Assumes the whole command came in one packet |
622 |
|
|
* |
623 |
|
|
* XXX Some ftp clients will have their TOS set to |
624 |
|
|
* LOWDELAY and so Nagel will kick in. Because of this, |
625 |
|
|
* we'll get the first letter, followed by the rest, so |
626 |
|
|
* we simply scan for ORT instead of PORT... |
627 |
|
|
* DCC doesn't have this problem because there's other stuff |
628 |
|
|
* in the packet before the DCC command. |
629 |
|
|
* |
630 |
|
|
* Return 1 if the mbuf m is still valid and should be |
631 |
|
|
* sbappend()ed |
632 |
|
|
* |
633 |
|
|
* NOTE: if you return 0 you MUST m_free() the mbuf! |
634 |
|
|
*/ |
635 |
|
|
int |
636 |
|
|
tcp_emu(so, m) |
637 |
|
|
struct socket *so; |
638 |
|
|
struct mbuf *m; |
639 |
|
|
{ |
640 |
|
|
u_int n1, n2, n3, n4, n5, n6; |
641 |
|
|
char buff[256]; |
642 |
|
|
u_int32_t laddr; |
643 |
|
|
u_int lport; |
644 |
|
|
char *bptr; |
645 |
|
|
|
646 |
|
|
DEBUG_CALL("tcp_emu"); |
647 |
|
|
DEBUG_ARG("so = %lx", (long)so); |
648 |
|
|
DEBUG_ARG("m = %lx", (long)m); |
649 |
|
|
|
650 |
|
|
switch(so->so_emu) { |
651 |
|
|
int x, i; |
652 |
|
|
|
653 |
|
|
case EMU_IDENT: |
654 |
|
|
/* |
655 |
|
|
* Identification protocol as per rfc-1413 |
656 |
|
|
*/ |
657 |
|
|
|
658 |
|
|
{ |
659 |
|
|
struct socket *tmpso; |
660 |
|
|
struct sockaddr_in addr; |
661 |
cebix |
1.2 |
socklen_t addrlen = sizeof(struct sockaddr_in); |
662 |
gbeauche |
1.1 |
struct sbuf *so_rcv = &so->so_rcv; |
663 |
|
|
|
664 |
|
|
memcpy(so_rcv->sb_wptr, m->m_data, m->m_len); |
665 |
|
|
so_rcv->sb_wptr += m->m_len; |
666 |
|
|
so_rcv->sb_rptr += m->m_len; |
667 |
|
|
m->m_data[m->m_len] = 0; /* NULL terminate */ |
668 |
|
|
if (strchr(m->m_data, '\r') || strchr(m->m_data, '\n')) { |
669 |
|
|
if (sscanf(so_rcv->sb_data, "%d%*[ ,]%d", &n1, &n2) == 2) { |
670 |
|
|
HTONS(n1); |
671 |
|
|
HTONS(n2); |
672 |
|
|
/* n2 is the one on our host */ |
673 |
|
|
for (tmpso = tcb.so_next; tmpso != &tcb; tmpso = tmpso->so_next) { |
674 |
|
|
if (tmpso->so_laddr.s_addr == so->so_laddr.s_addr && |
675 |
|
|
tmpso->so_lport == n2 && |
676 |
|
|
tmpso->so_faddr.s_addr == so->so_faddr.s_addr && |
677 |
|
|
tmpso->so_fport == n1) { |
678 |
|
|
if (getsockname(tmpso->s, |
679 |
|
|
(struct sockaddr *)&addr, &addrlen) == 0) |
680 |
|
|
n2 = ntohs(addr.sin_port); |
681 |
|
|
break; |
682 |
|
|
} |
683 |
|
|
} |
684 |
|
|
} |
685 |
|
|
so_rcv->sb_cc = sprintf(so_rcv->sb_data, "%d,%d\r\n", n1, n2); |
686 |
|
|
so_rcv->sb_rptr = so_rcv->sb_data; |
687 |
|
|
so_rcv->sb_wptr = so_rcv->sb_data + so_rcv->sb_cc; |
688 |
|
|
} |
689 |
|
|
m_free(m); |
690 |
|
|
return 0; |
691 |
|
|
} |
692 |
|
|
|
693 |
|
|
#if 0 |
694 |
|
|
case EMU_RLOGIN: |
695 |
|
|
/* |
696 |
|
|
* Rlogin emulation |
697 |
|
|
* First we accumulate all the initial option negotiation, |
698 |
|
|
* then fork_exec() rlogin according to the options |
699 |
|
|
*/ |
700 |
|
|
{ |
701 |
|
|
int i, i2, n; |
702 |
|
|
char *ptr; |
703 |
|
|
char args[100]; |
704 |
|
|
char term[100]; |
705 |
|
|
struct sbuf *so_snd = &so->so_snd; |
706 |
|
|
struct sbuf *so_rcv = &so->so_rcv; |
707 |
|
|
|
708 |
|
|
/* First check if they have a priveladged port, or too much data has arrived */ |
709 |
|
|
if (ntohs(so->so_lport) > 1023 || ntohs(so->so_lport) < 512 || |
710 |
|
|
(m->m_len + so_rcv->sb_wptr) > (so_rcv->sb_data + so_rcv->sb_datalen)) { |
711 |
|
|
memcpy(so_snd->sb_wptr, "Permission denied\n", 18); |
712 |
|
|
so_snd->sb_wptr += 18; |
713 |
|
|
so_snd->sb_cc += 18; |
714 |
|
|
tcp_sockclosed(sototcpcb(so)); |
715 |
|
|
m_free(m); |
716 |
|
|
return 0; |
717 |
|
|
} |
718 |
|
|
|
719 |
|
|
/* Append the current data */ |
720 |
|
|
memcpy(so_rcv->sb_wptr, m->m_data, m->m_len); |
721 |
|
|
so_rcv->sb_wptr += m->m_len; |
722 |
|
|
so_rcv->sb_rptr += m->m_len; |
723 |
|
|
m_free(m); |
724 |
|
|
|
725 |
|
|
/* |
726 |
|
|
* Check if we have all the initial options, |
727 |
|
|
* and build argument list to rlogin while we're here |
728 |
|
|
*/ |
729 |
|
|
n = 0; |
730 |
|
|
ptr = so_rcv->sb_data; |
731 |
|
|
args[0] = 0; |
732 |
|
|
term[0] = 0; |
733 |
|
|
while (ptr < so_rcv->sb_wptr) { |
734 |
|
|
if (*ptr++ == 0) { |
735 |
|
|
n++; |
736 |
|
|
if (n == 2) { |
737 |
|
|
sprintf(args, "rlogin -l %s %s", |
738 |
|
|
ptr, inet_ntoa(so->so_faddr)); |
739 |
|
|
} else if (n == 3) { |
740 |
|
|
i2 = so_rcv->sb_wptr - ptr; |
741 |
|
|
for (i = 0; i < i2; i++) { |
742 |
|
|
if (ptr[i] == '/') { |
743 |
|
|
ptr[i] = 0; |
744 |
|
|
#ifdef HAVE_SETENV |
745 |
|
|
sprintf(term, "%s", ptr); |
746 |
|
|
#else |
747 |
|
|
sprintf(term, "TERM=%s", ptr); |
748 |
|
|
#endif |
749 |
|
|
ptr[i] = '/'; |
750 |
|
|
break; |
751 |
|
|
} |
752 |
|
|
} |
753 |
|
|
} |
754 |
|
|
} |
755 |
|
|
} |
756 |
|
|
|
757 |
|
|
if (n != 4) |
758 |
|
|
return 0; |
759 |
|
|
|
760 |
|
|
/* We have it, set our term variable and fork_exec() */ |
761 |
|
|
#ifdef HAVE_SETENV |
762 |
|
|
setenv("TERM", term, 1); |
763 |
|
|
#else |
764 |
|
|
putenv(term); |
765 |
|
|
#endif |
766 |
|
|
fork_exec(so, args, 2); |
767 |
|
|
term[0] = 0; |
768 |
|
|
so->so_emu = 0; |
769 |
|
|
|
770 |
|
|
/* And finally, send the client a 0 character */ |
771 |
|
|
so_snd->sb_wptr[0] = 0; |
772 |
|
|
so_snd->sb_wptr++; |
773 |
|
|
so_snd->sb_cc++; |
774 |
|
|
|
775 |
|
|
return 0; |
776 |
|
|
} |
777 |
|
|
|
778 |
|
|
case EMU_RSH: |
779 |
|
|
/* |
780 |
|
|
* rsh emulation |
781 |
|
|
* First we accumulate all the initial option negotiation, |
782 |
|
|
* then rsh_exec() rsh according to the options |
783 |
|
|
*/ |
784 |
|
|
{ |
785 |
|
|
int n; |
786 |
|
|
char *ptr; |
787 |
|
|
char *user; |
788 |
|
|
char *args; |
789 |
|
|
struct sbuf *so_snd = &so->so_snd; |
790 |
|
|
struct sbuf *so_rcv = &so->so_rcv; |
791 |
|
|
|
792 |
|
|
/* First check if they have a priveladged port, or too much data has arrived */ |
793 |
|
|
if (ntohs(so->so_lport) > 1023 || ntohs(so->so_lport) < 512 || |
794 |
|
|
(m->m_len + so_rcv->sb_wptr) > (so_rcv->sb_data + so_rcv->sb_datalen)) { |
795 |
|
|
memcpy(so_snd->sb_wptr, "Permission denied\n", 18); |
796 |
|
|
so_snd->sb_wptr += 18; |
797 |
|
|
so_snd->sb_cc += 18; |
798 |
|
|
tcp_sockclosed(sototcpcb(so)); |
799 |
|
|
m_free(m); |
800 |
|
|
return 0; |
801 |
|
|
} |
802 |
|
|
|
803 |
|
|
/* Append the current data */ |
804 |
|
|
memcpy(so_rcv->sb_wptr, m->m_data, m->m_len); |
805 |
|
|
so_rcv->sb_wptr += m->m_len; |
806 |
|
|
so_rcv->sb_rptr += m->m_len; |
807 |
|
|
m_free(m); |
808 |
|
|
|
809 |
|
|
/* |
810 |
|
|
* Check if we have all the initial options, |
811 |
|
|
* and build argument list to rlogin while we're here |
812 |
|
|
*/ |
813 |
|
|
n = 0; |
814 |
|
|
ptr = so_rcv->sb_data; |
815 |
|
|
user=""; |
816 |
|
|
args=""; |
817 |
|
|
if (so->extra==NULL) { |
818 |
|
|
struct socket *ns; |
819 |
|
|
struct tcpcb* tp; |
820 |
|
|
int port=atoi(ptr); |
821 |
|
|
if (port <= 0) return 0; |
822 |
|
|
if (port > 1023 || port < 512) { |
823 |
|
|
memcpy(so_snd->sb_wptr, "Permission denied\n", 18); |
824 |
|
|
so_snd->sb_wptr += 18; |
825 |
|
|
so_snd->sb_cc += 18; |
826 |
|
|
tcp_sockclosed(sototcpcb(so)); |
827 |
|
|
return 0; |
828 |
|
|
} |
829 |
|
|
if ((ns=socreate()) == NULL) |
830 |
|
|
return 0; |
831 |
|
|
if (tcp_attach(ns)<0) { |
832 |
|
|
free(ns); |
833 |
|
|
return 0; |
834 |
|
|
} |
835 |
|
|
|
836 |
|
|
ns->so_laddr=so->so_laddr; |
837 |
|
|
ns->so_lport=htons(port); |
838 |
|
|
|
839 |
|
|
(void) tcp_mss(sototcpcb(ns), 0); |
840 |
|
|
|
841 |
|
|
ns->so_faddr=so->so_faddr; |
842 |
|
|
ns->so_fport=htons(IPPORT_RESERVED-1); /* Use a fake port. */ |
843 |
|
|
|
844 |
|
|
if (ns->so_faddr.s_addr == 0 || |
845 |
|
|
ns->so_faddr.s_addr == loopback_addr.s_addr) |
846 |
gbeauche |
1.4 |
ns->so_faddr = alias_addr; |
847 |
gbeauche |
1.1 |
|
848 |
|
|
ns->so_iptos = tcp_tos(ns); |
849 |
|
|
tp = sototcpcb(ns); |
850 |
|
|
|
851 |
|
|
tcp_template(tp); |
852 |
|
|
|
853 |
|
|
/* Compute window scaling to request. */ |
854 |
|
|
/* while (tp->request_r_scale < TCP_MAX_WINSHIFT && |
855 |
|
|
* (TCP_MAXWIN << tp->request_r_scale) < so->so_rcv.sb_hiwat) |
856 |
|
|
* tp->request_r_scale++; |
857 |
|
|
*/ |
858 |
|
|
|
859 |
|
|
/*soisfconnecting(ns);*/ |
860 |
|
|
|
861 |
|
|
tcpstat.tcps_connattempt++; |
862 |
|
|
|
863 |
|
|
tp->t_state = TCPS_SYN_SENT; |
864 |
|
|
tp->t_timer[TCPT_KEEP] = TCPTV_KEEP_INIT; |
865 |
|
|
tp->iss = tcp_iss; |
866 |
|
|
tcp_iss += TCP_ISSINCR/2; |
867 |
|
|
tcp_sendseqinit(tp); |
868 |
|
|
tcp_output(tp); |
869 |
|
|
so->extra=ns; |
870 |
|
|
} |
871 |
|
|
while (ptr < so_rcv->sb_wptr) { |
872 |
|
|
if (*ptr++ == 0) { |
873 |
|
|
n++; |
874 |
|
|
if (n == 2) { |
875 |
|
|
user=ptr; |
876 |
|
|
} else if (n == 3) { |
877 |
|
|
args=ptr; |
878 |
|
|
} |
879 |
|
|
} |
880 |
|
|
} |
881 |
|
|
|
882 |
|
|
if (n != 4) |
883 |
|
|
return 0; |
884 |
|
|
|
885 |
|
|
rsh_exec(so,so->extra, user, inet_ntoa(so->so_faddr), args); |
886 |
|
|
so->so_emu = 0; |
887 |
|
|
so->extra=NULL; |
888 |
|
|
|
889 |
|
|
/* And finally, send the client a 0 character */ |
890 |
|
|
so_snd->sb_wptr[0] = 0; |
891 |
|
|
so_snd->sb_wptr++; |
892 |
|
|
so_snd->sb_cc++; |
893 |
|
|
|
894 |
|
|
return 0; |
895 |
|
|
} |
896 |
|
|
|
897 |
|
|
case EMU_CTL: |
898 |
|
|
{ |
899 |
|
|
int num; |
900 |
|
|
struct sbuf *so_snd = &so->so_snd; |
901 |
|
|
struct sbuf *so_rcv = &so->so_rcv; |
902 |
|
|
|
903 |
|
|
/* |
904 |
|
|
* If there is binary data here, we save it in so->so_m |
905 |
|
|
*/ |
906 |
|
|
if (!so->so_m) { |
907 |
|
|
int rxlen; |
908 |
|
|
char *rxdata; |
909 |
|
|
rxdata=mtod(m, char *); |
910 |
|
|
for (rxlen=m->m_len; rxlen; rxlen--) { |
911 |
|
|
if (*rxdata++ & 0x80) { |
912 |
|
|
so->so_m = m; |
913 |
|
|
return 0; |
914 |
|
|
} |
915 |
|
|
} |
916 |
|
|
} /* if(so->so_m==NULL) */ |
917 |
|
|
|
918 |
|
|
/* |
919 |
|
|
* Append the line |
920 |
|
|
*/ |
921 |
|
|
sbappendsb(so_rcv, m); |
922 |
|
|
|
923 |
|
|
/* To avoid going over the edge of the buffer, we reset it */ |
924 |
|
|
if (so_snd->sb_cc == 0) |
925 |
|
|
so_snd->sb_wptr = so_snd->sb_rptr = so_snd->sb_data; |
926 |
|
|
|
927 |
|
|
/* |
928 |
|
|
* A bit of a hack: |
929 |
|
|
* If the first packet we get here is 1 byte long, then it |
930 |
|
|
* was done in telnet character mode, therefore we must echo |
931 |
|
|
* the characters as they come. Otherwise, we echo nothing, |
932 |
|
|
* because in linemode, the line is already echoed |
933 |
|
|
* XXX two or more control connections won't work |
934 |
|
|
*/ |
935 |
|
|
if (do_echo == -1) { |
936 |
|
|
if (m->m_len == 1) do_echo = 1; |
937 |
|
|
else do_echo = 0; |
938 |
|
|
} |
939 |
|
|
if (do_echo) { |
940 |
|
|
sbappendsb(so_snd, m); |
941 |
|
|
m_free(m); |
942 |
|
|
tcp_output(sototcpcb(so)); /* XXX */ |
943 |
|
|
} else |
944 |
|
|
m_free(m); |
945 |
|
|
|
946 |
|
|
num = 0; |
947 |
|
|
while (num < so->so_rcv.sb_cc) { |
948 |
|
|
if (*(so->so_rcv.sb_rptr + num) == '\n' || |
949 |
|
|
*(so->so_rcv.sb_rptr + num) == '\r') { |
950 |
|
|
int n; |
951 |
|
|
|
952 |
|
|
*(so_rcv->sb_rptr + num) = 0; |
953 |
|
|
if (ctl_password && !ctl_password_ok) { |
954 |
|
|
/* Need a password */ |
955 |
|
|
if (sscanf(so_rcv->sb_rptr, "pass %256s", buff) == 1) { |
956 |
|
|
if (strcmp(buff, ctl_password) == 0) { |
957 |
|
|
ctl_password_ok = 1; |
958 |
|
|
n = sprintf(so_snd->sb_wptr, |
959 |
|
|
"Password OK.\r\n"); |
960 |
|
|
goto do_prompt; |
961 |
|
|
} |
962 |
|
|
} |
963 |
|
|
n = sprintf(so_snd->sb_wptr, |
964 |
|
|
"Error: Password required, log on with \"pass PASSWORD\"\r\n"); |
965 |
|
|
goto do_prompt; |
966 |
|
|
} |
967 |
|
|
cfg_quitting = 0; |
968 |
|
|
n = do_config(so_rcv->sb_rptr, so, PRN_SPRINTF); |
969 |
|
|
if (!cfg_quitting) { |
970 |
|
|
/* Register the printed data */ |
971 |
|
|
do_prompt: |
972 |
|
|
so_snd->sb_cc += n; |
973 |
|
|
so_snd->sb_wptr += n; |
974 |
|
|
/* Add prompt */ |
975 |
|
|
n = sprintf(so_snd->sb_wptr, "Slirp> "); |
976 |
|
|
so_snd->sb_cc += n; |
977 |
|
|
so_snd->sb_wptr += n; |
978 |
|
|
} |
979 |
|
|
/* Drop so_rcv data */ |
980 |
|
|
so_rcv->sb_cc = 0; |
981 |
|
|
so_rcv->sb_wptr = so_rcv->sb_rptr = so_rcv->sb_data; |
982 |
|
|
tcp_output(sototcpcb(so)); /* Send the reply */ |
983 |
|
|
} |
984 |
|
|
num++; |
985 |
|
|
} |
986 |
|
|
return 0; |
987 |
|
|
} |
988 |
|
|
#endif |
989 |
|
|
case EMU_FTP: /* ftp */ |
990 |
|
|
*(m->m_data+m->m_len) = 0; /* NULL terminate for strstr */ |
991 |
|
|
if ((bptr = (char *)strstr(m->m_data, "ORT")) != NULL) { |
992 |
|
|
/* |
993 |
|
|
* Need to emulate the PORT command |
994 |
|
|
*/ |
995 |
|
|
x = sscanf(bptr, "ORT %d,%d,%d,%d,%d,%d\r\n%256[^\177]", |
996 |
|
|
&n1, &n2, &n3, &n4, &n5, &n6, buff); |
997 |
|
|
if (x < 6) |
998 |
|
|
return 1; |
999 |
|
|
|
1000 |
|
|
laddr = htonl((n1 << 24) | (n2 << 16) | (n3 << 8) | (n4)); |
1001 |
|
|
lport = htons((n5 << 8) | (n6)); |
1002 |
|
|
|
1003 |
|
|
if ((so = solisten(0, laddr, lport, SS_FACCEPTONCE)) == NULL) |
1004 |
|
|
return 1; |
1005 |
|
|
|
1006 |
|
|
n6 = ntohs(so->so_fport); |
1007 |
|
|
|
1008 |
|
|
n5 = (n6 >> 8) & 0xff; |
1009 |
|
|
n6 &= 0xff; |
1010 |
|
|
|
1011 |
|
|
laddr = ntohl(so->so_faddr.s_addr); |
1012 |
|
|
|
1013 |
|
|
n1 = ((laddr >> 24) & 0xff); |
1014 |
|
|
n2 = ((laddr >> 16) & 0xff); |
1015 |
|
|
n3 = ((laddr >> 8) & 0xff); |
1016 |
|
|
n4 = (laddr & 0xff); |
1017 |
|
|
|
1018 |
|
|
m->m_len = bptr - m->m_data; /* Adjust length */ |
1019 |
|
|
m->m_len += sprintf(bptr,"ORT %d,%d,%d,%d,%d,%d\r\n%s", |
1020 |
|
|
n1, n2, n3, n4, n5, n6, x==7?buff:""); |
1021 |
|
|
return 1; |
1022 |
|
|
} else if ((bptr = (char *)strstr(m->m_data, "27 Entering")) != NULL) { |
1023 |
|
|
/* |
1024 |
|
|
* Need to emulate the PASV response |
1025 |
|
|
*/ |
1026 |
|
|
x = sscanf(bptr, "27 Entering Passive Mode (%d,%d,%d,%d,%d,%d)\r\n%256[^\177]", |
1027 |
|
|
&n1, &n2, &n3, &n4, &n5, &n6, buff); |
1028 |
|
|
if (x < 6) |
1029 |
|
|
return 1; |
1030 |
|
|
|
1031 |
|
|
laddr = htonl((n1 << 24) | (n2 << 16) | (n3 << 8) | (n4)); |
1032 |
|
|
lport = htons((n5 << 8) | (n6)); |
1033 |
|
|
|
1034 |
|
|
if ((so = solisten(0, laddr, lport, SS_FACCEPTONCE)) == NULL) |
1035 |
|
|
return 1; |
1036 |
|
|
|
1037 |
|
|
n6 = ntohs(so->so_fport); |
1038 |
|
|
|
1039 |
|
|
n5 = (n6 >> 8) & 0xff; |
1040 |
|
|
n6 &= 0xff; |
1041 |
|
|
|
1042 |
|
|
laddr = ntohl(so->so_faddr.s_addr); |
1043 |
|
|
|
1044 |
|
|
n1 = ((laddr >> 24) & 0xff); |
1045 |
|
|
n2 = ((laddr >> 16) & 0xff); |
1046 |
|
|
n3 = ((laddr >> 8) & 0xff); |
1047 |
|
|
n4 = (laddr & 0xff); |
1048 |
|
|
|
1049 |
|
|
m->m_len = bptr - m->m_data; /* Adjust length */ |
1050 |
|
|
m->m_len += sprintf(bptr,"27 Entering Passive Mode (%d,%d,%d,%d,%d,%d)\r\n%s", |
1051 |
|
|
n1, n2, n3, n4, n5, n6, x==7?buff:""); |
1052 |
|
|
|
1053 |
|
|
return 1; |
1054 |
|
|
} |
1055 |
|
|
|
1056 |
|
|
return 1; |
1057 |
|
|
|
1058 |
|
|
case EMU_KSH: |
1059 |
|
|
/* |
1060 |
|
|
* The kshell (Kerberos rsh) and shell services both pass |
1061 |
|
|
* a local port port number to carry signals to the server |
1062 |
|
|
* and stderr to the client. It is passed at the beginning |
1063 |
|
|
* of the connection as a NUL-terminated decimal ASCII string. |
1064 |
|
|
*/ |
1065 |
|
|
so->so_emu = 0; |
1066 |
|
|
for (lport = 0, i = 0; i < m->m_len-1; ++i) { |
1067 |
|
|
if (m->m_data[i] < '0' || m->m_data[i] > '9') |
1068 |
|
|
return 1; /* invalid number */ |
1069 |
|
|
lport *= 10; |
1070 |
|
|
lport += m->m_data[i] - '0'; |
1071 |
|
|
} |
1072 |
|
|
if (m->m_data[m->m_len-1] == '\0' && lport != 0 && |
1073 |
|
|
(so = solisten(0, so->so_laddr.s_addr, htons(lport), SS_FACCEPTONCE)) != NULL) |
1074 |
|
|
m->m_len = sprintf(m->m_data, "%d", ntohs(so->so_fport))+1; |
1075 |
|
|
return 1; |
1076 |
|
|
|
1077 |
|
|
case EMU_IRC: |
1078 |
|
|
/* |
1079 |
|
|
* Need to emulate DCC CHAT, DCC SEND and DCC MOVE |
1080 |
|
|
*/ |
1081 |
|
|
*(m->m_data+m->m_len) = 0; /* NULL terminate the string for strstr */ |
1082 |
|
|
if ((bptr = (char *)strstr(m->m_data, "DCC")) == NULL) |
1083 |
|
|
return 1; |
1084 |
|
|
|
1085 |
|
|
/* The %256s is for the broken mIRC */ |
1086 |
|
|
if (sscanf(bptr, "DCC CHAT %256s %u %u", buff, &laddr, &lport) == 3) { |
1087 |
|
|
if ((so = solisten(0, htonl(laddr), htons(lport), SS_FACCEPTONCE)) == NULL) |
1088 |
|
|
return 1; |
1089 |
|
|
|
1090 |
|
|
m->m_len = bptr - m->m_data; /* Adjust length */ |
1091 |
|
|
m->m_len += sprintf(bptr, "DCC CHAT chat %lu %u%c\n", |
1092 |
|
|
(unsigned long)ntohl(so->so_faddr.s_addr), |
1093 |
|
|
ntohs(so->so_fport), 1); |
1094 |
|
|
} else if (sscanf(bptr, "DCC SEND %256s %u %u %u", buff, &laddr, &lport, &n1) == 4) { |
1095 |
|
|
if ((so = solisten(0, htonl(laddr), htons(lport), SS_FACCEPTONCE)) == NULL) |
1096 |
|
|
return 1; |
1097 |
|
|
|
1098 |
|
|
m->m_len = bptr - m->m_data; /* Adjust length */ |
1099 |
|
|
m->m_len += sprintf(bptr, "DCC SEND %s %lu %u %u%c\n", |
1100 |
|
|
buff, (unsigned long)ntohl(so->so_faddr.s_addr), |
1101 |
|
|
ntohs(so->so_fport), n1, 1); |
1102 |
|
|
} else if (sscanf(bptr, "DCC MOVE %256s %u %u %u", buff, &laddr, &lport, &n1) == 4) { |
1103 |
|
|
if ((so = solisten(0, htonl(laddr), htons(lport), SS_FACCEPTONCE)) == NULL) |
1104 |
|
|
return 1; |
1105 |
|
|
|
1106 |
|
|
m->m_len = bptr - m->m_data; /* Adjust length */ |
1107 |
|
|
m->m_len += sprintf(bptr, "DCC MOVE %s %lu %u %u%c\n", |
1108 |
|
|
buff, (unsigned long)ntohl(so->so_faddr.s_addr), |
1109 |
|
|
ntohs(so->so_fport), n1, 1); |
1110 |
|
|
} |
1111 |
|
|
return 1; |
1112 |
|
|
|
1113 |
|
|
case EMU_REALAUDIO: |
1114 |
|
|
/* |
1115 |
|
|
* RealAudio emulation - JP. We must try to parse the incoming |
1116 |
|
|
* data and try to find the two characters that contain the |
1117 |
|
|
* port number. Then we redirect an udp port and replace the |
1118 |
|
|
* number with the real port we got. |
1119 |
|
|
* |
1120 |
|
|
* The 1.0 beta versions of the player are not supported |
1121 |
|
|
* any more. |
1122 |
|
|
* |
1123 |
|
|
* A typical packet for player version 1.0 (release version): |
1124 |
|
|
* |
1125 |
|
|
* 0000:50 4E 41 00 05 |
1126 |
|
|
* 0000:00 01 00 02 1B D7 00 00 67 E6 6C DC 63 00 12 50 .....×..gælÜc..P |
1127 |
|
|
* 0010:4E 43 4C 49 45 4E 54 20 31 30 31 20 41 4C 50 48 NCLIENT 101 ALPH |
1128 |
|
|
* 0020:41 6C 00 00 52 00 17 72 61 66 69 6C 65 73 2F 76 Al..R..rafiles/v |
1129 |
|
|
* 0030:6F 61 2F 65 6E 67 6C 69 73 68 5F 2E 72 61 79 42 oa/english_.rayB |
1130 |
|
|
* |
1131 |
|
|
* Now the port number 0x1BD7 is found at offset 0x04 of the |
1132 |
|
|
* Now the port number 0x1BD7 is found at offset 0x04 of the |
1133 |
|
|
* second packet. This time we received five bytes first and |
1134 |
|
|
* then the rest. You never know how many bytes you get. |
1135 |
|
|
* |
1136 |
|
|
* A typical packet for player version 2.0 (beta): |
1137 |
|
|
* |
1138 |
|
|
* 0000:50 4E 41 00 06 00 02 00 00 00 01 00 02 1B C1 00 PNA...........Á. |
1139 |
|
|
* 0010:00 67 75 78 F5 63 00 0A 57 69 6E 32 2E 30 2E 30 .guxõc..Win2.0.0 |
1140 |
|
|
* 0020:2E 35 6C 00 00 52 00 1C 72 61 66 69 6C 65 73 2F .5l..R..rafiles/ |
1141 |
|
|
* 0030:77 65 62 73 69 74 65 2F 32 30 72 65 6C 65 61 73 website/20releas |
1142 |
|
|
* 0040:65 2E 72 61 79 53 00 00 06 36 42 e.rayS...6B |
1143 |
|
|
* |
1144 |
|
|
* Port number 0x1BC1 is found at offset 0x0d. |
1145 |
|
|
* |
1146 |
|
|
* This is just a horrible switch statement. Variable ra tells |
1147 |
|
|
* us where we're going. |
1148 |
|
|
*/ |
1149 |
|
|
|
1150 |
|
|
bptr = m->m_data; |
1151 |
|
|
while (bptr < m->m_data + m->m_len) { |
1152 |
|
|
u_short p; |
1153 |
|
|
static int ra = 0; |
1154 |
|
|
char ra_tbl[4]; |
1155 |
|
|
|
1156 |
|
|
ra_tbl[0] = 0x50; |
1157 |
|
|
ra_tbl[1] = 0x4e; |
1158 |
|
|
ra_tbl[2] = 0x41; |
1159 |
|
|
ra_tbl[3] = 0; |
1160 |
|
|
|
1161 |
|
|
switch (ra) { |
1162 |
|
|
case 0: |
1163 |
|
|
case 2: |
1164 |
|
|
case 3: |
1165 |
|
|
if (*bptr++ != ra_tbl[ra]) { |
1166 |
|
|
ra = 0; |
1167 |
|
|
continue; |
1168 |
|
|
} |
1169 |
|
|
break; |
1170 |
|
|
|
1171 |
|
|
case 1: |
1172 |
|
|
/* |
1173 |
|
|
* We may get 0x50 several times, ignore them |
1174 |
|
|
*/ |
1175 |
|
|
if (*bptr == 0x50) { |
1176 |
|
|
ra = 1; |
1177 |
|
|
bptr++; |
1178 |
|
|
continue; |
1179 |
|
|
} else if (*bptr++ != ra_tbl[ra]) { |
1180 |
|
|
ra = 0; |
1181 |
|
|
continue; |
1182 |
|
|
} |
1183 |
|
|
break; |
1184 |
|
|
|
1185 |
|
|
case 4: |
1186 |
|
|
/* |
1187 |
|
|
* skip version number |
1188 |
|
|
*/ |
1189 |
|
|
bptr++; |
1190 |
|
|
break; |
1191 |
|
|
|
1192 |
|
|
case 5: |
1193 |
|
|
/* |
1194 |
|
|
* The difference between versions 1.0 and |
1195 |
|
|
* 2.0 is here. For future versions of |
1196 |
|
|
* the player this may need to be modified. |
1197 |
|
|
*/ |
1198 |
|
|
if (*(bptr + 1) == 0x02) |
1199 |
|
|
bptr += 8; |
1200 |
|
|
else |
1201 |
|
|
bptr += 4; |
1202 |
|
|
break; |
1203 |
|
|
|
1204 |
|
|
case 6: |
1205 |
|
|
/* This is the field containing the port |
1206 |
|
|
* number that RA-player is listening to. |
1207 |
|
|
*/ |
1208 |
|
|
lport = (((u_char*)bptr)[0] << 8) |
1209 |
|
|
+ ((u_char *)bptr)[1]; |
1210 |
|
|
if (lport < 6970) |
1211 |
|
|
lport += 256; /* don't know why */ |
1212 |
|
|
if (lport < 6970 || lport > 7170) |
1213 |
|
|
return 1; /* failed */ |
1214 |
|
|
|
1215 |
|
|
/* try to get udp port between 6970 - 7170 */ |
1216 |
|
|
for (p = 6970; p < 7071; p++) { |
1217 |
|
|
if (udp_listen( htons(p), |
1218 |
|
|
so->so_laddr.s_addr, |
1219 |
|
|
htons(lport), |
1220 |
|
|
SS_FACCEPTONCE)) { |
1221 |
|
|
break; |
1222 |
|
|
} |
1223 |
|
|
} |
1224 |
|
|
if (p == 7071) |
1225 |
|
|
p = 0; |
1226 |
|
|
*(u_char *)bptr++ = (p >> 8) & 0xff; |
1227 |
|
|
*(u_char *)bptr++ = p & 0xff; |
1228 |
|
|
ra = 0; |
1229 |
|
|
return 1; /* port redirected, we're done */ |
1230 |
|
|
break; |
1231 |
|
|
|
1232 |
|
|
default: |
1233 |
|
|
ra = 0; |
1234 |
|
|
} |
1235 |
|
|
ra++; |
1236 |
|
|
} |
1237 |
|
|
return 1; |
1238 |
|
|
|
1239 |
|
|
default: |
1240 |
|
|
/* Ooops, not emulated, won't call tcp_emu again */ |
1241 |
|
|
so->so_emu = 0; |
1242 |
|
|
return 1; |
1243 |
|
|
} |
1244 |
|
|
} |
1245 |
|
|
|
1246 |
|
|
/* |
1247 |
|
|
* Do misc. config of SLiRP while its running. |
1248 |
|
|
* Return 0 if this connections is to be closed, 1 otherwise, |
1249 |
|
|
* return 2 if this is a command-line connection |
1250 |
|
|
*/ |
1251 |
|
|
int |
1252 |
|
|
tcp_ctl(so) |
1253 |
|
|
struct socket *so; |
1254 |
|
|
{ |
1255 |
|
|
struct sbuf *sb = &so->so_snd; |
1256 |
|
|
int command; |
1257 |
|
|
struct ex_list *ex_ptr; |
1258 |
|
|
int do_pty; |
1259 |
|
|
// struct socket *tmpso; |
1260 |
|
|
|
1261 |
|
|
DEBUG_CALL("tcp_ctl"); |
1262 |
|
|
DEBUG_ARG("so = %lx", (long )so); |
1263 |
|
|
|
1264 |
|
|
#if 0 |
1265 |
|
|
/* |
1266 |
|
|
* Check if they're authorised |
1267 |
|
|
*/ |
1268 |
|
|
if (ctl_addr.s_addr && (ctl_addr.s_addr == -1 || (so->so_laddr.s_addr != ctl_addr.s_addr))) { |
1269 |
|
|
sb->sb_cc = sprintf(sb->sb_wptr,"Error: Permission denied.\r\n"); |
1270 |
|
|
sb->sb_wptr += sb->sb_cc; |
1271 |
|
|
return 0; |
1272 |
|
|
} |
1273 |
|
|
#endif |
1274 |
|
|
command = (ntohl(so->so_faddr.s_addr) & 0xff); |
1275 |
|
|
|
1276 |
|
|
switch(command) { |
1277 |
|
|
default: /* Check for exec's */ |
1278 |
|
|
|
1279 |
|
|
/* |
1280 |
|
|
* Check if it's pty_exec |
1281 |
|
|
*/ |
1282 |
|
|
for (ex_ptr = exec_list; ex_ptr; ex_ptr = ex_ptr->ex_next) { |
1283 |
|
|
if (ex_ptr->ex_fport == so->so_fport && |
1284 |
|
|
command == ex_ptr->ex_addr) { |
1285 |
|
|
do_pty = ex_ptr->ex_pty; |
1286 |
|
|
goto do_exec; |
1287 |
|
|
} |
1288 |
|
|
} |
1289 |
|
|
|
1290 |
|
|
/* |
1291 |
|
|
* Nothing bound.. |
1292 |
|
|
*/ |
1293 |
|
|
/* tcp_fconnect(so); */ |
1294 |
|
|
|
1295 |
|
|
/* FALLTHROUGH */ |
1296 |
|
|
case CTL_ALIAS: |
1297 |
|
|
sb->sb_cc = sprintf(sb->sb_wptr, |
1298 |
|
|
"Error: No application configured.\r\n"); |
1299 |
|
|
sb->sb_wptr += sb->sb_cc; |
1300 |
|
|
return(0); |
1301 |
|
|
|
1302 |
|
|
do_exec: |
1303 |
|
|
DEBUG_MISC((dfd, " executing %s \n",ex_ptr->ex_exec)); |
1304 |
|
|
return(fork_exec(so, ex_ptr->ex_exec, do_pty)); |
1305 |
|
|
|
1306 |
|
|
#if 0 |
1307 |
|
|
case CTL_CMD: |
1308 |
|
|
for (tmpso = tcb.so_next; tmpso != &tcb; tmpso = tmpso->so_next) { |
1309 |
|
|
if (tmpso->so_emu == EMU_CTL && |
1310 |
|
|
!(tmpso->so_tcpcb? |
1311 |
|
|
(tmpso->so_tcpcb->t_state & (TCPS_TIME_WAIT|TCPS_LAST_ACK)) |
1312 |
|
|
:0)) { |
1313 |
|
|
/* Ooops, control connection already active */ |
1314 |
|
|
sb->sb_cc = sprintf(sb->sb_wptr,"Sorry, already connected.\r\n"); |
1315 |
|
|
sb->sb_wptr += sb->sb_cc; |
1316 |
|
|
return 0; |
1317 |
|
|
} |
1318 |
|
|
} |
1319 |
|
|
so->so_emu = EMU_CTL; |
1320 |
|
|
ctl_password_ok = 0; |
1321 |
|
|
sb->sb_cc = sprintf(sb->sb_wptr, "Slirp command-line ready (type \"help\" for help).\r\nSlirp> "); |
1322 |
|
|
sb->sb_wptr += sb->sb_cc; |
1323 |
|
|
do_echo=-1; |
1324 |
|
|
return(2); |
1325 |
|
|
#endif |
1326 |
|
|
} |
1327 |
|
|
} |