summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/net/if_bridge.c236
-rw-r--r--sys/net/if_bridge.h55
-rw-r--r--sys/sys/sockio.h8
-rw-r--r--usr.sbin/brconfig/brconfig.c144
4 files changed, 336 insertions, 107 deletions
diff --git a/sys/net/if_bridge.c b/sys/net/if_bridge.c
index 72646e8a9de..e1eb0d47fee 100644
--- a/sys/net/if_bridge.c
+++ b/sys/net/if_bridge.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_bridge.c,v 1.2 1999/02/27 21:19:27 jason Exp $ */
+/* $OpenBSD: if_bridge.c,v 1.3 1999/03/01 04:44:44 jason Exp $ */
/*
* Copyright (c) 1999 Jason L. Wright (jason@thought.net)
@@ -67,6 +67,13 @@
#endif
#define BRIDGE_RTABLE_MASK (BRIDGE_RTABLE_SIZE - 1)
+/*
+ * Maximum number of addresses to cache
+ */
+#ifndef BRIDGE_RTABLE_MAX
+#define BRIDGE_RTABLE_MAX 100
+#endif
+
extern int ifqmaxlen;
/*
@@ -74,7 +81,7 @@ extern int ifqmaxlen;
*/
struct bridge_iflist {
LIST_ENTRY(bridge_iflist) next; /* next in list */
- struct ifnet *ifp; /* member interface */
+ struct ifnet *ifp; /* member interface */
u_int32_t bif_flags; /* member flags */
};
@@ -83,17 +90,19 @@ struct bridge_iflist {
*/
struct bridge_rtnode {
LIST_ENTRY(bridge_rtnode) brt_next; /* next in list */
- struct ifnet *brt_if; /* destination ifs */
+ struct ifnet *brt_if; /* destination ifs */
u_int32_t brt_age; /* age counter */
- struct ether_addr brt_addr; /* destination addr */
+ struct ether_addr brt_addr; /* dst addr */
};
/*
* Software state for each bridge
*/
struct bridge_softc {
- struct ifnet sc_if; /* the interface */
+ struct ifnet sc_if; /* the interface */
u_int32_t sc_brtageidx; /* route age index */
+ u_int32_t sc_brtmax; /* max # addresses */
+ u_int32_t sc_brtcnt; /* current # addrs */
LIST_HEAD(, bridge_iflist) sc_iflist; /* interface list */
LIST_HEAD(bridge_rthead, bridge_rtnode) *sc_rts;/* hash table */
};
@@ -107,9 +116,11 @@ struct mbuf * bridge_broadcast __P((struct bridge_softc *sc,
struct ifnet *, struct ether_header *, struct mbuf *));
void bridge_stop __P((struct bridge_softc *));
void bridge_init __P((struct bridge_softc *));
+int bridge_bifconf __P((struct bridge_softc *, struct ifbifconf *));
-int bridge_rtfind __P((struct bridge_softc *, struct ifbrtreq *));
+int bridge_rtfind __P((struct bridge_softc *, struct ifbaconf *));
void bridge_rtage __P((void *));
+void bridge_rttrim __P((struct bridge_softc *));
void bridge_rtdelete __P((struct bridge_softc *, struct ifnet *));
struct ifnet * bridge_rtupdate __P((struct bridge_softc *,
struct ether_addr *, struct ifnet *ifp));
@@ -137,12 +148,12 @@ void
bridgeattach(unused)
int unused;
{
- register int i;
+ int i;
struct ifnet *ifp;
for (i = 0; i < NBRIDGE; i++) {
+ bridgectl[i].sc_brtmax = BRIDGE_RTABLE_MAX;
LIST_INIT(&bridgectl[i].sc_iflist);
-
ifp = &bridgectl[i].sc_if;
sprintf(ifp->if_xname, "bridge%d", i);
ifp->if_softc = &bridgectl[i];
@@ -153,7 +164,6 @@ bridgeattach(unused)
ifp->if_type = IFT_PROPVIRTUAL;
ifp->if_snd.ifq_maxlen = ifqmaxlen;
ifp->if_hdrlen = sizeof(struct ether_header);
-
if_attach(ifp);
}
}
@@ -168,8 +178,10 @@ bridge_ioctl(ifp, cmd, data)
struct ifnet *ifs;
struct bridge_softc *sc = (struct bridge_softc *)ifp->if_softc;
struct ifbreq *req = (struct ifbreq *)data;
- struct ifbrtreq *brtreq = (struct ifbrtreq *)data;
- int error = 0, s, i;
+ struct ifbaconf *baconf = (struct ifbaconf *)data;
+ struct ifbcachereq *bcachereq = (struct ifbcachereq *)data;
+ struct ifbifconf *bifconf = (struct ifbifconf *)data;
+ int error = 0, s;
struct bridge_iflist *p;
s = splimp();
@@ -184,7 +196,7 @@ bridge_ioctl(ifp, cmd, data)
/*
* Get pointer to ifnet structure for the named interface.
*/
- ifs = ifunit(req->ifsname);
+ ifs = ifunit(req->ifbr_ifsname);
if (ifs == NULL) { /* no such interface */
error = ENOENT;
break;
@@ -249,7 +261,7 @@ bridge_ioctl(ifp, cmd, data)
p = LIST_FIRST(&sc->sc_iflist);
while (p != NULL) {
- if (strncmp(p->ifp->if_xname, req->ifsname,
+ if (strncmp(p->ifp->if_xname, req->ifbr_ifsname,
sizeof(p->ifp->if_xname)) == 0) {
/*
* Remove the pointer back to us.
@@ -278,34 +290,22 @@ bridge_ioctl(ifp, cmd, data)
break;
}
break;
- case SIOCBRDGIDX:
- /*
- * Search through list looking for right element.
- */
- i = 0;
- p = LIST_FIRST(&sc->sc_iflist);
- while (p != NULL && i < req->index) {
- i++;
- p = LIST_NEXT(p, next);
- }
-
- /*
- * Make sure we didn't go off the end of the list.
- */
- if (p == NULL || req->index != i) {
- error = ENOENT;
- break;
- }
- /* found it */
- strncpy(req->ifsname, p->ifp->if_xname, sizeof(req->ifsname)-1);
- req->ifsname[sizeof(req->ifsname)-1] = '\0';
+ case SIOCBRDGIFS:
+ error = bridge_bifconf(sc, bifconf);
break;
- case SIOCBRDGRT:
+ case SIOCBRDGRTS:
if ((ifp->if_flags & IFF_RUNNING) == 0) {
error = ENETDOWN;
break;
}
- error = bridge_rtfind(sc, brtreq);
+ error = bridge_rtfind(sc, baconf);
+ break;
+ case SIOCBRDGGCACHE:
+ bcachereq->ifbc_size = sc->sc_brtmax;
+ break;
+ case SIOCBRDGSCACHE:
+ sc->sc_brtmax = bcachereq->ifbc_size;
+ bridge_rttrim(sc);
break;
case SIOCSIFFLAGS:
if ((ifp->if_flags & IFF_UP) == IFF_UP)
@@ -322,6 +322,51 @@ bridge_ioctl(ifp, cmd, data)
return (error);
}
+int
+bridge_bifconf(sc, bifc)
+ struct bridge_softc *sc;
+ struct ifbifconf *bifc;
+{
+ struct bridge_iflist *p;
+ u_int32_t total, i;
+ int error;
+ struct ifbreq breq;
+
+ p = LIST_FIRST(&sc->sc_iflist);
+ while (p != NULL) {
+ total++;
+ p = LIST_NEXT(p, next);
+ }
+
+ if (bifc->ifbic_len == 0) {
+ bifc->ifbic_len = total * sizeof(struct ifbreq);
+ return (0);
+ }
+
+ p = LIST_FIRST(&sc->sc_iflist);
+ i = 0;
+ while (p != NULL && bifc->ifbic_len > sizeof(breq)) {
+ strncpy(breq.ifbr_name, sc->sc_if.if_xname,
+ sizeof (breq.ifbr_name));
+ breq.ifbr_name[sizeof(breq.ifbr_name)-1] = '\0';
+ strncpy(breq.ifbr_ifsname, p->ifp->if_xname,
+ sizeof (breq.ifbr_ifsname));
+ breq.ifbr_ifsname[sizeof(breq.ifbr_ifsname)-1] = '\0';
+ breq.ifbr_ifsflags = p->bif_flags;
+ error = copyout((caddr_t)&breq,
+ (caddr_t)(bifc->ifbic_req + i), sizeof(breq));
+ if (error) {
+ bifc->ifbic_len = i * sizeof(breq);
+ return (error);
+ }
+ p = LIST_NEXT(p, next);
+ i++;
+ bifc->ifbic_len -= sizeof(breq);
+ }
+ bifc->ifbic_len = i * sizeof(breq);
+ return (0);
+}
+
void
bridge_init(sc)
struct bridge_softc *sc;
@@ -385,6 +430,7 @@ bridge_stop(sc)
p = LIST_NEXT(n, brt_next);
LIST_REMOVE(n, brt_next);
free(n, M_DEVBUF);
+ sc->sc_brtcnt--;
n = p;
}
}
@@ -812,6 +858,10 @@ bridge_rtupdate(sc, ea, ifp)
h = bridge_hash(ea);
p = LIST_FIRST(&sc->sc_rts[h]);
if (p == NULL) {
+ if (sc->sc_brtcnt >= sc->sc_brtmax) {
+ splx(s);
+ return (NULL);
+ }
p = (struct bridge_rtnode *)
malloc(sizeof(struct bridge_rtnode),
M_DEVBUF, M_NOWAIT);
@@ -824,6 +874,7 @@ bridge_rtupdate(sc, ea, ifp)
p->brt_if = ifp;
p->brt_age = 1;
LIST_INSERT_HEAD(&sc->sc_rts[h], p, brt_next);
+ sc->sc_brtcnt++;
splx(s);
return (ifp);
}
@@ -841,6 +892,10 @@ bridge_rtupdate(sc, ea, ifp)
}
if (dir > 0) {
+ if (sc->sc_brtcnt >= sc->sc_brtmax) {
+ splx(s);
+ return (NULL);
+ }
p = (struct bridge_rtnode *)
malloc(sizeof(struct bridge_rtnode),
M_DEVBUF, M_NOWAIT);
@@ -853,11 +908,16 @@ bridge_rtupdate(sc, ea, ifp)
p->brt_if = ifp;
p->brt_age = 1;
LIST_INSERT_BEFORE(q, p, brt_next);
+ sc->sc_brtcnt++;
splx(s);
return (ifp);
}
if (p == NULL) {
+ if (sc->sc_brtcnt >= sc->sc_brtmax) {
+ splx(s);
+ return (NULL);
+ }
p = (struct bridge_rtnode *)
malloc(sizeof(struct bridge_rtnode),
M_DEVBUF, M_NOWAIT);
@@ -870,6 +930,7 @@ bridge_rtupdate(sc, ea, ifp)
p->brt_if = ifp;
p->brt_age = 1;
LIST_INSERT_AFTER(q, p, brt_next);
+ sc->sc_brtcnt++;
splx(s);
return (ifp);
}
@@ -954,6 +1015,60 @@ bridge_hash(addr)
return (c & BRIDGE_RTABLE_MASK);
}
+/*
+ * Trim the routing table so that we've got a number of routes
+ * less than or equal to the maximum.
+ */
+void
+bridge_rttrim(sc)
+ struct bridge_softc *sc;
+{
+ struct bridge_rtnode *n, *p;
+ int s, i;
+
+ s = splhigh();
+ if (sc->sc_rts == NULL) {
+ splx(s);
+ return;
+ }
+
+ /*
+ * Make sure we have to trim the address table
+ */
+ if (sc->sc_brtcnt <= sc->sc_brtmax) {
+ splx(s);
+ return;
+ }
+
+ /*
+ * Force an aging cycle, this might trim enough addresses.
+ */
+ splx(s);
+ bridge_rtage(sc);
+ s = splhigh();
+
+ if (sc->sc_brtcnt <= sc->sc_brtmax) {
+ splx(s);
+ return;
+ }
+
+ for (i = 0; i < BRIDGE_RTABLE_SIZE; i++) {
+ n = LIST_FIRST(&sc->sc_rts[i]);
+ while (n != NULL) {
+ p = LIST_NEXT(n, brt_next);
+ LIST_REMOVE(n, brt_next);
+ sc->sc_brtcnt--;
+ free(n, M_DEVBUF);
+ n = p;
+ if (sc->sc_brtcnt <= sc->sc_brtmax) {
+ splx(s);
+ return;
+ }
+ }
+ }
+ splx(s);
+}
+
void
bridge_rtage(vsc)
void *vsc;
@@ -987,6 +1102,7 @@ bridge_rtage(vsc)
n->brt_addr.ether_addr_octet[5]);
#endif
LIST_REMOVE(n, brt_next);
+ sc->sc_brtcnt--;
free(n, M_DEVBUF);
n = p;
}
@@ -1022,6 +1138,7 @@ bridge_rtdelete(sc, ifp)
if (n->brt_if == ifp) { /* found one */
p = LIST_NEXT(n, brt_next);
LIST_REMOVE(n, brt_next);
+ sc->sc_brtcnt--;
free(n, M_DEVBUF);
n = p;
}
@@ -1034,41 +1151,62 @@ bridge_rtdelete(sc, ifp)
}
/*
- * Find the route for a given index value in rq->index.
- * Returns 0 on success, otherwise an error value.
+ * Gather all of the routes for this interface.
*/
int
-bridge_rtfind(sc, rq)
+bridge_rtfind(sc, baconf)
struct bridge_softc *sc;
- struct ifbrtreq *rq;
+ struct ifbaconf *baconf;
{
- int i, s;
+ int i, s, error;
u_int32_t cnt;
struct bridge_rtnode *n;
+ struct ifbareq bareq;
s = splhigh();
+
if (sc->sc_rts == NULL) {
+ baconf->ifbac_len = 0;
+ splx(s);
+ return (0);
+ }
+
+ if (baconf->ifbac_len == 0) {
+ baconf->ifbac_len = sc->sc_brtcnt * sizeof(struct ifbareq);
splx(s);
- return (ENOENT);
+ return (0);
}
- for (cnt = 0, i = 0; i < BRIDGE_RTABLE_SIZE; i++) {
+ for (i = 0, cnt = 0; i < BRIDGE_RTABLE_SIZE; i++) {
n = LIST_FIRST(&sc->sc_rts[i]);
while (n != NULL) {
- if (cnt == rq->index) {
- bcopy(&n->brt_addr, &rq->dst, sizeof(rq->dst));
- bcopy(n->brt_if->if_xname, rq->ifsname,
- IFNAMSIZ);
- rq->age = n->brt_age;
+ if (baconf->ifbac_len < sizeof(struct ifbareq)) {
+ baconf->ifbac_len = cnt *
+ sizeof(struct ifbareq);
splx(s);
return (0);
}
+ bcopy(n->brt_if->if_xname, bareq.ifba_name,
+ sizeof(bareq.ifba_name));
+ bcopy(&n->brt_addr, &bareq.ifba_dst,
+ sizeof(bareq.ifba_dst));
+ bareq.ifba_age = n->brt_age;
+ error = copyout((caddr_t)&bareq,
+ (caddr_t)(baconf->ifbac_req + cnt), sizeof(bareq));
+ if (error) {
+ baconf->ifbac_len = cnt *
+ sizeof(struct ifbareq);
+ splx(s);
+ return (error);
+ }
n = LIST_NEXT(n, brt_next);
cnt++;
}
}
+
+ baconf->ifbac_len = cnt * sizeof(struct ifbareq);
splx(s);
- return (ENOENT);
+ return (0);
}
#if defined(INET) && (defined(IPFILTER) || defined(IPFILTER_LKM))
diff --git a/sys/net/if_bridge.h b/sys/net/if_bridge.h
index d127c091604..ec873697c1a 100644
--- a/sys/net/if_bridge.h
+++ b/sys/net/if_bridge.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_bridge.h,v 1.1 1999/02/26 17:01:32 jason Exp $ */
+/* $OpenBSD: if_bridge.h,v 1.2 1999/03/01 04:44:44 jason Exp $ */
/*
* Copyright (c) 1999 Jason L. Wright (jason@thought.net)
@@ -32,23 +32,54 @@
*/
/*
- * Bridge control request: (add/delete/iterate) member interfaces.
+ * Bridge control request: add/delete member interfaces.
*/
struct ifbreq {
- char ifbname[IFNAMSIZ]; /* bridge ifs name */
- char ifsname[IFNAMSIZ]; /* member ifs name */
- u_int32_t index; /* iteration index */
+ char ifbr_name[IFNAMSIZ]; /* bridge ifs name */
+ char ifbr_ifsname[IFNAMSIZ]; /* member ifs name */
+ u_int32_t ifbr_ifsflags; /* memver ifs flags */
};
/*
- * Bridge routing request: iterate known routes.
+ * Interface list structure
*/
-struct ifbrtreq {
- char ifbname[IFNAMSIZ]; /* bridge ifs name */
- u_int32_t index; /* iteration index */
- struct ether_addr dst; /* destination addr */
- char ifsname[IFNAMSIZ]; /* destination ifs */
- u_int16_t age; /* route age */
+struct ifbifconf {
+ char ifbic_name[IFNAMSIZ]; /* bridge ifs name */
+ u_int32_t ifbic_len; /* buffer size */
+ union {
+ caddr_t ifbicu_buf;
+ struct ifbreq *ifbicu_req;
+ } ifbic_ifbicu;
+#define ifbic_buf ifbic_ifbicu.ifbicu_buf
+#define ifbic_req ifbic_ifbicu.ifbicu_req
+};
+
+/*
+ * Bridge address request
+ */
+struct ifbareq {
+ char ifba_name[IFNAMSIZ]; /* destination ifs */
+ u_int32_t ifba_age; /* route age */
+ struct ether_addr ifba_dst; /* destination addr */
+};
+
+struct ifbaconf {
+ char ifbac_name[IFNAMSIZ]; /* bridge ifs name */
+ u_int32_t ifbac_len; /* buffer size */
+ union {
+ caddr_t ifbacu_buf; /* buffer */
+ struct ifbareq *ifbacu_req; /* request pointer */
+ } ifbac_ifbacu;
+#define ifbac_buf ifbac_ifbacu.ifbacu_buf
+#define ifbac_req ifbac_ifbacu.ifbacu_req
+};
+
+/*
+ * Bridge cache size get/set
+ */
+struct ifbcachereq {
+ char ifbc_name[IFNAMSIZ]; /* bridge ifs name */
+ u_int32_t ifbc_size; /* cache size */
};
#ifdef _KERNEL
diff --git a/sys/sys/sockio.h b/sys/sys/sockio.h
index 7a3bb7c543a..4b593d3ce7b 100644
--- a/sys/sys/sockio.h
+++ b/sys/sys/sockio.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: sockio.h,v 1.5 1999/02/26 17:01:33 jason Exp $ */
+/* $OpenBSD: sockio.h,v 1.6 1999/03/01 04:44:43 jason Exp $ */
/* $NetBSD: sockio.h,v 1.5 1995/08/23 00:40:47 thorpej Exp $ */
/*-
@@ -85,8 +85,10 @@
#define SIOCBRDGADD _IOWR('i', 60, struct ifbreq) /* add bridge ifs */
#define SIOCBRDGDEL _IOWR('i', 61, struct ifbreq) /* del bridge ifs */
-#define SIOCBRDGIDX _IOWR('i', 62, struct ifbreq) /* get ifs by index */
-#define SIOCBRDGRT _IOWR('i', 63, struct ifbrtreq) /* get rts by index */
+#define SIOCBRDGSCACHE _IOWR('i', 64, struct ifbcachereq) /* set cache size */
+#define SIOCBRDGGCACHE _IOWR('i', 65, struct ifbcachereq) /* get cache size */
+#define SIOCBRDGIFS _IOWR('i', 66, struct ifbreq) /* get member ifs */
+#define SIOCBRDGRTS _IOWR('i', 67, struct ifbaconf) /* get addresses */
#define SIOCSIFMTU _IOW('i', 127, struct ifreq) /* set ifnet mtu */
#define SIOCGIFMTU _IOWR('i', 126, struct ifreq) /* get ifnet mtu */
diff --git a/usr.sbin/brconfig/brconfig.c b/usr.sbin/brconfig/brconfig.c
index 3071d2541c4..6c561fa21f4 100644
--- a/usr.sbin/brconfig/brconfig.c
+++ b/usr.sbin/brconfig/brconfig.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: brconfig.c,v 1.2 1999/02/27 18:29:54 jason Exp $ */
+/* $OpenBSD: brconfig.c,v 1.3 1999/03/01 04:44:44 jason Exp $ */
/*
* Copyright (c) 1999 Jason L. Wright (jason@thought.net)
@@ -46,6 +46,8 @@
#include <string.h>
#include <err.h>
#include <sysexits.h>
+#include <stdlib.h>
+#include <limits.h>
void usage(void);
int main(int, char **);
@@ -53,6 +55,7 @@ int bridge_setflag(int, char *, short);
int bridge_clrflag(int, char *, short);
int bridge_list(int, char *, char *);
int bridge_addrs(int, char *, char *);
+int bridge_maxaddr(int, char *, char *);
int bridge_add(int, char *, char *);
int bridge_delete(int, char *, char *);
int bridge_status(int, char *);
@@ -162,6 +165,16 @@ main(argc, argv)
if (error)
return (error);
}
+ else if (strcmp("maxaddr", argv[0]) == 0) {
+ argc--; argv++;
+ if (argc == 0) {
+ warnx("maxaddr requires an argument");
+ return (EX_USAGE);
+ }
+ error = bridge_maxaddr(sock, brdg, argv[0]);
+ if (error)
+ return (error);
+ }
else {
warnx("unrecognized option: %s", argv[0]);
return (EX_USAGE);
@@ -266,30 +279,34 @@ bridge_clrflag(s, brdg, f)
}
int
-bridge_list(sock, brdg, delim)
- int sock;
+bridge_list(s, brdg, delim)
+ int s;
char *brdg, *delim;
{
- struct ifbreq req;
- u_int32_t i = 0;
- char buf[sizeof(req.ifsname) + 1];
+ struct ifbreq *reqp;
+ struct ifbifconf bifc;
+ int i, len = 8192;
+ char buf[sizeof(reqp->ifbr_ifsname) + 1], *inbuf = NULL;
while (1) {
- strncpy(req.ifbname, brdg, sizeof(req.ifbname));
- req.index = i;
- if (ioctl(sock, SIOCBRDGIDX, &req) < 0) {
- if (errno == ENOENT) /* end of list */
- return (0);
- warn("ioctl(SIOCBRDGIDX)");
- return (EX_IOERR);
- }
-
+ strncpy(bifc.ifbic_name, brdg, sizeof(bifc.ifbic_name));
+ bifc.ifbic_len = len;
+ bifc.ifbic_buf = inbuf = realloc(inbuf, len);
+ if (inbuf == NULL)
+ err(1, "malloc");
+ if (ioctl(s, SIOCBRDGIFS, &bifc) < 0)
+ err(1, "ioctl(SIOCBRDGIFS)");
+ if (bifc.ifbic_len + sizeof(*reqp) < len)
+ break;
+ len *= 2;
+ }
+ for (i = 0; i < bifc.ifbic_len / sizeof(*reqp); i++) {
+ reqp = bifc.ifbic_req + i;
bzero(buf, sizeof(buf));
- strncpy(buf, req.ifsname, sizeof(req.ifsname));
+ strncpy(buf, reqp->ifbr_ifsname, sizeof(reqp->ifbr_ifsname));
printf("%s%s\n", delim, buf);
- i++;
}
-
+ free(bifc.ifbic_buf);
return (0); /* NOTREACHED */
}
@@ -300,8 +317,8 @@ bridge_add(s, brdg, ifn)
{
struct ifbreq req;
- strncpy(req.ifbname, brdg, sizeof(req.ifbname));
- strncpy(req.ifsname, ifn, sizeof(req.ifsname));
+ strncpy(req.ifbr_name, brdg, sizeof(req.ifbr_name));
+ strncpy(req.ifbr_ifsname, ifn, sizeof(req.ifbr_ifsname));
if (ioctl(s, SIOCBRDGADD, &req) < 0) {
warn("ioctl(SIOCADDBRDG)");
if (errno == EPERM)
@@ -318,8 +335,8 @@ bridge_delete(s, brdg, ifn)
{
struct ifbreq req;
- strncpy(req.ifbname, brdg, sizeof(req.ifbname));
- strncpy(req.ifsname, ifn, sizeof(req.ifsname));
+ strncpy(req.ifbr_name, brdg, sizeof(req.ifbr_name));
+ strncpy(req.ifbr_ifsname, ifn, sizeof(req.ifbr_ifsname));
if (ioctl(s, SIOCBRDGDEL, &req) < 0) {
warn("ioctl(SIOCDELBRDG)");
if (errno == EPERM)
@@ -330,28 +347,61 @@ bridge_delete(s, brdg, ifn)
}
int
+bridge_maxaddr(s, brdg, arg)
+ int s;
+ char *brdg, *arg;
+{
+ struct ifbcachereq ifbc;
+ u_int32_t newsize;
+ char *endptr;
+
+ newsize = strtoul(arg, &endptr, 0);
+ if (arg[0] == '\0' || endptr[0] != '\0') {
+ printf("invalid arg for maxaddr: %s\n", arg);
+ return (EX_USAGE);
+ }
+
+ strncpy(ifbc.ifbc_name, brdg, sizeof ifbc.ifbc_name);
+ ifbc.ifbc_size = newsize;
+ if (ioctl(s, SIOCBRDGSCACHE, (caddr_t)&ifbc) < 0) {
+ warn("ioctl(SIOCBRDGGCACHE)");
+ return (EX_IOERR);
+ }
+ return (0);
+}
+
+int
bridge_addrs(s, brdg, delim)
int s;
char *brdg, *delim;
{
- struct ifbrtreq req;
- u_int32_t i = 0;
- int r = 0;
+ struct ifbaconf ifbac;
+ struct ifbareq *ifba;
+ char *inbuf = NULL, buf[sizeof(ifba->ifba_name) + 1];
+ int i, len = 8192;
- while (r == 0) {
- strncpy(req.ifbname, brdg, sizeof(req.ifbname));
- req.index = i;
-
- r = ioctl(s, SIOCBRDGRT, &req);
- if (r != 0) {
- if (errno == ENOENT || errno == ENETDOWN)
+ while (1) {
+ ifbac.ifbac_len = len;
+ ifbac.ifbac_buf = inbuf = realloc(inbuf, len);
+ strncpy(ifbac.ifbac_name, brdg, sizeof(ifbac.ifbac_name));
+ if (inbuf == NULL)
+ err(EX_IOERR, "malloc");
+ if (ioctl(s, SIOCBRDGRTS, &ifbac) < 0) {
+ if (errno == ENETDOWN)
return (0);
- warn("ioctl(SIOCBRDGRT)");
- return (EX_IOERR);
+ err(EX_IOERR, "ioctl(SIOCBRDGRTS)");
}
- printf("%s%s %u %s\n", delim, ether_ntoa(&req.dst),
- req.age, req.ifsname);
- i++;
+ if (ifbac.ifbac_len + sizeof(*ifba) < len)
+ break;
+ len *= 2;
+ }
+
+ for (i = 0; i < ifbac.ifbac_len / sizeof(*ifba); i++) {
+ ifba = ifbac.ifbac_req + i;
+ bzero(buf, sizeof(buf));
+ strncpy(buf, ifba->ifba_name, sizeof(ifba->ifba_name));
+ printf("%s%s %s %u\n", delim, ether_ntoa(&ifba->ifba_dst),
+ buf, ifba->ifba_age);
}
return (0);
@@ -366,7 +416,7 @@ is_bridge(s, brdg)
char *brdg;
{
struct ifreq ifr;
- struct ifbrtreq req;
+ struct ifbaconf ifbac;
strncpy(ifr.ifr_name, brdg, sizeof ifr.ifr_name);
@@ -375,10 +425,10 @@ is_bridge(s, brdg)
return (0);
}
- strncpy(req.ifbname, brdg, sizeof(req.ifbname));
- req.index = 0;
- if (ioctl(s, SIOCBRDGRT, (caddr_t)&req) < 0) {
- if (errno == ENOENT || errno == ENETDOWN)
+ ifbac.ifbac_len = 0;
+ strncpy(ifbac.ifbac_name, brdg, sizeof(ifbac.ifbac_name));
+ if (ioctl(s, SIOCBRDGRTS, (caddr_t)&ifbac) < 0) {
+ if (errno == ENETDOWN)
return (1);
return (0);
}
@@ -391,6 +441,7 @@ bridge_status(s, brdg)
char *brdg;
{
struct ifreq ifr;
+ struct ifbcachereq ifbc;
int err;
strncpy(ifr.ifr_name, brdg, sizeof ifr.ifr_name);
@@ -411,7 +462,14 @@ bridge_status(s, brdg)
if (err)
return (err);
- printf("\tAddresses:\n");
+ strncpy(ifbc.ifbc_name, brdg, sizeof ifbc.ifbc_name);
+ if (ioctl(s, SIOCBRDGGCACHE, (caddr_t)&ifbc) < 0) {
+ warn("ioctl(SIOCBRDGGCACHE)");
+ return (EX_IOERR);
+ }
+
+ printf("\tAddresses (max cache: %u):\n", ifbc.ifbc_size);
+
err = bridge_addrs(s, brdg, "\t\t");
return (err);
}