diff options
author | Alexander Yurchenko <grange@cvs.openbsd.org> | 2004-05-28 08:16:24 +0000 |
---|---|---|
committer | Alexander Yurchenko <grange@cvs.openbsd.org> | 2004-05-28 08:16:24 +0000 |
commit | c2888a975d59102b19a3bfe63b7e9ac0f50fe419 (patch) | |
tree | 012dd3bdc3c642e4288a59bbaa7d165fc22a7b7d /sys/net | |
parent | 427354c6146786ff2e7c23b0c93956082f11ff1a (diff) |
bpf device cloning.
Now to have more bpf devices just add device nodes in /dev,
no need to recompile kernel anymore.
Code from form@pdp-11.org.ru, some help from markus@.
ok markus@ canacar@ deraadt@
Diffstat (limited to 'sys/net')
-rw-r--r-- | sys/net/bpf.c | 103 | ||||
-rw-r--r-- | sys/net/bpfdesc.h | 4 |
2 files changed, 72 insertions, 35 deletions
diff --git a/sys/net/bpf.c b/sys/net/bpf.c index 94a8fb9b5bb..8e6c216e67f 100644 --- a/sys/net/bpf.c +++ b/sys/net/bpf.c @@ -1,4 +1,4 @@ -/* $OpenBSD: bpf.c,v 1.46 2004/05/25 17:36:49 canacar Exp $ */ +/* $OpenBSD: bpf.c,v 1.47 2004/05/28 08:16:23 grange Exp $ */ /* $NetBSD: bpf.c,v 1.33 1997/02/21 23:59:35 thorpej Exp $ */ /* @@ -72,11 +72,10 @@ int bpf_maxbufsize = BPF_MAXBUFSIZE; /* * bpf_iflist is the list of interfaces; each corresponds to an ifnet - * bpf_dtab holds the descriptors, indexed by minor device # + * bpf_d_list is the list of descriptors */ struct bpf_if *bpf_iflist; -struct bpf_d *bpf_dtab; -int nbpfilter; +LIST_HEAD(, bpf_d) bpf_d_list; int bpf_allocbufs(struct bpf_d *); void bpf_freed(struct bpf_d *); @@ -97,6 +96,10 @@ void bpf_reset_d(struct bpf_d *); void filt_bpfrdetach(struct knote *); int filt_bpfread(struct knote *, long); +struct bpf_d *bpfilter_lookup(int); +struct bpf_d *bpfilter_create(int); +void bpfilter_destroy(struct bpf_d *); + int bpf_movein(uio, linktype, mp, sockp, filter) struct uio *uio; @@ -306,19 +309,7 @@ void bpfilterattach(n) int n; { - int i; - - bpf_dtab = malloc(n * sizeof(*bpf_dtab), M_DEVBUF, M_NOWAIT); - if (!bpf_dtab) - return; - nbpfilter = n; - bzero(bpf_dtab, n * sizeof(*bpf_dtab)); - /* - * Mark all the descriptors free if this hasn't been done. - */ - if (!D_ISFREE(&bpf_dtab[0])) - for (i = 0; i < nbpfilter; ++i) - D_MARKFREE(&bpf_dtab[i]); + LIST_INIT(&bpf_d_list); } /* @@ -335,18 +326,17 @@ bpfopen(dev, flag, mode, p) { struct bpf_d *d; - if (minor(dev) >= nbpfilter) + /* create on demand */ + if ((d = bpfilter_create(minor(dev))) == NULL) return (ENXIO); /* * Each minor can be opened by only one process. If the requested * minor is in use, return EBUSY. */ - d = &bpf_dtab[minor(dev)]; if (!D_ISFREE(d)) return (EBUSY); /* Mark "free" and do most initialization. */ - bzero((char *)d, sizeof(*d)); d->bd_bufsize = bpf_bufsize; d->bd_sig = SIGIO; @@ -367,9 +357,10 @@ bpfclose(dev, flag, mode, p) int mode; struct proc *p; { - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; int s; + d = bpfilter_lookup(minor(dev)); s = splimp(); if (d->bd_bif) bpf_detachd(d); @@ -400,10 +391,11 @@ bpfread(dev, uio, ioflag) struct uio *uio; int ioflag; { - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; int error; int s; + d = bpfilter_lookup(minor(dev)); if (d->bd_bif == 0) return (ENXIO); @@ -538,12 +530,13 @@ bpfwrite(dev, uio, ioflag) struct uio *uio; int ioflag; { - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; struct ifnet *ifp; struct mbuf *m; int error, s; struct sockaddr_storage dst; + d = bpfilter_lookup(minor(dev)); if (d->bd_bif == 0) return (ENXIO); @@ -620,9 +613,10 @@ bpfioctl(dev, cmd, addr, flag, p) int flag; struct proc *p; { - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; int s, error = 0; + d = bpfilter_lookup(minor(dev)); if (d->bd_locked && suser(p, 0) != 0) { /* list of allowed ioctls when locked and not root */ switch (cmd) { @@ -1039,7 +1033,7 @@ bpfpoll(dev, events, p) /* * An imitation of the FIONREAD ioctl code. */ - d = &bpf_dtab[minor(dev)]; + d = bpfilter_lookup(minor(dev)); s = splimp(); if (d->bd_hlen == 0 && (!d->bd_immediate || d->bd_slen == 0)) { revents = 0; /* no data waiting */ @@ -1060,10 +1054,11 @@ struct filterops bpfread_filtops = int bpfkqfilter(dev_t dev,struct knote *kn) { - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; struct klist *klist; int s; + d = bpfilter_lookup(minor(dev)); switch (kn->kn_filter) { case EVFILT_READ: klist = &d->bd_sel.si_note; @@ -1087,9 +1082,10 @@ void filt_bpfrdetach(struct knote *kn) { dev_t dev = (dev_t)((u_long)kn->kn_hook); - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; int s = splimp(); + d = bpfilter_lookup(minor(dev)); SLIST_REMOVE(&d->bd_sel.si_note, kn, knote, kn_selnext); splx(s); } @@ -1098,11 +1094,12 @@ int filt_bpfread(struct knote *kn, long hint) { dev_t dev = (dev_t)((u_long)kn->kn_hook); - struct bpf_d *d = &bpf_dtab[minor(dev)]; + struct bpf_d *d; int res, s; kn->kn_data = 0; + d = bpfilter_lookup(minor(dev)); s = splimp(); res = d->bd_hlen != 0 || (d->bd_immediate && d->bd_slen != 0); splx(s); @@ -1327,7 +1324,7 @@ bpf_freed(d) if (d->bd_wfilter) free((caddr_t)d->bd_wfilter, M_DEVBUF); - D_MARKFREE(d); + bpfilter_destroy(d); } /* @@ -1373,7 +1370,7 @@ bpfdetach(ifp) { struct bpf_if *bp, *nbp, **pbp = &bpf_iflist; struct bpf_d *bd; - int maj, mn; + int maj; for (bp = bpf_iflist; bp; bp = nbp) { nbp= bp->bif_next; @@ -1385,16 +1382,20 @@ bpfdetach(ifp) if (cdevsw[maj].d_open == bpfopen) break; - for (bd = bp->bif_dlist; bd; bd = bp->bif_dlist) + for (bd = bp->bif_dlist; bd; bd = bp->bif_dlist) { + struct bpf_d *d; + /* * Locate the minor number and nuke the vnode * for any open instance. */ - for (mn = 0; mn < nbpfilter; mn++) - if (&bpf_dtab[mn] == bd) { - vdevgone(maj, mn, mn, VCHR); + LIST_FOREACH(d, &bpf_d_list, bd_list) + if (d == bd) { + vdevgone(maj, d->bd_unit, + d->bd_unit, VCHR); break; } + } free(bp, M_DEVBUF); } else @@ -1437,3 +1438,37 @@ bpf_sysctl(int *name, u_int namelen, void *oldp, size_t *oldlenp, void *newp, } return (0); } + +struct bpf_d * +bpfilter_lookup(int unit) +{ + struct bpf_d *bd; + + LIST_FOREACH(bd, &bpf_d_list, bd_list) + if (bd->bd_unit == unit) + return (bd); + return (NULL); +} + +struct bpf_d * +bpfilter_create(int unit) +{ + struct bpf_d *bd; + + if ((bd = bpfilter_lookup(unit)) != NULL) + return (bd); + if ((bd = malloc(sizeof(*bd), M_DEVBUF, M_NOWAIT)) != NULL) { + bzero(bd, sizeof(*bd)); + bd->bd_unit = unit; + D_MARKFREE(bd); + LIST_INSERT_HEAD(&bpf_d_list, bd, bd_list); + } + return (bd); +} + +void +bpfilter_destroy(struct bpf_d *bd) +{ + LIST_REMOVE(bd, bd_list); + free(bd, M_DEVBUF); +} diff --git a/sys/net/bpfdesc.h b/sys/net/bpfdesc.h index 57a5081cb13..4087eb2685d 100644 --- a/sys/net/bpfdesc.h +++ b/sys/net/bpfdesc.h @@ -1,4 +1,4 @@ -/* $OpenBSD: bpfdesc.h,v 1.12 2004/05/08 20:54:13 canacar Exp $ */ +/* $OpenBSD: bpfdesc.h,v 1.13 2004/05/28 08:16:23 grange Exp $ */ /* $NetBSD: bpfdesc.h,v 1.11 1995/09/27 18:30:42 thorpej Exp $ */ /* @@ -84,6 +84,8 @@ struct bpf_d { uid_t bd_sigeuid; /* euid for process that set pgid */ u_int bd_ref; /* reference count */ struct selinfo bd_sel; /* bsd select info */ + int bd_unit; /* logical unit number */ + LIST_ENTRY(bpf_d) bd_list; /* descriptor list */ }; /* |