Annotation of 42BSD/sys/netinet/in_pcb.c, revision 1.1

1.1     ! root        1: /*     in_pcb.c        6.1     83/07/29        */
        !             2: 
        !             3: #include "../h/param.h"
        !             4: #include "../h/systm.h"
        !             5: #include "../h/dir.h"
        !             6: #include "../h/user.h"
        !             7: #include "../h/mbuf.h"
        !             8: #include "../h/socket.h"
        !             9: #include "../h/socketvar.h"
        !            10: #include "../netinet/in.h"
        !            11: #include "../netinet/in_systm.h"
        !            12: #include "../net/if.h"
        !            13: #include "../net/route.h"
        !            14: #include "../netinet/in_pcb.h"
        !            15: #include "../h/protosw.h"
        !            16: 
        !            17: struct in_addr zeroin_addr;
        !            18: 
        !            19: in_pcballoc(so, head)
        !            20:        struct socket *so;
        !            21:        struct inpcb *head;
        !            22: {
        !            23:        struct mbuf *m;
        !            24:        register struct inpcb *inp;
        !            25: 
        !            26:        m = m_getclr(M_DONTWAIT, MT_PCB);
        !            27:        if (m == NULL)
        !            28:                return (ENOBUFS);
        !            29:        inp = mtod(m, struct inpcb *);
        !            30:        inp->inp_head = head;
        !            31:        inp->inp_socket = so;
        !            32:        insque(inp, head);
        !            33:        so->so_pcb = (caddr_t)inp;
        !            34:        return (0);
        !            35: }
        !            36:        
        !            37: in_pcbbind(inp, nam)
        !            38:        register struct inpcb *inp;
        !            39:        struct mbuf *nam;
        !            40: {
        !            41:        register struct socket *so = inp->inp_socket;
        !            42:        register struct inpcb *head = inp->inp_head;
        !            43:        register struct sockaddr_in *sin;
        !            44:        u_short lport = 0;
        !            45: 
        !            46:        if (ifnet == 0)
        !            47:                return (EADDRNOTAVAIL);
        !            48:        if (inp->inp_lport || inp->inp_laddr.s_addr != INADDR_ANY)
        !            49:                return (EINVAL);
        !            50:        if (nam == 0)
        !            51:                goto noname;
        !            52:        sin = mtod(nam, struct sockaddr_in *);
        !            53:        if (nam->m_len != sizeof (*sin))
        !            54:                return (EINVAL);
        !            55:        if (sin->sin_addr.s_addr != INADDR_ANY) {
        !            56:                int tport = sin->sin_port;
        !            57: 
        !            58:                sin->sin_port = 0;              /* yech... */
        !            59:                if (if_ifwithaddr((struct sockaddr *)sin) == 0)
        !            60:                        return (EADDRNOTAVAIL);
        !            61:                sin->sin_port = tport;
        !            62:        }
        !            63:        lport = sin->sin_port;
        !            64:        if (lport) {
        !            65:                u_short aport = htons(lport);
        !            66:                int wild = 0;
        !            67: 
        !            68:                /* GROSS */
        !            69:                if (aport < IPPORT_RESERVED && u.u_uid != 0)
        !            70:                        return (EACCES);
        !            71:                /* even GROSSER, but this is the Internet */
        !            72:                if ((so->so_options & SO_REUSEADDR) == 0 &&
        !            73:                    ((so->so_proto->pr_flags & PR_CONNREQUIRED) == 0 ||
        !            74:                     (so->so_options & SO_ACCEPTCONN) == 0))
        !            75:                        wild = INPLOOKUP_WILDCARD;
        !            76:                if (in_pcblookup(head,
        !            77:                    zeroin_addr, 0, sin->sin_addr, lport, wild))
        !            78:                        return (EADDRINUSE);
        !            79:        }
        !            80:        inp->inp_laddr = sin->sin_addr;
        !            81: noname:
        !            82:        if (lport == 0)
        !            83:                do {
        !            84:                        if (head->inp_lport++ < IPPORT_RESERVED)
        !            85:                                head->inp_lport = IPPORT_RESERVED;
        !            86:                        lport = htons(head->inp_lport);
        !            87:                } while (in_pcblookup(head,
        !            88:                            zeroin_addr, 0, inp->inp_laddr, lport, 0));
        !            89:        inp->inp_lport = lport;
        !            90:        return (0);
        !            91: }
        !            92: 
        !            93: /*
        !            94:  * Connect from a socket to a specified address.
        !            95:  * Both address and port must be specified in argument sin.
        !            96:  * If don't have a local address for this socket yet,
        !            97:  * then pick one.
        !            98:  */
        !            99: in_pcbconnect(inp, nam)
        !           100:        struct inpcb *inp;
        !           101:        struct mbuf *nam;
        !           102: {
        !           103:        struct ifnet *ifp;
        !           104:        struct sockaddr_in *ifaddr;
        !           105:        register struct sockaddr_in *sin = mtod(nam, struct sockaddr_in *);
        !           106: 
        !           107:        if (nam->m_len != sizeof (*sin))
        !           108:                return (EINVAL);
        !           109:        if (sin->sin_family != AF_INET)
        !           110:                return (EAFNOSUPPORT);
        !           111:        if (sin->sin_addr.s_addr == INADDR_ANY || sin->sin_port == 0)
        !           112:                return (EADDRNOTAVAIL);
        !           113:        if (inp->inp_laddr.s_addr == INADDR_ANY) {
        !           114:                ifp = if_ifonnetof(in_netof(sin->sin_addr));
        !           115:                if (ifp == 0) {
        !           116:                        /*
        !           117:                         * We should select the interface based on
        !           118:                         * the route to be used, but for udp this would
        !           119:                         * result in two calls to rtalloc for each packet
        !           120:                         * sent; hardly worthwhile...
        !           121:                         */
        !           122:                        ifp = if_ifwithaf(AF_INET);
        !           123:                        if (ifp == 0)
        !           124:                                return (EADDRNOTAVAIL);
        !           125:                }
        !           126:                ifaddr = (struct sockaddr_in *)&ifp->if_addr;
        !           127:        }
        !           128:        if (in_pcblookup(inp->inp_head,
        !           129:            sin->sin_addr,
        !           130:            sin->sin_port,
        !           131:            inp->inp_laddr.s_addr ? inp->inp_laddr : ifaddr->sin_addr,
        !           132:            inp->inp_lport,
        !           133:            0))
        !           134:                return (EADDRINUSE);
        !           135:        if (inp->inp_laddr.s_addr == INADDR_ANY) {
        !           136:                if (inp->inp_lport == 0)
        !           137:                        in_pcbbind(inp, (struct mbuf *)0);
        !           138:                inp->inp_laddr = ifaddr->sin_addr;
        !           139:        }
        !           140:        inp->inp_faddr = sin->sin_addr;
        !           141:        inp->inp_fport = sin->sin_port;
        !           142:        return (0);
        !           143: }
        !           144: 
        !           145: in_pcbdisconnect(inp)
        !           146:        struct inpcb *inp;
        !           147: {
        !           148: 
        !           149:        inp->inp_faddr.s_addr = INADDR_ANY;
        !           150:        inp->inp_fport = 0;
        !           151:        if (inp->inp_socket->so_state & SS_NOFDREF)
        !           152:                in_pcbdetach(inp);
        !           153: }
        !           154: 
        !           155: in_pcbdetach(inp)
        !           156:        struct inpcb *inp;
        !           157: {
        !           158:        struct socket *so = inp->inp_socket;
        !           159: 
        !           160:        so->so_pcb = 0;
        !           161:        sofree(so);
        !           162:        if (inp->inp_route.ro_rt)
        !           163:                rtfree(inp->inp_route.ro_rt);
        !           164:        remque(inp);
        !           165:        (void) m_free(dtom(inp));
        !           166: }
        !           167: 
        !           168: in_setsockaddr(inp, nam)
        !           169:        register struct inpcb *inp;
        !           170:        struct mbuf *nam;
        !           171: {
        !           172:        register struct sockaddr_in *sin = mtod(nam, struct sockaddr_in *);
        !           173:        
        !           174:        nam->m_len = sizeof (*sin);
        !           175:        sin = mtod(nam, struct sockaddr_in *);
        !           176:        bzero((caddr_t)sin, sizeof (*sin));
        !           177:        sin->sin_family = AF_INET;
        !           178:        sin->sin_port = inp->inp_lport;
        !           179:        sin->sin_addr = inp->inp_laddr;
        !           180: }
        !           181: 
        !           182: in_setpeeraddr(inp, nam)
        !           183:        register struct inpcb *inp;
        !           184:        struct mbuf *nam;
        !           185: {
        !           186:        register struct sockaddr_in *sin = mtod(nam, struct sockaddr_in *);
        !           187:        
        !           188:        nam->m_len = sizeof (*sin);
        !           189:        sin = mtod(nam, struct sockaddr_in *);
        !           190:        bzero((caddr_t)sin, sizeof (*sin));
        !           191:        sin->sin_family = AF_INET;
        !           192:        sin->sin_port = inp->inp_fport;
        !           193:        sin->sin_addr = inp->inp_faddr;
        !           194: }
        !           195: 
        !           196: /*
        !           197:  * Pass an error to all internet connections
        !           198:  * associated with address sin.  Call the
        !           199:  * protocol specific routine to clean up the
        !           200:  * mess afterwards.
        !           201:  */
        !           202: in_pcbnotify(head, dst, errno, abort)
        !           203:        struct inpcb *head;
        !           204:        register struct in_addr *dst;
        !           205:        int errno, (*abort)();
        !           206: {
        !           207:        register struct inpcb *inp, *oinp;
        !           208:        int s = splimp();
        !           209: 
        !           210:        for (inp = head->inp_next; inp != head;) {
        !           211:                if (inp->inp_faddr.s_addr != dst->s_addr) {
        !           212:        next:
        !           213:                        inp = inp->inp_next;
        !           214:                        continue;
        !           215:                }
        !           216:                if (inp->inp_socket == 0)
        !           217:                        goto next;
        !           218:                inp->inp_socket->so_error = errno;
        !           219:                oinp = inp;
        !           220:                inp = inp->inp_next;
        !           221:                (*abort)(oinp);
        !           222:        }
        !           223:        splx(s);
        !           224: }
        !           225: 
        !           226: struct inpcb *
        !           227: in_pcblookup(head, faddr, fport, laddr, lport, flags)
        !           228:        struct inpcb *head;
        !           229:        struct in_addr faddr, laddr;
        !           230:        u_short fport, lport;
        !           231:        int flags;
        !           232: {
        !           233:        register struct inpcb *inp, *match = 0;
        !           234:        int matchwild = 3, wildcard;
        !           235: 
        !           236:        for (inp = head->inp_next; inp != head; inp = inp->inp_next) {
        !           237:                if (inp->inp_lport != lport)
        !           238:                        continue;
        !           239:                wildcard = 0;
        !           240:                if (inp->inp_laddr.s_addr != INADDR_ANY) {
        !           241:                        if (laddr.s_addr == INADDR_ANY)
        !           242:                                wildcard++;
        !           243:                        else if (inp->inp_laddr.s_addr != laddr.s_addr)
        !           244:                                continue;
        !           245:                } else {
        !           246:                        if (laddr.s_addr != INADDR_ANY)
        !           247:                                wildcard++;
        !           248:                }
        !           249:                if (inp->inp_faddr.s_addr != INADDR_ANY) {
        !           250:                        if (faddr.s_addr == INADDR_ANY)
        !           251:                                wildcard++;
        !           252:                        else if (inp->inp_faddr.s_addr != faddr.s_addr ||
        !           253:                            inp->inp_fport != fport)
        !           254:                                continue;
        !           255:                } else {
        !           256:                        if (faddr.s_addr != INADDR_ANY)
        !           257:                                wildcard++;
        !           258:                }
        !           259:                if (wildcard && (flags & INPLOOKUP_WILDCARD) == 0)
        !           260:                        continue;
        !           261:                if (wildcard < matchwild) {
        !           262:                        match = inp;
        !           263:                        matchwild = wildcard;
        !           264:                        if (matchwild == 0)
        !           265:                                break;
        !           266:                }
        !           267:        }
        !           268:        return (match);
        !           269: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.