|
|
1.1 ! root 1: /* ! 2: * Copyright (c) 2000 Apple Computer, Inc. All rights reserved. ! 3: * ! 4: * @APPLE_LICENSE_HEADER_START@ ! 5: * ! 6: * The contents of this file constitute Original Code as defined in and ! 7: * are subject to the Apple Public Source License Version 1.1 (the ! 8: * "License"). You may not use this file except in compliance with the ! 9: * License. Please obtain a copy of the License at ! 10: * http://www.apple.com/publicsource and read it before using this file. ! 11: * ! 12: * This Original Code and all software distributed under the License are ! 13: * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER ! 14: * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, ! 15: * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, ! 16: * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the ! 17: * License for the specific language governing rights and limitations ! 18: * under the License. ! 19: * ! 20: * @APPLE_LICENSE_HEADER_END@ ! 21: */ ! 22: /* ! 23: * Copyright (c) 1992, 1993 ! 24: * The Regents of the University of California. All rights reserved. ! 25: * ! 26: * Redistribution and use in source and binary forms, with or without ! 27: * modification, are permitted provided that the following conditions ! 28: * are met: ! 29: * 1. Redistributions of source code must retain the above copyright ! 30: * notice, this list of conditions and the following disclaimer. ! 31: * 2. Redistributions in binary form must reproduce the above copyright ! 32: * notice, this list of conditions and the following disclaimer in the ! 33: * documentation and/or other materials provided with the distribution. ! 34: * 3. All advertising materials mentioning features or use of this software ! 35: * must display the following acknowledgement: ! 36: * This product includes software developed by the University of ! 37: * California, Berkeley and its contributors. ! 38: * 4. Neither the name of the University nor the names of its contributors ! 39: * may be used to endorse or promote products derived from this software ! 40: * without specific prior written permission. ! 41: * ! 42: * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND ! 43: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE ! 44: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ! 45: * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE ! 46: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL ! 47: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS ! 48: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) ! 49: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT ! 50: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY ! 51: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF ! 52: * SUCH DAMAGE. ! 53: * ! 54: * @(#)tuba_usrreq.c 8.1 (Berkeley) 6/10/93 ! 55: */ ! 56: ! 57: #include <sys/param.h> ! 58: #include <sys/systm.h> ! 59: #include <sys/malloc.h> ! 60: #include <sys/mbuf.h> ! 61: #include <sys/socket.h> ! 62: #include <sys/socketvar.h> ! 63: #include <sys/protosw.h> ! 64: #include <sys/errno.h> ! 65: #include <sys/stat.h> ! 66: ! 67: #include <net/if.h> ! 68: #include <net/route.h> ! 69: ! 70: #include <netinet/in.h> ! 71: #include <netinet/in_systm.h> ! 72: #include <netinet/ip.h> ! 73: #include <netinet/in_pcb.h> ! 74: #include <netinet/ip_var.h> ! 75: #include <netinet/tcp.h> ! 76: #include <netinet/tcp_fsm.h> ! 77: #include <netinet/tcp_seq.h> ! 78: #include <netinet/tcp_timer.h> ! 79: #include <netinet/tcp_var.h> ! 80: #include <netinet/tcpip.h> ! 81: #include <netinet/tcp_debug.h> ! 82: ! 83: #include <netiso/argo_debug.h> ! 84: #include <netiso/iso.h> ! 85: #include <netiso/clnp.h> ! 86: #include <netiso/iso_pcb.h> ! 87: #include <netiso/iso_var.h> ! 88: #include <netiso/tuba_table.h> ! 89: /* ! 90: * TCP protocol interface to socket abstraction. ! 91: */ ! 92: extern char *tcpstates[]; ! 93: extern struct inpcb tuba_inpcb; ! 94: extern struct isopcb tuba_isopcb; ! 95: ! 96: /* ! 97: * Process a TCP user request for TCP tb. If this is a send request ! 98: * then m is the mbuf chain of send data. If this is a timer expiration ! 99: * (called from the software clock routine), then timertype tells which timer. ! 100: */ ! 101: /*ARGSUSED*/ ! 102: tuba_usrreq(so, req, m, nam, control) ! 103: struct socket *so; ! 104: int req; ! 105: struct mbuf *m, *nam, *control; ! 106: { ! 107: register struct inpcb *inp; ! 108: register struct isopcb *isop; ! 109: register struct tcpcb *tp; ! 110: int s; ! 111: int error = 0; ! 112: int ostate; ! 113: struct sockaddr_iso *siso; ! 114: ! 115: if (req == PRU_CONTROL) ! 116: return (iso_control(so, (int)m, (caddr_t)nam, ! 117: (struct ifnet *)control)); ! 118: ! 119: s = splnet(); ! 120: inp = sotoinpcb(so); ! 121: /* ! 122: * When a TCP is attached to a socket, then there will be ! 123: * a (struct inpcb) pointed at by the socket, and this ! 124: * structure will point at a subsidary (struct tcpcb). ! 125: */ ! 126: if (inp == 0 && req != PRU_ATTACH) { ! 127: splx(s); ! 128: return (EINVAL); /* XXX */ ! 129: } ! 130: if (inp) { ! 131: tp = intotcpcb(inp); ! 132: if (tp == 0) ! 133: panic("tuba_usrreq"); ! 134: ostate = tp->t_state; ! 135: isop = (struct isopcb *)tp->t_tuba_pcb; ! 136: if (isop == 0) ! 137: panic("tuba_usrreq 2"); ! 138: } else ! 139: ostate = 0; ! 140: switch (req) { ! 141: ! 142: /* ! 143: * TCP attaches to socket via PRU_ATTACH, reserving space, ! 144: * and an internet control block. We also need to ! 145: * allocate an isopcb and separate the control block from ! 146: * tcp/ip ones. ! 147: */ ! 148: case PRU_ATTACH: ! 149: if (error = iso_pcballoc(so, &tuba_isopcb)) ! 150: break; ! 151: isop = (struct isopcb *)so->so_pcb; ! 152: so->so_pcb = 0; ! 153: if (error = tcp_usrreq(so, req, m, nam, control)) { ! 154: isop->isop_socket = 0; ! 155: iso_pcbdetach(isop); ! 156: } else { ! 157: inp = sotoinpcb(so); ! 158: remque(inp); ! 159: insque(inp, &tuba_inpcb); ! 160: inp->inp_head = &tuba_inpcb; ! 161: tp = intotcpcb(inp); ! 162: if (tp == 0) ! 163: panic("tuba_usrreq 3"); ! 164: tp->t_tuba_pcb = (caddr_t) isop; ! 165: } ! 166: goto notrace; ! 167: ! 168: /* ! 169: * PRU_DETACH detaches the TCP protocol from the socket. ! 170: * If the protocol state is non-embryonic, then can't ! 171: * do this directly: have to initiate a PRU_DISCONNECT, ! 172: * which may finish later; embryonic TCB's can just ! 173: * be discarded here. ! 174: */ ! 175: case PRU_DETACH: ! 176: if (tp->t_state > TCPS_LISTEN) ! 177: tp = tcp_disconnect(tp); ! 178: else ! 179: tp = tcp_close(tp); ! 180: if (tp == 0) ! 181: tuba_pcbdetach(isop); ! 182: break; ! 183: ! 184: /* ! 185: * Give the socket an address. ! 186: */ ! 187: case PRU_BIND: ! 188: siso = mtod(nam, struct sockaddr_iso *); ! 189: if (siso->siso_tlen && siso->siso_tlen != 2) { ! 190: error = EINVAL; ! 191: break; ! 192: } ! 193: if ((error = iso_pcbbind(isop, nam)) || ! 194: (siso = isop->isop_laddr) == 0) ! 195: break; ! 196: bcopy(TSEL(siso), &inp->inp_lport, 2); ! 197: if (siso->siso_nlen && ! 198: !(inp->inp_laddr.s_addr = tuba_lookup(siso, M_WAITOK))) ! 199: error = ENOBUFS; ! 200: break; ! 201: ! 202: /* ! 203: * Prepare to accept connections. ! 204: */ ! 205: case PRU_CONNECT: ! 206: case PRU_LISTEN: ! 207: if (inp->inp_lport == 0 && ! 208: (error = iso_pcbbind(isop, (struct mbuf *)0))) ! 209: break; ! 210: bcopy(TSEL(isop->isop_laddr), &inp->inp_lport, 2); ! 211: if (req == PRU_LISTEN) { ! 212: tp->t_state = TCPS_LISTEN; ! 213: break; ! 214: } ! 215: /*FALLTHROUGH*/ ! 216: /* ! 217: * Initiate connection to peer. ! 218: * Create a template for use in transmissions on this connection. ! 219: * Enter SYN_SENT state, and mark socket as connecting. ! 220: * Start keep-alive timer, and seed output sequence space. ! 221: * Send initial segment on connection. ! 222: */ ! 223: /* case PRU_CONNECT: */ ! 224: if (error = iso_pcbconnect(isop, nam)) ! 225: break; ! 226: if ((siso = isop->isop_laddr) && siso->siso_nlen > 1) ! 227: siso->siso_data[siso->siso_nlen - 1] = ISOPROTO_TCP; ! 228: else ! 229: panic("tuba_usrreq: connect"); ! 230: siso = mtod(nam, struct sockaddr_iso *); ! 231: if (!(inp->inp_faddr.s_addr = tuba_lookup(siso, M_WAITOK))) { ! 232: unconnect: ! 233: iso_pcbdisconnect(isop); ! 234: error = ENOBUFS; ! 235: break; ! 236: } ! 237: bcopy(TSEL(isop->isop_faddr), &inp->inp_fport, 2); ! 238: if (inp->inp_laddr.s_addr == 0 && ! 239: (inp->inp_laddr.s_addr = ! 240: tuba_lookup(isop->isop_laddr, M_WAITOK)) == 0) ! 241: goto unconnect; ! 242: if ((tp->t_template = tcp_template(tp)) == 0) ! 243: goto unconnect; ! 244: soisconnecting(so); ! 245: tcpstat.tcps_connattempt++; ! 246: tp->t_state = TCPS_SYN_SENT; ! 247: tp->t_timer[TCPT_KEEP] = TCPTV_KEEP_INIT; ! 248: tp->iss = tcp_iss; tcp_iss += TCP_ISSINCR/2; ! 249: tcp_sendseqinit(tp); ! 250: error = tcp_output(tp); ! 251: tuba_refcnt(isop, 1); ! 252: break; ! 253: ! 254: /* ! 255: * Initiate disconnect from peer. ! 256: * If connection never passed embryonic stage, just drop; ! 257: * else if don't need to let data drain, then can just drop anyways, ! 258: * else have to begin TCP shutdown process: mark socket disconnecting, ! 259: * drain unread data, state switch to reflect user close, and ! 260: * send segment (e.g. FIN) to peer. Socket will be really disconnected ! 261: * when peer sends FIN and acks ours. ! 262: * ! 263: * SHOULD IMPLEMENT LATER PRU_CONNECT VIA REALLOC TCPCB. ! 264: */ ! 265: case PRU_DISCONNECT: ! 266: if ((tp = tcp_disconnect(tp)) == 0) ! 267: tuba_pcbdetach(isop); ! 268: break; ! 269: ! 270: /* ! 271: * Accept a connection. Essentially all the work is ! 272: * done at higher levels; just return the address ! 273: * of the peer, storing through addr. ! 274: */ ! 275: case PRU_ACCEPT: ! 276: bcopy((caddr_t)isop->isop_faddr, mtod(nam, caddr_t), ! 277: nam->m_len = isop->isop_faddr->siso_len); ! 278: break; ! 279: ! 280: /* ! 281: * Mark the connection as being incapable of further output. ! 282: */ ! 283: case PRU_SHUTDOWN: ! 284: socantsendmore(so); ! 285: tp = tcp_usrclosed(tp); ! 286: if (tp) ! 287: error = tcp_output(tp); ! 288: else ! 289: tuba_pcbdetach(isop); ! 290: break; ! 291: /* ! 292: * Abort the TCP. ! 293: */ ! 294: case PRU_ABORT: ! 295: if ((tp = tcp_drop(tp, ECONNABORTED)) == 0) ! 296: tuba_pcbdetach(isop); ! 297: break; ! 298: ! 299: ! 300: case PRU_SOCKADDR: ! 301: if (isop->isop_laddr) ! 302: bcopy((caddr_t)isop->isop_laddr, mtod(nam, caddr_t), ! 303: nam->m_len = isop->isop_laddr->siso_len); ! 304: break; ! 305: ! 306: case PRU_PEERADDR: ! 307: if (isop->isop_faddr) ! 308: bcopy((caddr_t)isop->isop_faddr, mtod(nam, caddr_t), ! 309: nam->m_len = isop->isop_faddr->siso_len); ! 310: break; ! 311: ! 312: default: ! 313: error = tcp_usrreq(so, req, m, nam, control); ! 314: goto notrace; ! 315: } ! 316: if (tp && (so->so_options & SO_DEBUG)) ! 317: tcp_trace(TA_USER, ostate, tp, (struct tcpiphdr *)0, req); ! 318: notrace: ! 319: splx(s); ! 320: return(error); ! 321: } ! 322: ! 323: tuba_ctloutput(op, so, level, optname, mp) ! 324: int op; ! 325: struct socket *so; ! 326: int level, optname; ! 327: struct mbuf **mp; ! 328: { ! 329: int clnp_ctloutput(), tcp_ctloutput(); ! 330: ! 331: return ((level != IPPROTO_TCP ? clnp_ctloutput : tcp_ctloutput) ! 332: (op, so, level, optname, mp)); ! 333: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.