Annotation of researchv9/sys/inet/ip_input.c, revision 1.1.1.1

1.1       root        1: /*     ip_input.c      6.1     83/08/16        */
                      2: 
                      3: #include "../h/param.h"
                      4: #include "../h/systm.h"
                      5: #include "../h/stream.h"
                      6: #include "../h/inet/mbuf.h"
                      7: #include "../h/inet/in.h"
                      8: #include "../h/inet/ip.h"
                      9: #include "../h/inet/ip_var.h"
                     10: 
                     11: int    ipqmaxlen = 50;
                     12: 
                     13: u_char ipcksum = 1;
                     14: struct ip *ip_reass();
                     15: 
                     16: /*
                     17:  * Ip input routine.  Checksum and byte swap header.  If fragmented
                     18:  * try to reassamble.  If complete and fragment queue exists, discard.
                     19:  * Process options.  Pass to next level.
                     20:  */
                     21: ip_input(m)
                     22: register struct mbuf *m;
                     23: {
                     24:        register struct ip *ip;
                     25:        register struct ipq *fp;
                     26:        register int i;
                     27:        int hlen;
                     28: 
                     29:        if (m == 0)
                     30:                return;
                     31:        if (BLEN(m) < sizeof (struct ip) &&
                     32:            (m = m_pullup(m, sizeof (struct ip))) == 0) {
                     33:                ipstat.ips_toosmall++;
                     34:                return;
                     35:        }
                     36:        ip = mtod(m, struct ip *);
                     37:        if ((hlen = ip->ip_hl << 2) > BLEN(m)) {
                     38:                if ((m = m_pullup(m, hlen)) == 0) {
                     39:                        ipstat.ips_badhlen++;
                     40:                        return;
                     41:                }
                     42:                ip = mtod(m, struct ip *);
                     43:        }
                     44:        if (ipcksum)
                     45:                if (ip->ip_sum = in_cksum(m, hlen)) {
                     46:                        ipstat.ips_badsum++;
                     47:                        goto bad;
                     48:                }
                     49: 
                     50:        /*
                     51:         * Convert fields to host representation.
                     52:         */
                     53:        ip->ip_len = ntohs((u_short)ip->ip_len);
                     54:        if (ip->ip_len < hlen) {
                     55:                ipstat.ips_badlen++;
                     56:                goto bad;
                     57:        }
                     58:        ip->ip_id = ntohs(ip->ip_id);
                     59:        ip->ip_off = ntohs((u_short)ip->ip_off);
                     60: 
                     61:        /*
                     62:         * Check that the amount of data in the buffers
                     63:         * is as at least much as the IP header would have us expect.
                     64:         * Trim mbufs if longer than we expect.
                     65:         * Drop packet if shorter than we expect.
                     66:         */
                     67:        i = bp_len(m) - ip->ip_len;
                     68:        if (i < 0) {
                     69:                ipstat.ips_tooshort++;
                     70:                goto bad;
                     71:        }
                     72:        if (i > 0)
                     73:                m_adj(m, -i);
                     74:        /*
                     75:         * Process options and, if not destined for us,
                     76:         * ship it on.  ip_dooptions returns 1 when an
                     77:         * error was detected (causing an icmp message
                     78:         * to be sent).
                     79:         */
                     80:        ip->ip_dst = ntohl(ip->ip_dst);
                     81:        ip->ip_src = ntohl(ip->ip_src);
                     82:        if (hlen > sizeof (struct ip) && ip_dooptions(ip))
                     83:                return;
                     84: 
                     85:        if (ip_ifwithaddr(ip->ip_dst) == 0) {
                     86:                ip_forward(ip);
                     87:                return;
                     88:        }
                     89: 
                     90:        /*
                     91:         * Look for queue of fragments
                     92:         * of this datagram.
                     93:         */
                     94:        if(ipq.next == 0 && ipq.prev == 0)      /* init, only once */
                     95:                ipq.next = ipq.prev = &ipq;
                     96:        for (fp = ipq.next; fp != &ipq; fp = fp->next)
                     97:                if (ip->ip_id == fp->ipq_id &&
                     98:                    ip->ip_src == fp->ipq_src &&
                     99:                    ip->ip_dst == fp->ipq_dst &&
                    100:                    ip->ip_p == fp->ipq_p)
                    101:                        goto found;
                    102:        fp = 0;
                    103: found:
                    104: 
                    105:        /*
                    106:         * Adjust ip_len to not reflect header,
                    107:         * set ip_mff if more fragments are expected,
                    108:         * convert offset of this to bytes.
                    109:         */
                    110:        ip->ip_len -= hlen;
                    111:        ((struct ipasfrag *)ip)->ipf_mff = 0;
                    112:        if (ip->ip_off & IP_MF)
                    113:                ((struct ipasfrag *)ip)->ipf_mff = 1;
                    114:        ip->ip_off <<= 3;
                    115: 
                    116:        /*
                    117:         * If datagram marked as having more fragments
                    118:         * or if this is not the first fragment,
                    119:         * attempt reassembly; if it succeeds, proceed.
                    120:         */
                    121:        if (((struct ipasfrag *)ip)->ipf_mff || ip->ip_off) {
                    122:                ip = ip_reass((struct ipasfrag *)ip, fp);
                    123:                if (ip == 0)
                    124:                        return;
                    125:                hlen = ip->ip_hl << 2;
                    126:                m = dtom(ip);
                    127:        } else
                    128:                if (fp)
                    129:                        ip_freef(fp);
                    130: 
                    131:        /*
                    132:         * Switch out to protocol's input routine.
                    133:         */
                    134:        ipdrint(m, (unsigned int)(ip->ip_p));
                    135:        return;
                    136: bad:
                    137:        m_freem(m);
                    138: }
                    139: 
                    140: /*
                    141:  * Take incoming datagram fragment and try to
                    142:  * reassemble it into whole datagram.  If a chain for
                    143:  * reassembly of this datagram already exists, then it
                    144:  * is given as fp; otherwise have to make a chain.
                    145:  */
                    146: struct ip *
                    147: ip_reass(ip, fp)
                    148:        register struct ipasfrag *ip;
                    149:        register struct ipq *fp;
                    150: {
                    151:        register struct mbuf *m = dtom(ip);
                    152:        register struct ipasfrag *q;
                    153:        struct mbuf *t;
                    154:        int hlen = ip->ip_hl << 2;
                    155:        int i, next;
                    156: 
                    157:        /*
                    158:         * Presence of header sizes in mbufs
                    159:         * would confuse code below.
                    160:         */
                    161:        m->rptr += hlen;
                    162: 
                    163:        /*
                    164:         * If first fragment to arrive, create a reassembly queue.
                    165:         */
                    166:        if (fp == 0) {
                    167:                if ((t = m_get(M_WAIT, MT_FTABLE)) == NULL)
                    168:                        goto dropfrag;
                    169:                t->m_next = 0;
                    170:                fp = mtod(t, struct ipq *);
                    171:                insque(fp, &ipq);
                    172:                fp->ipq_ttl = IPFRAGTTL;
                    173:                fp->ipq_p = ip->ip_p;
                    174:                fp->ipq_id = ip->ip_id;
                    175:                fp->ipq_next = fp->ipq_prev = (struct ipasfrag *)fp;
                    176:                fp->ipq_src = ((struct ip *)ip)->ip_src;
                    177:                fp->ipq_dst = ((struct ip *)ip)->ip_dst;
                    178:                q = (struct ipasfrag *)fp;
                    179:                goto insert;
                    180:        }
                    181: 
                    182:        /*
                    183:         * Find a segment which begins after this one does.
                    184:         */
                    185:        for (q = fp->ipq_next; q != (struct ipasfrag *)fp; q = q->ipf_next)
                    186:                if (q->ip_off > ip->ip_off)
                    187:                        break;
                    188: 
                    189:        /*
                    190:         * If there is a preceding segment, it may provide some of
                    191:         * our data already.  If so, drop the data from the incoming
                    192:         * segment.  If it provides all of our data, drop us.
                    193:         */
                    194:        if (q->ipf_prev != (struct ipasfrag *)fp) {
                    195:                i = q->ipf_prev->ip_off + q->ipf_prev->ip_len - ip->ip_off;
                    196:                if (i > 0) {
                    197:                        if (i >= ip->ip_len)
                    198:                                goto dropfrag;
                    199:                        m_adj(dtom(ip), i);
                    200:                        ip->ip_off += i;
                    201:                        ip->ip_len -= i;
                    202:                }
                    203:        }
                    204: 
                    205:        /*
                    206:         * While we overlap succeeding segments trim them or,
                    207:         * if they are completely covered, dequeue them.
                    208:         */
                    209:        while (q != (struct ipasfrag *)fp && ip->ip_off + ip->ip_len > q->ip_off) {
                    210:                i = (ip->ip_off + ip->ip_len) - q->ip_off;
                    211:                if (i < q->ip_len) {
                    212:                        q->ip_len -= i;
                    213:                        q->ip_off += i;
                    214:                        m_adj(dtom(q), i);
                    215:                        break;
                    216:                }
                    217:                q = q->ipf_next;
                    218:                m_freem(dtom(q->ipf_prev));
                    219:                ip_deq(q->ipf_prev);
                    220:        }
                    221: 
                    222: insert:
                    223:        /*
                    224:         * Stick new segment in its place;
                    225:         * check for complete reassembly.
                    226:         */
                    227:        ip_enq(ip, q->ipf_prev);
                    228:        next = 0;
                    229:        for (q = fp->ipq_next; q != (struct ipasfrag *)fp; q = q->ipf_next) {
                    230:                if (q->ip_off != next)
                    231:                        return (0);
                    232:                next += q->ip_len;
                    233:        }
                    234:        if (q->ipf_prev->ipf_mff)
                    235:                return (0);
                    236: 
                    237:        /*
                    238:         * Reassembly is complete; concatenate fragments.
                    239:         */
                    240:        q = fp->ipq_next;
                    241:        m = dtom(q);
                    242:        t = m->m_next;
                    243:        m->m_next = 0;
                    244:        m_cat(m, t);
                    245:        q = q->ipf_next;
                    246:        while (q != (struct ipasfrag *)fp) {
                    247:                t = dtom(q);
                    248:                q = q->ipf_next;
                    249:                m_cat(m, t);
                    250:        }
                    251: 
                    252:        /*
                    253:         * Create header for new ip packet by
                    254:         * modifying header of first packet;
                    255:         * dequeue and discard fragment reassembly header.
                    256:         * Make header visible.
                    257:         */
                    258:        ip = fp->ipq_next;
                    259:        ip->ip_len = next;
                    260:        ((struct ip *)ip)->ip_src = fp->ipq_src;
                    261:        ((struct ip *)ip)->ip_dst = fp->ipq_dst;
                    262:        remque(fp);
                    263:        (void) m_free(dtom(fp));
                    264:        m = dtom(ip);
                    265:        m->rptr -= sizeof (struct ipasfrag);
                    266:        return ((struct ip *)ip);
                    267: 
                    268: dropfrag:
                    269:        m_freem(m);
                    270:        return (0);
                    271: }
                    272: 
                    273: /*
                    274:  * Free a fragment reassembly header and all
                    275:  * associated datagrams.
                    276:  */
                    277: ip_freef(fp)
                    278:        struct ipq *fp;
                    279: {
                    280:        register struct ipasfrag *q, *p;
                    281: 
                    282:        for (q = fp->ipq_next; q != (struct ipasfrag *)fp; q = p) {
                    283:                p = q->ipf_next;
                    284:                ip_deq(q);
                    285:                m_freem(dtom(q));
                    286:        }
                    287:        remque(fp);
                    288:        (void) m_free(dtom(fp));
                    289: }
                    290: 
                    291: /*
                    292:  * Put an ip fragment on a reassembly chain.
                    293:  * Like insque, but pointers in middle of structure.
                    294:  */
                    295: ip_enq(p, prev)
                    296:        register struct ipasfrag *p, *prev;
                    297: {
                    298: 
                    299:        p->ipf_prev = prev;
                    300:        p->ipf_next = prev->ipf_next;
                    301:        prev->ipf_next->ipf_prev = p;
                    302:        prev->ipf_next = p;
                    303: }
                    304: 
                    305: /*
                    306:  * To ip_enq as remque is to insque.
                    307:  */
                    308: ip_deq(p)
                    309:        register struct ipasfrag *p;
                    310: {
                    311: 
                    312:        p->ipf_prev->ipf_next = p->ipf_next;
                    313:        p->ipf_next->ipf_prev = p->ipf_prev;
                    314: }
                    315: 
                    316: /*
                    317:  * IP timer processing;
                    318:  * if a timer expires on a reassembly
                    319:  * queue, discard it.
                    320:  */
                    321: ip_slowtimo()
                    322: {
                    323:        register struct ipq *fp;
                    324:        int s = spl6();
                    325: 
                    326:        fp = ipq.next;
                    327:        if (fp == 0) {
                    328:                splx(s);
                    329:                return;
                    330:        }
                    331:        while (fp != &ipq) {
                    332:                --fp->ipq_ttl;
                    333:                fp = fp->next;
                    334:                if (fp->prev->ipq_ttl == 0)
                    335:                        ip_freef(fp->prev);
                    336:        }
                    337:        timeout(ip_slowtimo, (caddr_t)0, hz);
                    338:        splx(s);
                    339: }
                    340: 
                    341: #if NOTDEF
                    342: /* who calls this? */
                    343: /*
                    344:  * Drain off all datagram fragments.
                    345:  */
                    346: ip_drain()
                    347: {
                    348: 
                    349:        while (ipq.next != &ipq)
                    350:                ip_freef(ipq.next);
                    351: }
                    352: #endif
                    353: 
                    354: /*
                    355:  * Do option processing on a datagram,
                    356:  * possibly discarding it if bad options
                    357:  * are encountered.
                    358:  */
                    359: ip_dooptions(ip)
                    360:        struct ip *ip;
                    361: {
                    362:        register u_char *cp;
                    363:        int opt, optlen, cnt;
                    364: 
                    365:        cp = (u_char *)(ip + 1);
                    366:        cnt = (ip->ip_hl << 2) - sizeof (struct ip);
                    367:        for (; cnt > 0; cnt -= optlen, cp += optlen) {
                    368:                opt = cp[0];
                    369:                if (opt == IPOPT_EOL)
                    370:                        break;
                    371:                if (opt == IPOPT_NOP)
                    372:                        optlen = 1;
                    373:                else
                    374:                        optlen = cp[1];
                    375:                switch (opt) {
                    376: 
                    377:                default:
                    378:                        break;
                    379: #ifdef FAT_CHANCE
                    380:                /*
                    381:                 * Source routing with record.
                    382:                 * Find interface with current destination address.
                    383:                 * If none on this machine then drop if strictly routed,
                    384:                 * or do nothing if loosely routed.
                    385:                 * Record interface address and bring up next address
                    386:                 * component.  If strictly routed make sure next
                    387:                 * address on directly accessible net.
                    388:                 */
                    389:                case IPOPT_LSRR:
                    390:                case IPOPT_SSRR:
                    391:                        if (cp[2] < 4 || cp[2] > optlen - (sizeof (long) - 1))
                    392:                                break;
                    393:                        sin = (struct in_addr *)(cp + cp[2]);
                    394:                        ipaddr.sin_addr = *sin;
                    395:                        ifp = if_ifwithaddr((struct sockaddr *)&ipaddr);
                    396:                        type = ICMP_UNREACH, code = ICMP_UNREACH_SRCFAIL;
                    397:                        if (ifp == 0) {
                    398:                                if (opt == IPOPT_SSRR)
                    399:                                        goto bad;
                    400:                                break;
                    401:                        }
                    402:                        t = ip->ip_dst; ip->ip_dst = *sin; *sin = t;
                    403:                        cp[2] += 4;
                    404:                        if (cp[2] > optlen - (sizeof (long) - 1))
                    405:                                break;
                    406:                        ip->ip_dst = sin[1];
                    407:                        if (opt == IPOPT_SSRR &&
                    408:                            if_ifonnetof(in_netof(ip->ip_dst)) == 0)
                    409:                                goto bad;
                    410:                        break;
                    411: 
                    412:                case IPOPT_TS:
                    413:                        code = cp - (u_char *)ip;
                    414:                        type = ICMP_PARAMPROB;
                    415:                        ipt = (struct ip_timestamp *)cp;
                    416:                        if (ipt->ipt_len < 5)
                    417:                                goto bad;
                    418:                        if (ipt->ipt_ptr > ipt->ipt_len - sizeof (long)) {
                    419:                                if (++ipt->ipt_oflw == 0)
                    420:                                        goto bad;
                    421:                                break;
                    422:                        }
                    423:                        sin = (struct in_addr *)(cp+cp[2]);
                    424:                        switch (ipt->ipt_flg) {
                    425: 
                    426:                        case IPOPT_TS_TSONLY:
                    427:                                break;
                    428: 
                    429:                        case IPOPT_TS_TSANDADDR:
                    430:                                if (ipt->ipt_ptr + 8 > ipt->ipt_len)
                    431:                                        goto bad;
                    432:                                if (ifinet == 0)
                    433:                                        goto bad;       /* ??? */
                    434:                                *sin++ = ((struct sockaddr_in *)&ifinet->if_addr)->sin_addr;
                    435:                                break;
                    436: 
                    437:                        case IPOPT_TS_PRESPEC:
                    438:                                ipaddr.sin_addr = *sin;
                    439:                                if (if_ifwithaddr((struct sockaddr *)&ipaddr) == 0)
                    440:                                        continue;
                    441:                                if (ipt->ipt_ptr + 8 > ipt->ipt_len)
                    442:                                        goto bad;
                    443:                                ipt->ipt_ptr += 4;
                    444:                                break;
                    445: 
                    446:                        default:
                    447:                                goto bad;
                    448:                        }
                    449:                        *(n_time *)sin = iptime();
                    450:                        ipt->ipt_ptr += 4;
                    451: #endif FATCHANCE
                    452:                }
                    453:        }
                    454:        return (0);
                    455: }
                    456: 
                    457: /*
                    458:  * Strip out IP options, at higher
                    459:  * level protocol in the kernel.
                    460:  * Second argument is buffer to which options
                    461:  * will be moved, and return value is their length.
                    462:  */
                    463: ip_stripoptions(ip, mopt)
                    464:        struct ip *ip;
                    465:        struct mbuf *mopt;
                    466: {
                    467:        register int i;
                    468:        register struct mbuf *m;
                    469:        int olen;
                    470: 
                    471:        olen = (ip->ip_hl<<2) - sizeof (struct ip);
                    472:        m = dtom(ip);
                    473:        ip++;
                    474:        if (mopt) {
                    475:                mopt->wptr = mopt->base + olen;
                    476:                mopt->rptr = mopt->base;
                    477:                bcopy((caddr_t)ip, mtod(m, caddr_t), (unsigned)olen);
                    478:        }
                    479:        i = BLEN(m) - (sizeof (struct ip) + olen);
                    480:        bcopy((caddr_t)ip+olen, (caddr_t)ip, (unsigned)i);
                    481:        m->wptr -= olen;
                    482: }
                    483: 
                    484: int    ipforwarding = 1;
                    485: extern ipprintfs;
                    486: /*
                    487:  * Forward a packet.  If some error occurs return the sender
                    488:  * and icmp packet.  Note we can't always generate a meaningful
                    489:  * icmp message because icmp doesn't have a large enough repetoire
                    490:  * of codes and types.
                    491:  */
                    492: ip_forward(ip)
                    493:        register struct ip *ip;
                    494: {
                    495:        struct mbuf *mopt;
                    496: 
                    497:        if(ipprintfs)
                    498:                printf("forward: src %x dst %x ttl %x\n", ip->ip_src,
                    499:                        ip->ip_dst, ip->ip_ttl);
                    500:        if (ipforwarding == 0) {
                    501:                m_freem(dtom(ip));
                    502:                return;
                    503:        }
                    504:        if (ip->ip_ttl < IPTTLDEC) {
                    505:                m_freem(dtom(ip));
                    506:                return;
                    507:        }
                    508:        ip->ip_ttl -= IPTTLDEC;
                    509:        mopt = m_get(M_DONTWAIT, MT_DATA);
                    510:        if (mopt == NULL) {
                    511:                m_freem(dtom(ip));
                    512:                return;
                    513:        }
                    514:        mopt->next = 0;
                    515: 
                    516:        ip_stripoptions(ip, mopt);
                    517: 
                    518:        ip_output(dtom(ip), mopt, IP_FORWARDING);
                    519: }

unix.superglobalmegacorp.com

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