summaryrefslogtreecommitdiffstats
path: root/sys/net
diff options
context:
space:
mode:
authorarchie <archie@FreeBSD.org>2000-06-26 23:34:54 +0000
committerarchie <archie@FreeBSD.org>2000-06-26 23:34:54 +0000
commit9af816e94656f88812336df4be5d2ddac233080f (patch)
tree3c138021e860caf3702977959a7cf77c2d4eb2e3 /sys/net
parent9b2c6ebcad4b7500a4fb6f89dcd04a814bc3e7ac (diff)
downloadFreeBSD-src-9af816e94656f88812336df4be5d2ddac233080f.zip
FreeBSD-src-9af816e94656f88812336df4be5d2ddac233080f.tar.gz
Make the ng_ether(4) node type dynamically loadable like the rest.
This means 'options NETGRAPH' is no longer necessary in order to get netgraph-enabled Ethernet interfaces. This supports loading/unloading the ng_ether.ko and attaching/detaching the Ethernet interface in any order. Add two new hooks 'upper' and 'lower' to allow access to the protocol demux engine and the raw device, respectively. This enables bridging to be defined as a netgraph node, if so desired. Reviewed by: freebsd-net@freebsd.org
Diffstat (limited to 'sys/net')
-rw-r--r--sys/net/ethernet.h16
-rw-r--r--sys/net/if.c12
-rw-r--r--sys/net/if_arp.h4
-rw-r--r--sys/net/if_ethersubr.c518
-rw-r--r--sys/net/if_var.h3
5 files changed, 126 insertions, 427 deletions
diff --git a/sys/net/ethernet.h b/sys/net/ethernet.h
index 7295e5a..5bb3448 100644
--- a/sys/net/ethernet.h
+++ b/sys/net/ethernet.h
@@ -80,7 +80,18 @@ struct ether_addr {
#define ETHERMTU (ETHER_MAX_LEN-ETHER_HDR_LEN-ETHER_CRC_LEN)
#define ETHERMIN (ETHER_MIN_LEN-ETHER_HDR_LEN-ETHER_CRC_LEN)
-#ifndef _KERNEL
+#ifdef _KERNEL
+
+extern void (*ng_ether_input_p)(struct ifnet *ifp,
+ struct mbuf **mp, struct ether_header *eh);
+extern void (*ng_ether_input_orphan_p)(struct ifnet *ifp,
+ struct mbuf *m, struct ether_header *eh);
+extern int (*ng_ether_output_p)(struct ifnet *ifp, struct mbuf **mp);
+extern void (*ng_ether_attach_p)(struct ifnet *ifp);
+extern void (*ng_ether_detach_p)(struct ifnet *ifp);
+
+#else /* _KERNEL */
+
#include <sys/cdefs.h>
/*
@@ -93,6 +104,7 @@ int ether_line __P((char *, struct ether_addr *, char *));
char *ether_ntoa __P((struct ether_addr *));
int ether_ntohost __P((char *, struct ether_addr *));
__END_DECLS
-#endif
+
+#endif /* !_KERNEL */
#endif /* !_NET_ETHERNET_H_ */
diff --git a/sys/net/if.c b/sys/net/if.c
index 2dce111..e33e86c 100644
--- a/sys/net/if.c
+++ b/sys/net/if.c
@@ -54,6 +54,7 @@
#include <net/if.h>
#include <net/if_arp.h>
#include <net/if_dl.h>
+#include <net/if_types.h>
#include <net/radix.h>
#include <net/route.h>
@@ -236,6 +237,17 @@ if_detach(ifp)
if_down(ifp);
/*
+ * Do any type-specific detach operation
+ */
+ switch (ifp->if_type) {
+ case IFT_ETHER:
+ ether_ifdetach(ifp);
+ break;
+ default:
+ break;
+ }
+
+ /*
* Remove address from ifnet_addrs[] and maybe decrement if_index.
* Clean up all addresses.
*/
diff --git a/sys/net/if_arp.h b/sys/net/if_arp.h
index 5bfbe4b..6760972 100644
--- a/sys/net/if_arp.h
+++ b/sys/net/if_arp.h
@@ -102,9 +102,7 @@ struct arpcom {
struct ifnet ac_if; /* network-visible interface */
u_char ac_enaddr[6]; /* ethernet hardware address */
int ac_multicnt; /* length of ac_multiaddrs list */
-/* #ifdef NETGRAPH */
- void *ac_ng; /* hook to hang netgraph stuff off */
-/* #endif */
+ void *ac_netgraph; /* ng_ether(4) netgraph node info */
};
extern u_char etherbroadcastaddr[6];
diff --git a/sys/net/if_ethersubr.c b/sys/net/if_ethersubr.c
index 22101e7..2084aac 100644
--- a/sys/net/if_ethersubr.c
+++ b/sys/net/if_ethersubr.c
@@ -57,6 +57,7 @@
#include <net/if_dl.h>
#include <net/if_types.h>
#include <net/bpf.h>
+#include <net/ethernet.h>
#if defined(INET) || defined(INET6)
#include <netinet/in.h>
@@ -105,48 +106,21 @@ extern u_char aarp_org_code[3];
#include <net/if_vlan_var.h>
#endif /* NVLAN > 0 */
+/* netgraph node hooks for ng_ether(4) */
+void (*ng_ether_input_p)(struct ifnet *ifp,
+ struct mbuf **mp, struct ether_header *eh);
+void (*ng_ether_input_orphan_p)(struct ifnet *ifp,
+ struct mbuf *m, struct ether_header *eh);
+int (*ng_ether_output_p)(struct ifnet *ifp, struct mbuf **mp);
+void (*ng_ether_attach_p)(struct ifnet *ifp);
+void (*ng_ether_detach_p)(struct ifnet *ifp);
+
static int ether_resolvemulti __P((struct ifnet *, struct sockaddr **,
struct sockaddr *));
u_char etherbroadcastaddr[6] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
#define senderr(e) do { error = (e); goto bad;} while (0)
#define IFP2AC(IFP) ((struct arpcom *)IFP)
-#ifdef NETGRAPH
-#include <netgraph/ng_ether.h>
-#include <netgraph/ng_message.h>
-#include <netgraph/netgraph.h>
-
-static void ngether_init(void* ignored);
-static void ngether_send(struct arpcom *ac,
- struct ether_header *eh, struct mbuf *m);
-static ng_constructor_t ngether_constructor;
-static ng_rcvmsg_t ngether_rcvmsg;
-static ng_shutdown_t ngether_rmnode;
-static ng_newhook_t ngether_newhook;
-static ng_connect_t ngether_connect;
-static ng_rcvdata_t ngether_rcvdata;
-static ng_disconnect_t ngether_disconnect;
-
-static struct ng_type typestruct = {
- NG_VERSION,
- NG_ETHER_NODE_TYPE,
- NULL,
- ngether_constructor,
- ngether_rcvmsg,
- ngether_rmnode,
- ngether_newhook,
- NULL,
- ngether_connect,
- ngether_rcvdata,
- ngether_rcvdata,
- ngether_disconnect,
- NULL
-};
-
-#define AC2NG(AC) ((node_p)((AC)->ac_ng))
-#define NGEF_DIVERT NGF_TYPE1 /* all packets sent to netgraph */
-#endif /* NETGRAPH */
-
/*
* Ethernet output routine.
* Encapsulate a packet of type family for the local net.
@@ -162,11 +136,11 @@ ether_output(ifp, m, dst, rt0)
struct rtentry *rt0;
{
short type;
- int s, error = 0, hdrcmplt = 0;
+ int error = 0, hdrcmplt = 0;
u_char esrc[6], edst[6];
register struct rtentry *rt;
register struct ether_header *eh;
- int off, len = m->m_pkthdr.len, loop_copy = 0;
+ int off, loop_copy = 0;
int hlen; /* link layer header lenght */
struct arpcom *ac = IFP2AC(ifp);
@@ -249,7 +223,6 @@ ether_output(ifp, m, dst, rt0)
struct llc llc;
M_PREPEND(m, sizeof(struct llc), M_WAIT);
- len += sizeof(struct llc);
llc.llc_dsap = llc.llc_ssap = LLC_SNAP_LSAP;
llc.llc_control = LLC_UI;
bcopy(at_org_code, llc.llc_snap_org_code, sizeof(at_org_code));
@@ -367,6 +340,34 @@ ether_output(ifp, m, dst, rt0)
}
}
+ /* Handle ng_ether(4) processing, if any */
+ if (ng_ether_output_p != NULL) {
+ if ((error = (*ng_ether_output_p)(ifp, &m)) != 0) {
+bad: if (m != NULL)
+ m_freem(m);
+ return (error);
+ }
+ if (m == NULL)
+ return (0);
+ }
+
+ /* Continue with link-layer output */
+ return ether_output_frame(ifp, m);
+}
+
+/*
+ * Ethernet link layer output routine to send a raw frame to the device.
+ *
+ * This assumes that the 14 byte Ethernet header is present and contiguous
+ * in the first mbuf (if BRIDGE'ing).
+ */
+int
+ether_output_frame(ifp, m)
+ struct ifnet *ifp;
+ struct mbuf *m;
+{
+ int s, error = 0;
+
#ifdef BRIDGE
if (do_bridge) {
struct ether_header hdr;
@@ -390,20 +391,16 @@ ether_output(ifp, m, dst, rt0)
if (IF_QFULL(&ifp->if_snd)) {
IF_DROP(&ifp->if_snd);
splx(s);
- senderr(ENOBUFS);
+ m_freem(m);
+ return (ENOBUFS);
}
+ ifp->if_obytes += m->m_pkthdr.len;
if (m->m_flags & M_MCAST)
ifp->if_omcasts++;
IF_ENQUEUE(&ifp->if_snd, m);
if ((ifp->if_flags & IFF_OACTIVE) == 0)
(*ifp->if_start)(ifp);
splx(s);
- ifp->if_obytes += len + sizeof (struct ether_header);
- return (error);
-
-bad:
- if (m)
- m_freem(m);
return (error);
}
@@ -411,19 +408,16 @@ bad:
* Process a received Ethernet packet;
* the packet is in the mbuf chain m without
* the ether header, which is provided separately.
+ *
+ * First we perform any link layer operations, then continue
+ * to the upper layers with ether_demux().
*/
void
ether_input(ifp, eh, m)
struct ifnet *ifp;
- register struct ether_header *eh;
+ struct ether_header *eh;
struct mbuf *m;
{
- register struct ifqueue *inq;
- u_short ether_type;
- int s;
-#if defined(NETATALK)
- register struct llc *l;
-#endif
/* Check for a BPF tap */
if (ifp->if_bpf != NULL) {
@@ -436,6 +430,13 @@ ether_input(ifp, eh, m)
bpf_mtap(ifp, (struct mbuf *)&mh);
}
+ /* Handle ng_ether(4) processing, if any */
+ if (ng_ether_input_p != NULL) {
+ (*ng_ether_input_p)(ifp, &m, eh);
+ if (m == NULL)
+ return;
+ }
+
#ifdef BRIDGE
/* Check for bridging mode */
if (do_bridge) {
@@ -473,6 +474,26 @@ ether_input(ifp, eh, m)
#ifdef BRIDGE
recvLocal:
#endif
+ /* Continue with upper layer processing */
+ ether_demux(ifp, eh, m);
+}
+
+/*
+ * Upper layer processing for a received Ethernet packet.
+ */
+void
+ether_demux(ifp, eh, m)
+ struct ifnet *ifp;
+ struct ether_header *eh;
+ struct mbuf *m;
+{
+ struct ifqueue *inq;
+ u_short ether_type;
+ int s;
+#if defined(NETATALK)
+ register struct llc *l;
+#endif
+
/* Discard packet if interface is not up */
if ((ifp->if_flags & IFF_UP) == 0) {
m_freem(m);
@@ -491,16 +512,6 @@ recvLocal:
ether_type = ntohs(eh->ether_type);
-#ifdef NETGRAPH
- {
- struct arpcom *ac = IFP2AC(ifp);
- if (AC2NG(ac) && (AC2NG(ac)->flags & NGEF_DIVERT)) {
- ngether_send(ac, eh, m);
- return;
- }
- }
-#endif /* NETGRAPH */
-
#if NVLAN > 0
if (ether_type == vlan_proto) {
if (vlan_input(eh, m) < 0)
@@ -608,20 +619,18 @@ recvLocal:
break;
dropanyway:
default:
-#ifdef NETGRAPH
- ngether_send(IFP2AC(ifp), eh, m);
-#else /* NETGRAPH */
- m_freem(m);
-#endif /* NETGRAPH */
+ if (ng_ether_input_orphan_p != NULL)
+ (*ng_ether_input_orphan_p)(ifp, m, eh);
+ else
+ m_freem(m);
return;
}
#else /* NETATALK */
-#ifdef NETGRAPH
- ngether_send(IFP2AC(ifp), eh, m);
-#else /* NETGRAPH */
- m_freem(m);
-#endif /* NETGRAPH */
- return;
+ if (ng_ether_input_orphan_p != NULL)
+ (*ng_ether_input_orphan_p)(ifp, m, eh);
+ else
+ m_freem(m);
+ return;
#endif /* NETATALK */
}
@@ -660,12 +669,22 @@ ether_ifattach(ifp)
sdl->sdl_type = IFT_ETHER;
sdl->sdl_alen = ifp->if_addrlen;
bcopy((IFP2AC(ifp))->ac_enaddr, LLADDR(sdl), ifp->if_addrlen);
-#ifdef NETGRAPH
- ngether_init(ifp);
-#endif /* NETGRAPH */
#ifdef INET6
in6_ifattach_getifid(ifp);
#endif
+ if (ng_ether_attach_p != NULL)
+ (*ng_ether_attach_p)(ifp);
+}
+
+/*
+ * Perform common duties while detaching an Ethernet interface
+ */
+void
+ether_ifdetach(ifp)
+ struct ifnet *ifp;
+{
+ if (ng_ether_detach_p != NULL)
+ (*ng_ether_detach_p)(ifp);
}
SYSCTL_DECL(_net_link);
@@ -846,348 +865,3 @@ ether_resolvemulti(ifp, llsa, sa)
}
}
-#ifdef NETGRAPH
-
-/***********************************************************************
- * This section contains the methods for the Netgraph interface
- ***********************************************************************/
-/* It's Ascii-art time!
- * The ifnet is the first part of the arpcom which must be
- * the first part of the device's softc.. yuk.
- *
- * +--------------------------+-----+---------+
- * | struct ifnet (*ifp) | | |
- * | | | |
- * +--------------------------+ | |
- * +--|[ac_ng] struct arpcom (*ac) | |
- * | +--------------------------------+ |
- * | | struct softc (*ifp->if_softc) (device) |
- * | +------------------------------------------+
- * | ^
- * AC2NG() |
- * | v
- * | +----------------------+
- * | | [private] [flags] |
- * +------>| struct ng_node |
- * | [hooks] | ** we only allow one hook
- * +----------------------+
- * ^
- * |
- * v
- * +-------------+
- * | [node] |
- * | hook |
- * | [private]|-- *unused*
- * +-------------+
- */
-
-/*
- * called during interface attaching
- */
-static void
-ngether_init(void *ifpvoid)
-{
- struct ifnet *ifp = ifpvoid;
- struct arpcom *ac = IFP2AC(ifp);
- static int ngether_done_init;
- char namebuf[32];
- node_p node;
-
- /*
- * we have found a node, make sure our 'type' is availabe.
- */
- if (ngether_done_init == 0) {
- if (ng_newtype(&typestruct)) {
- printf("ngether install failed\n");
- return;
- }
- ngether_done_init = 1;
- }
- if (ng_make_node_common(&typestruct, &node) != 0)
- return;
- ac->ac_ng = node;
- node->private = ifp;
- sprintf(namebuf, "%s%d", ifp->if_name, ifp->if_unit);
- ng_name_node(AC2NG(ac), namebuf);
-}
-
-/*
- * It is not possible or allowable to create a node of this type.
- * If the hardware exists, it will already have created it.
- */
-static int
-ngether_constructor(node_p *nodep)
-{
- return (EINVAL);
-}
-
-/*
- * Give our ok for a hook to be added...
- *
- * Allow one hook at a time (rawdata).
- * It can eiteh rdivert everything or only unclaimed packets.
- */
-static int
-ngether_newhook(node_p node, hook_p hook, const char *name)
-{
-
- /* check if there is already a hook */
- if (LIST_FIRST(&(node->hooks)))
- return(EISCONN);
- /*
- * Check for which mode hook we want.
- */
- if (strcmp(name, NG_ETHER_HOOK_ORPHAN) != 0) {
- if (strcmp(name, NG_ETHER_HOOK_DIVERT) != 0) {
- return (EINVAL);
- }
- node->flags |= NGEF_DIVERT;
- } else {
- node->flags &= ~NGEF_DIVERT;
- }
- return (0);
-}
-
-/*
- * incoming messages.
- * Just respond to the generic TEXT_STATUS message
- */
-static int
-ngether_rcvmsg(node_p node, struct ng_mesg *msg, const char *retaddr,
- struct ng_mesg **resp, hook_p lasthook)
-{
- struct ifnet *ifp;
- int error = 0;
-
- ifp = node->private;
- switch (msg->header.typecookie) {
- case NGM_ETHER_COOKIE:
- error = EINVAL;
- break;
- case NGM_GENERIC_COOKIE:
- switch(msg->header.cmd) {
- case NGM_TEXT_STATUS: {
- char *arg;
- int pos = 0;
- int resplen = sizeof(struct ng_mesg) + 512;
- MALLOC(*resp, struct ng_mesg *, resplen,
- M_NETGRAPH, M_NOWAIT);
- if (*resp == NULL) {
- error = ENOMEM;
- break;
- }
- bzero(*resp, resplen);
- arg = (*resp)->data;
-
- /*
- * Put in the throughput information.
- */
- pos = sprintf(arg, "%ld bytes in, %ld bytes out\n",
- ifp->if_ibytes, ifp->if_obytes);
- pos += sprintf(arg + pos,
- "%ld output errors\n",
- ifp->if_oerrors);
- pos += sprintf(arg + pos,
- "ierrors = %ld\n",
- ifp->if_ierrors);
-
- (*resp)->header.version = NG_VERSION;
- (*resp)->header.arglen = strlen(arg) + 1;
- (*resp)->header.token = msg->header.token;
- (*resp)->header.typecookie = NGM_ETHER_COOKIE;
- (*resp)->header.cmd = msg->header.cmd;
- strncpy((*resp)->header.cmdstr, "status",
- NG_CMDSTRLEN);
- }
- break;
- default:
- error = EINVAL;
- break;
- }
- break;
- default:
- error = EINVAL;
- break;
- }
- free(msg, M_NETGRAPH);
- return (error);
-}
-
-/*
- * Receive a completed ethernet packet.
- * Queue it for output.
- */
-static int
-ngether_rcvdata(hook_p hook, struct mbuf *m, meta_p meta,
- struct mbuf **ret_m, meta_p *ret_meta)
-{
- struct ifnet *ifp;
- int error = 0;
- int s;
- struct ether_header *eh;
-
- ifp = hook->node->private;
-
- if ((ifp->if_flags & (IFF_UP|IFF_RUNNING)) != (IFF_UP|IFF_RUNNING))
- senderr(ENETDOWN);
- /* drop in the MAC address */
- eh = mtod(m, struct ether_header *);
- bcopy(IFP2AC(ifp)->ac_enaddr, eh->ether_shost, 6);
- /*
- * If a simplex interface, and the packet is being sent to our
- * Ethernet address or a broadcast address, loopback a copy.
- * XXX To make a simplex device behave exactly like a duplex
- * device, we should copy in the case of sending to our own
- * ethernet address (thus letting the original actually appear
- * on the wire). However, we don't do that here for security
- * reasons and compatibility with the original behavior.
- */
- if (ifp->if_flags & IFF_SIMPLEX) {
- if (m->m_flags & M_BCAST) {
- struct mbuf *n = m_copy(m, 0, (int)M_COPYALL);
-
- ng_queue_data(hook, n, meta);
- } else if (bcmp(eh->ether_dhost,
- eh->ether_shost, ETHER_ADDR_LEN) == 0) {
- ng_queue_data(hook, m, meta);
- return (0); /* XXX */
- }
- }
- s = splimp();
- /*
- * Queue message on interface, and start output if interface
- * not yet active.
- * XXX if we lookead at the priority in the meta data we could
- * queue high priority items at the head.
- */
- if (IF_QFULL(&ifp->if_snd)) {
- IF_DROP(&ifp->if_snd);
- splx(s);
- senderr(ENOBUFS);
- }
- ifp->if_obytes += m->m_pkthdr.len;
- if (m->m_flags & M_MCAST)
- ifp->if_omcasts++;
- IF_ENQUEUE(&ifp->if_snd, m);
- if ((ifp->if_flags & IFF_OACTIVE) == 0)
- (*ifp->if_start)(ifp);
- splx(s);
- return (error);
-
-bad:
- NG_FREE_DATA(m, meta);
- return (error);
-}
-
-/*
- * pass an mbuf out to the connected hook
- * More complicated than just an m_prepend, as it tries to save later nodes
- * from needing to do lots of m_pullups.
- */
-static void
-ngether_send(struct arpcom *ac, struct ether_header *eh, struct mbuf *m)
-{
- int room;
- node_p node = AC2NG(ac);
- struct ether_header *eh2;
-
- if (node && LIST_FIRST(&(node->hooks))) {
- /*
- * Possibly the header is already on the front,
- */
- eh2 = mtod(m, struct ether_header *) - 1;
- if ( eh == eh2) {
- /*
- * This is the case so just move the markers back to
- * re-include it. We lucked out.
- * This allows us to avoid a yucky m_pullup
- * in later nodes if it works.
- */
- m->m_len += sizeof(*eh);
- m->m_data -= sizeof(*eh);
- m->m_pkthdr.len += sizeof(*eh);
- } else {
- /*
- * Alternatively there may be room even though
- * it is stored somewhere else. If so, copy it in.
- * This only safe because we KNOW that this packet has
- * just been generated by an ethernet card, so there
- * are no aliases to the buffer. (unlike in outgoing
- * packets).
- * Nearly all ethernet cards will end up producing mbufs
- * that fall into these cases. So we are not optimising
- * contorted cases.
- */
-
- if (m->m_flags & M_EXT) {
- room = (mtod(m, caddr_t) - m->m_ext.ext_buf);
- if (room > m->m_ext.ext_size) /* garbage */
- room = 0; /* fail immediatly */
- } else {
- room = (mtod(m, caddr_t) - m->m_pktdat);
- }
- if (room > sizeof (*eh)) {
- /* we have room, just copy it and adjust */
- m->m_len += sizeof(*eh);
- m->m_data -= sizeof(*eh);
- m->m_pkthdr.len += sizeof(*eh);
- } else {
- /*
- * Doing anything more is likely to get more
- * expensive than it's worth..
- * it's probable that everything else is in one
- * big lump. The next node will do an m_pullup()
- * for exactly the amount of data it needs and
- * hopefully everything after that will not
- * need one. So let's just use M_PREPEND.
- */
- M_PREPEND(m, sizeof (*eh), M_DONTWAIT);
- if (m == NULL)
- return;
- }
- bcopy ((caddr_t)eh, mtod(m, struct ether_header *),
- sizeof(*eh));
- }
- ng_queue_data(LIST_FIRST(&(node->hooks)), m, NULL);
- } else {
- m_freem(m);
- }
-}
-
-/*
- * do local shutdown processing..
- * This node will refuse to go away, unless the hardware says to..
- * don't unref the node, or remove our name. just clear our links up.
- */
-static int
-ngether_rmnode(node_p node)
-{
- ng_cutlinks(node);
- node->flags &= ~NG_INVALID; /* bounce back to life */
- return (0);
-}
-
-/* already linked */
-static int
-ngether_connect(hook_p hook)
-{
- /* be really amiable and just say "YUP that's OK by me! " */
- return (0);
-}
-
-/*
- * notify on hook disconnection (destruction)
- *
- * For this type, removal of the last lins no effect. The interface can run
- * independently.
- * Since we have no per-hook information, this is rather simple.
- */
-static int
-ngether_disconnect(hook_p hook)
-{
- hook->node->flags &= ~NGEF_DIVERT;
- return (0);
-}
-#endif /* NETGRAPH */
-
-/********************************** END *************************************/
diff --git a/sys/net/if_var.h b/sys/net/if_var.h
index 2ec532b..5234a58 100644
--- a/sys/net/if_var.h
+++ b/sys/net/if_var.h
@@ -324,9 +324,12 @@ extern int if_index;
extern struct ifaddr **ifnet_addrs;
void ether_ifattach __P((struct ifnet *));
+void ether_ifdetach __P((struct ifnet *));
void ether_input __P((struct ifnet *, struct ether_header *, struct mbuf *));
+void ether_demux __P((struct ifnet *, struct ether_header *, struct mbuf *));
int ether_output __P((struct ifnet *,
struct mbuf *, struct sockaddr *, struct rtentry *));
+int ether_output_frame __P((struct ifnet *, struct mbuf *));
int ether_ioctl __P((struct ifnet *, int, caddr_t));
int if_addmulti __P((struct ifnet *, struct sockaddr *,
OpenPOWER on IntegriCloud