/* * (C)opyright 1993,1994,1995 by Darren Reed. * * Redistribution and use in source and binary forms are permitted * provided that this notice is preserved and due credit is given * to the original author and the contributors. */ /* #pragma ident "@(#)solaris.c 1.12 6/5/96 (C) 1995 Darren Reed"*/ #pragma ident "$Id: solaris.c,v 2.0.1.4 1997/02/08 06:38:30 darrenr Exp $"; #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ipl.h" #include "ip_fil.h" #include "ip_compat.h" #include #include #include char _depends_on[] = "drv/ip"; extern int iplopen(), iplclose(), iplread(), iplioctl(); extern int iplattach(), ipldetach(); extern void copyout_mblk(), copyin_mblk(); int solattach(), soldetach(); extern struct filterstats frstats[]; extern kmutex_t ipl_mutex, ipf_mutex, ipfs_mutex; extern int fr_flags; static qif_t *qif_head = NULL; static int ipl_getinfo(), ipl_probe(), ipl_identify(), ipl_attach(); static int ipl_detach(); static struct cb_ops ipl_cb_ops = { iplopen, iplclose, nodev, /* strategy */ nodev, /* print */ nodev, /* dump */ iplread, nodev, /* write */ iplioctl, /* ioctl */ nodev, /* devmap */ nodev, /* mmap */ nodev, /* segmap */ nochpoll, /* poll */ ddi_prop_op, NULL, D_MTSAFE }; static struct dev_ops ipl_ops = { DEVO_REV, 0, ipl_getinfo, ipl_identify, ipl_probe, ipl_attach, ipl_detach, nodev, /* reset */ &ipl_cb_ops, (struct bus_ops *)0 }; extern struct mod_ops mod_driverops; static struct modldrv iplmod = { &mod_driverops, IPL_VERSION, &ipl_ops }; static struct modlinkage modlink1 = { MODREV_1, &iplmod, 0 }; static dev_info_t *ipf_dev_info = NULL; int _init() { #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipf_init()"); #endif return mod_install(&modlink1); } int _fini(void) { #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipf_fini()"); #endif return mod_remove(&modlink1); } int _info(struct modinfo *modinfop) { #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipf_info(%x)", modinfop); #endif return mod_info(&modlink1, modinfop); } static int ipl_probe(dev_info_t *dip) { #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipl_probe(%x)", dip); #endif return DDI_PROBE_SUCCESS; } static int ipl_identify(dev_info_t *dip) { #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipl_identify(%x)", dip); #endif if (strcmp(ddi_get_name(dip), "ipf") == 0) return (DDI_IDENTIFIED); return (DDI_NOT_IDENTIFIED); } static int ipl_attach(dev_info_t *dip, ddi_attach_cmd_t cmd) { int instance; #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipl_attach(%x,%x)", dip, cmd); #endif switch (cmd) { case DDI_ATTACH: instance = ddi_get_instance(dip); #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: attach ipf instace %d", instance); #endif if (ddi_create_minor_node(dip, "ipf", S_IFCHR, instance, DDI_PSEUDO, 0) == DDI_FAILURE) { ddi_remove_minor_node(dip, NULL); goto attach_failed; } ipf_dev_info = dip; sync(); /* * Initialize mutex's */ iplattach(); solattach(); cmn_err(CE_CONT, "IP Filter: attaching complete.\n"); return (DDI_SUCCESS); default: return (DDI_FAILURE); } attach_failed: /* * Use our own detach routine to toss * away any stuff we allocated above. */ (void) ipl_detach(dip, DDI_DETACH); return (DDI_FAILURE); } static int ipl_detach(dip, cmd) dev_info_t *dip; ddi_detach_cmd_t cmd; { int instance; #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipl_detach(%x,%x)", dip, cmd); #endif switch (cmd) { case DDI_DETACH: /* * Undo what we did in ipl_attach, freeing resources * and removing things we installed. The system * framework guarantees we are not active with this devinfo * node in any other entry points at this time. */ ddi_prop_remove_all(dip); instance = ddi_get_instance(dip); ddi_remove_minor_node(dip, NULL); sync(); if (!soldetach()) { cmn_err(CE_CONT, "IP Filter: detached\n"); return (DDI_SUCCESS); } default: return (DDI_FAILURE); } } static int ipl_getinfo(dip, infocmd, arg, result) dev_info_t *dip; ddi_info_cmd_t infocmd; void *arg, **result; { int error = DDI_FAILURE; #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: ipl_getinfo(%x,%x)", dip, infocmd); #endif switch (infocmd) { case DDI_INFO_DEVT2DEVINFO: *result = ipf_dev_info; error = DDI_SUCCESS; break; case DDI_INFO_DEVT2INSTANCE: *result = (void *)getminor((dev_t) arg); error = DDI_SUCCESS; break; default: break; } return (error); } /* * find the filter structure setup for this queue */ qif_t *qif_from_queue(q) queue_t *q; { qif_t *qif; for (qif = qif_head; qif; qif = qif->qf_next) if ((qif->qf_iptr == q->q_ptr) || (qif->qf_optr == q->q_ptr)) break; return qif; } /* * OK, this is pretty scrappy code, but then it's essentially just here for * debug purposes and that's it. Packets should not normally come through * here, and if they do, well, we would like to see as much information as * possible about them and what they claim to hold. */ void fr_donotip(out, qif, q, m, mt, ip, off) int out; qif_t *qif; queue_t *q; mblk_t *m, *mt; ip_t *ip; int off; { u_char *s, outb[256], *t; int i; outb[0] = '\0'; outb[1] = '\0'; outb[2] = '\0'; outb[3] = '\0'; s = ip ? (u_char *)ip : outb; if (!ip && (m == mt) && m->b_cont && (MTYPE(m) != M_DATA)) m = m->b_cont; printf("!IP %s:%d %x %x %x %d %x %x %x %d %d %x\n%02x%02x%02x%02x\n", qif ? qif->qf_name : "?", out, q, q ? q->q_ptr : NULL, q ? q->q_qinfo : NULL, mt->b_wptr - mt->b_rptr, m, mt, m->b_rptr, m->b_wptr - m->b_rptr, off, ip, *s, *(s+1), *(s+2), *(s+3)); if (m != mt) { i = 0; t = outb; s = mt->b_rptr; sprintf(t, "%d:", MTYPE(mt)); t += strlen(t); for (; (i < 100) && (s < mt->b_wptr); i++) { sprintf(t, "%02x%s", *s++, ((i & 3) == 3) ? " " : ""); t += ((i & 3) == 3) ? 3 : 2; } *t++ = '\n'; *t = '\0'; printf("%s", outb); } i = 0; t = outb; s = m->b_rptr; sprintf(t, "%d:", MTYPE(m)); t += strlen(t); for (; (i < 100) && (s < m->b_wptr); i++) { sprintf(t, "%02x%s", *s++, ((i & 3) == 3) ? " " : ""); t += ((i & 3) == 3) ? 3 : 2; } *t++ = '\n'; *t = '\0'; printf("%s", outb); } /* * find the first data mblk, if present, in the chain we're processing. Also * make a few sanity checks to try prevent the filter from causing a panic - * none of the nice IP sanity checks (including checksumming) should have been * done yet - dangerous! */ int fr_precheck(mp, q, qif, out) mblk_t **mp; queue_t *q; qif_t *qif; int out; { u_long lbuf[48]; mblk_t *m, *mp1, *mt = *mp; register ip_t *ip; int iphlen, hlen, len, err, mlen, off, synced = 0; #ifndef sparc u_short __iplen, __ipoff; #endif tryagain: /* * If there is only M_DATA for a packet going out, then any header * information (which would otherwise appear in an M_PROTO mblk before * the M_DATA) is prepended before the IP header. We need to set the * offset to account for this. - see MMM */ off = (out) ? qif->qf_hl : 0; /* * Find the first data block, count the data blocks in this chain and * the total amount of data. */ for (m = mt; m && (MTYPE(m) != M_DATA); m = m->b_cont) off = 0; /* Any non-M_DATA cancels the offset */ if (!m) return 0; /* No data blocks */ ip = (ip_t *)(m->b_rptr + off); /* MMM */ /* * We might have a 1st data block which is really M_PROTO, i.e. it is * only big enough for the link layer header */ while ((u_char *)ip >= m->b_wptr) { len = (u_char *)ip - m->b_wptr; if (!(m = m->b_cont)) return 0; /* not enough data for IP */ ip = (ip_t *)(m->b_rptr + len); } if ((off = (u_char *)ip - m->b_rptr)) m->b_rptr = (u_char *)ip; mlen = msgdsize(m); /* * Ok, the IP header isn't on a 32bit aligned address. To get around * this, we copy the data to an aligned buffer and work with that. */ if (!OK_32PTR(ip)) { len = MIN(mlen, sizeof(ip_t)); copyout_mblk(m, 0, lbuf, len); frstats[out].fr_pull[0]++; ip = (ip_t *)lbuf; } else len = m->b_wptr - (u_char *)ip; if (ip->ip_v != IPVERSION) { m->b_rptr -= off; if (!synced) { synced = 1; ipfsync(); goto tryagain; } fr_donotip(out, qif, q, m, mt, ip, off); frstats[out].fr_notip++; return (fr_flags & FF_BLOCKNONIP) ? -1 : 0; } hlen = iphlen = ip->ip_hl << 2; /* * Make hlen the total size of the IP header plus TCP/UDP/ICMP header * (if it is one of these three). */ if (!(ip->ip_off & 0x1fff)) switch (ip->ip_p) { case IPPROTO_TCP : hlen += sizeof(tcphdr_t); break; case IPPROTO_UDP : hlen += sizeof(udphdr_t); break; case IPPROTO_ICMP : hlen += sizeof(icmphdr_t); break; default : break; } /* * If we don't have enough data in the mblk or we haven't yet copied * enough (above), then copy some more. */ if ((hlen > len)) { len = MIN(hlen, sizeof(lbuf)); len = MIN(mlen, len); copyout_mblk(m, 0, lbuf, len); frstats[out].fr_pull[0]++; ip = (ip_t *)lbuf; } #ifndef sparc __iplen = ip->ip_len; ip->ip_len = ntohs(__iplen); __ipoff = ip->ip_off; ip->ip_off = ntohs(__ipoff); #endif if ((iphlen < sizeof(ip_t)) || (iphlen > (u_short)ip->ip_len) || (mlen < (u_short)ip->ip_len)) { /* * Bad IP packet or not enough data/data length mismatches */ m->b_rptr -= off; frstats[out].fr_bad++; return -1; } qif->qf_m = m; qif->qf_len = len; err = fr_check(ip, iphlen, qif->qf_ill, out, qif, q, mp); /* * Copy back the ip header data if it was changed, we haven't yet * freed the message and we aren't going to drop the packet. */ #ifndef sparc if (*mp) { ip->ip_len = __iplen; ip->ip_off = __ipoff; } #endif if (err == 1) { if (*mp && (ip == (ip_t *)lbuf)) { copyin_mblk(m, 0, lbuf, len); frstats[out].fr_pull[1]++; } err = 0; } m->b_rptr -= off; return err; } int fr_qin(q, mb) queue_t *q; mblk_t *mb; { int (*pnext)(), type, synced = 0; qif_t qfb, *qif; again: mutex_enter(&ipfs_mutex); while (!(qif = qif_from_queue(q))) { for (qif = qif_head; qif; qif = qif->qf_next) if (qif->qf_rqinfo == q->q_qinfo && qif->qf_inp) { pnext = qif->qf_inp; mutex_exit(&ipfs_mutex); frstats[0].fr_notip++; if (!synced) { ipfsync(); synced = 1; goto again; } /* fr_donotip(0, NULL, q, mb, mb, NULL, 0); */ return (*pnext)(q, mb); } mutex_exit(&ipfs_mutex); if (!synced) { ipfsync(); synced = 1; goto again; } cmn_err(CE_WARN, "IP Filter: dropped: fr_qin(%x,%x): type %x qif %x", q, mb, MTYPE(mb), qif); cmn_err(CE_CONT, "info %x next %x ptr %x fsrv %x bsrv %x\n", q->q_qinfo, q->q_next, q->q_ptr, q->q_nfsrv, q->q_nbsrv); cmn_err(CE_CONT, "IP Filter: info: putp %x srvp %x info %x\n", q->q_qinfo->qi_putp, q->q_qinfo->qi_srvp, q->q_qinfo->qi_infop); frstats[0].fr_drop++; freemsg(mb); return 0; } /* * So we can be more re-entrant. */ bcopy((char *)qif, (char *)&qfb, sizeof(*qif)); mutex_exit(&ipfs_mutex); qif = &qfb; pnext = qif->qf_inp; type = MTYPE(mb); if (type == M_DATA || type == M_PROTO || type == M_PCPROTO) if (fr_precheck(&mb, q, qif, 0)) { if (mb) freemsg(mb); return 0; } if (mb) { if (pnext) return (*pnext)(q, mb); cmn_err(CE_WARN, "IP Filter: inp NULL: qif %x %s q %x info %x", qif, qif->qf_name, q, q->q_qinfo); freemsg(mb); } return 0; } int fr_qout(q, mb) queue_t *q; mblk_t *mb; { int (*pnext)(), type, synced = 0; qif_t qfb, *qif; again: mutex_enter(&ipfs_mutex); if (!(qif = qif_from_queue(q))) { for (qif = qif_head; qif; qif = qif->qf_next) if (qif->qf_wqinfo == q->q_qinfo && qif->qf_outp) { pnext = qif->qf_outp; mutex_exit(&ipfs_mutex); frstats[1].fr_notip++; if (!synced) { ipfsync(); synced = 1; goto again; } /* fr_donotip(0, NULL, q, mb, mb, NULL, 0); */ return (*pnext)(q, mb); } mutex_exit(&ipfs_mutex); if (!synced) { ipfsync(); synced = 1; goto again; } cmn_err(CE_WARN, "IP Filter: dropped: fr_qout(%x,%x): type %x: qif %x", q, mb, MTYPE(mb), qif); cmn_err(CE_CONT, "info %x next %x ptr %x fsrv %x bsrv %x\n", q->q_qinfo, q->q_next, q->q_ptr, q->q_nfsrv, q->q_nbsrv); cmn_err(CE_CONT, "IP Filter: info: putp %x srvp %x info %x\n", q->q_qinfo->qi_putp, q->q_qinfo->qi_srvp, q->q_qinfo->qi_infop); if (q->q_nfsrv) cmn_err(CE_CONT, "nfsrv: info %x next %x ptr %x\n", q->q_nfsrv->q_qinfo, q->q_nfsrv->q_next, q->q_nfsrv->q_ptr); if (q->q_nbsrv) cmn_err(CE_CONT, "nbsrv: info %x next %x ptr %x\n", q->q_nbsrv->q_qinfo, q->q_nbsrv->q_next, q->q_nbsrv->q_ptr); frstats[1].fr_drop++; freemsg(mb); return 0; } /* * So we can be more re-entrant. */ bcopy((char *)qif, (char *)&qfb, sizeof(*qif)); mutex_exit(&ipfs_mutex); qif = &qfb; pnext = qif->qf_outp; type = MTYPE(mb); if (type == M_DATA || type == M_PROTO || type == M_PCPROTO) if (fr_precheck(&mb, q, qif, 1)) { if (mb) freemsg(mb); return 0; } if (mb) { if (pnext) return (*pnext)(q, mb); cmn_err(CE_WARN, "IP Filter: outp NULL: qif %x %s q %x info %x", qif, qif->qf_name, q, q->q_qinfo); freemsg(mb); } return 0; } /* * attach the packet filter to each interface that is defined as having an * IP address associated with it and save some of the info. for that struct * so we're not out of date as soon as te ill disappears - but we must sync * to be correct! */ int solattach() { queue_t *in, *out; qif_t *qif, *qf2; ill_t *il; for (il = ill_g_head; il; il = il->ill_next) { in = il->ill_rq; if (!in || !il->ill_wq) continue; out = il->ill_wq->q_next; mutex_enter(&ipfs_mutex); /* * Look for entry already setup for this device */ for (qif = qif_head; qif; qif = qif->qf_next) if (qif->qf_iptr == in->q_ptr && qif->qf_optr == out->q_ptr) break; if (qif) { mutex_exit(&ipfs_mutex); continue; } #ifdef IPFDEBUG cmn_err(CE_NOTE, "IP Filter: il %x ipt %x opt %x ipu %x opu %x i %x/%x", il, in->q_ptr, out->q_ptr, in->q_qinfo->qi_putp, out->q_qinfo->qi_putp, out->q_qinfo, in->q_qinfo); #endif qif = (qif_t *)KMALLOC(sizeof(*qif)); if (in->q_qinfo->qi_putp == fr_qin) { for (qf2 = qif_head; qf2; qf2 = qf2->qf_next) if (qf2->qf_rqinfo == in->q_qinfo) { qif->qf_inp = qf2->qf_inp; break; } if (!qf2) { #ifdef IPFDEBUG cmn_err(CE_WARN, "IP Filter: rq:%s put %x qi %x", il->ill_name, in->q_qinfo->qi_putp, in->q_qinfo); #endif mutex_exit(&ipfs_mutex); KFREE(qif); continue; } } else qif->qf_inp = in->q_qinfo->qi_putp; if (out->q_qinfo->qi_putp == fr_qout) { for (qf2 = qif_head; qf2; qf2 = qf2->qf_next) if (qf2->qf_wqinfo == out->q_qinfo) { qif->qf_outp = qf2->qf_outp; break; } if (!qf2) { #ifdef IPFDEBUG cmn_err(CE_WARN, "IP Filter: wq:%s put %x qi %x", il->ill_name, out->q_qinfo->qi_putp, out->q_qinfo); #endif mutex_exit(&ipfs_mutex); KFREE(qif); continue; } } else qif->qf_outp = out->q_qinfo->qi_putp; qif->qf_ill = il; qif->qf_iptr = in->q_ptr; qif->qf_optr = out->q_ptr; qif->qf_hl = il->ill_hdr_length; strncpy(qif->qf_name, il->ill_name, sizeof(qif->qf_name)); qif->qf_name[sizeof(qif->qf_name) - 1] = '\0'; qif->qf_wqinfo = out->q_qinfo; qif->qf_rqinfo = in->q_qinfo; qif->qf_next = qif_head; qif_head = qif; in->q_qinfo->qi_putp = fr_qin; out->q_qinfo->qi_putp = fr_qout; mutex_exit(&ipfs_mutex); cmn_err(CE_CONT, "IP Filter: attach to [%s,%d]\n", qif->qf_name, il->ill_ppa); } if (!qif_head) { cmn_err(CE_CONT, "IP Filter: not attached to any interfaces\n"); return -1; } return 0; } /* * look for bad consistancies between the list of interfaces the filter knows * about and those which are currently configured. */ int ipfsync() { register struct frentry *f, **fp; register qif_t *qif, **qp; register ill_t *il; mutex_enter(&ipfs_mutex); for (qp = &qif_head; (qif = *qp); ) { for (il = ill_g_head; il; il = il->ill_next) if ((qif->qf_ill == il) && !strcmp(qif->qf_name, il->ill_name)) { mblk_t *m = il->ill_hdr_mp; qif->qf_hl = il->ill_hdr_length; if (m && qif->qf_hl != (m->b_wptr - m->b_rptr)) printf("ILL Header Length Mismatch\n"); break; } if (il) { qp = &qif->qf_next; continue; } cmn_err(CE_CONT, "IP Filter: detaching [%s]\n", qif->qf_name); *qp = qif->qf_next; /* * Delete any rules directly associated with this interface */ mutex_enter(&ipf_mutex); for (fp = &ipfilter[0][0]; (f = *fp); ) if ((void *)f->fr_ifa == (void *)qif->qf_ill) { *fp = f->fr_next; KFREE(f); } else fp = &f->fr_next; for (fp = &ipfilter[1][0]; (f = *fp); ) if ((void *)f->fr_ifa == (void *)qif->qf_ill) { *fp = f->fr_next; KFREE(f); } else fp = &f->fr_next; mutex_exit(&ipf_mutex); KFREE(qif); qif = *qp; } mutex_exit(&ipfs_mutex); return solattach(); } /* * unhook the IP filter from all defined interfaces with IP addresses */ int soldetach() { queue_t *in, *out; qif_t *qif, *qf2, **qp; ill_t *il; mutex_enter(&ipfs_mutex); /* * Make two passes, first get rid of all the unknown devices, next * unlink known devices. */ for (qp = &qif_head; (qif = *qp); ) { for (il = ill_g_head; il; il = il->ill_next) if (qif->qf_ill == il) break; if (il) { qp = &qif->qf_next; continue; } cmn_err(CE_CONT, "IP Filter: removing [%s]\n", qif->qf_name); *qp = qif->qf_next; KFREE(qif); } while ((qif = qif_head)) { qif_head = qif->qf_next; for (il = ill_g_head; il; il = il->ill_next) if (qif->qf_ill == il) break; if (il) { in = il->ill_rq; out = il->ill_wq->q_next; printf("IP Filter: detaching [%s,%d]\n", qif->qf_name, il->ill_ppa); in->q_qinfo->qi_putp = qif->qf_inp; /* * and change back if something is still interested * in filtering (read side) on this interface. */ for (qf2 = qif_head; qf2; qf2 = qf2->qf_next) if (qf2->qf_rqinfo == in->q_qinfo) { in->q_qinfo->qi_putp = fr_qin; break; } /* * and the write queue... */ out->q_qinfo->qi_putp = qif->qf_outp; for (qf2 = qif_head; qf2; qf2 = qf2->qf_next) if (qf2->qf_wqinfo == out->q_qinfo) { out->q_qinfo->qi_putp = fr_qout; break; } } KFREE(qif); } mutex_exit(&ipfs_mutex); return ipldetach(); } printire(ire) ire_t *ire; { printf("ire: ll_hdr_mp %x rfq %x stq %x src_addr %x max_frag %d\n", ire->ire_ll_hdr_mp, ire->ire_rfq, ire->ire_stq, ire->ire_src_addr, ire->ire_max_frag); printf("ire: mask %x addr %x gateway_addr %x type %d\n", ire->ire_mask, ire->ire_addr, ire->ire_gateway_addr, ire->ire_type); printf("ire: ll_hdr_length %d ll_hdr_saved_mp %x\n", ire->ire_ll_hdr_length, ire->ire_ll_hdr_saved_mp); } int ipfr_fastroute(qf, ip, mb, mpp, fin, fdp) qif_t *qf; ip_t *ip; mblk_t *mb, **mpp; fr_info_t *fin; frdest_t *fdp; { mblk_t *mp = NULL; struct in_addr dst; ire_t *ir, *dir; int hlen = 0; u_char *s; queue_t *q = NULL; #ifndef sparc u_short __iplen, __ipoff; /* * If this is a duplicate mblk then we want ip to point at that * data, not the original, if and only if it is already pointing at * the current mblk data. */ if (ip == (ip_t *)qf->qf_m->b_rptr && qf->qf_m != mb) ip = (ip_t *)mb->b_rptr; /* * In fr_precheck(), we modify ip_len and ip_off in an aligned data * area. However, we only need to change it back if we didn't copy * the IP header data out. */ __iplen = (u_short)ip->ip_len, __ipoff = (u_short)ip->ip_off; ip->ip_len = htons(__iplen); ip->ip_off = htons(__ipoff); #endif if (ip != (ip_t *)mb->b_rptr) { copyin_mblk(mb, 0, ip, qf->qf_len); frstats[fin->fin_out].fr_pull[1]++; } /* * If there is another M_PROTO, we don't want it */ if (*mpp != mb) { (*mpp)->b_cont = NULL; freemsg(*mpp); } ir = (ire_t *)fdp->fd_ifp; if (fdp->fd_ip.s_addr) dst = fdp->fd_ip; else dst = fin->fin_fi.fi_dst; if (dir = ire_lookup(dst.s_addr)) if (!dir->ire_ll_hdr_mp || !dir->ire_ll_hdr_length) dir = NULL; if (!ir) ir = dir; if (ir && dir) { if ((mp = dir->ire_ll_hdr_mp)) { hlen = dir->ire_ll_hdr_length; s = mb->b_rptr; if (hlen && (s - mb->b_datap->db_base) >= hlen) { s -= hlen; mb->b_rptr = (u_char *)s; bcopy((char *)mp->b_rptr, (char *)s, hlen); } else { mblk_t *mp2; mp2 = copyb(mp); if (!mp2) goto bad_fastroute; mp2->b_cont = mb; mb = mp2; } } if (ir->ire_stq) q = ir->ire_stq; else if (ir->ire_rfq) q = WR(ir->ire_rfq); if (q) { putnext(q, mb); return 0; } } bad_fastroute: return -1; } void copyout_mblk(m, off, buf, len) mblk_t *m; int off, len; char *buf; { char *s, *bp = buf; int mlen, olen, clen; for (; m && len; m = m->b_cont) { if (MTYPE(m) != M_DATA) continue; s = m->b_rptr; mlen = (char *)m->b_wptr - s; olen = MIN(off, mlen); if ((olen == mlen) || (olen < off)) { off -= olen; continue; } else if (olen) { off -= olen; s += olen; mlen -= olen; } clen = MIN(mlen, len); bcopy(s, bp, clen); len -= clen; bp += clen; } } void copyin_mblk(m, off, buf, len) mblk_t *m; int off, len; char *buf; { char *s, *bp = buf; int mlen, olen, clen; for (; m && len; m = m->b_cont) { if (MTYPE(m) != M_DATA) continue; s = m->b_rptr; mlen = (char *)m->b_wptr - s; olen = MIN(off, mlen); if ((olen == mlen) || (olen < off)) { off -= olen; continue; } else if (olen) { off -= olen; s += olen; mlen -= olen; } clen = MIN(mlen, len); bcopy(bp, s, clen); len -= clen; bp += clen; } }