summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJoel Sherrill <joel.sherrill@OARcorp.com>2011-04-11 17:28:04 +0000
committerJoel Sherrill <joel.sherrill@OARcorp.com>2011-04-11 17:28:04 +0000
commit43ce60a47a3c9461aad91bd6bbef80e53b20a14b (patch)
treed32ecc84c1ba7e7d95df0bb332420ba0ef68679c
parent425927e16d079c5783054a93fb08d66adeae5143 (diff)
2011-04-11 Keith Robertson <kjrobert at alumni dot uwaterloo dot ca>
* ne2000/ne2000.c: Add multicast support. Patch submitted to mailing list 2005-12-21.
-rw-r--r--c/src/lib/libbsp/i386/pc386/ChangeLog5
-rw-r--r--c/src/lib/libbsp/i386/pc386/ne2000/ne2000.c43
2 files changed, 45 insertions, 3 deletions
diff --git a/c/src/lib/libbsp/i386/pc386/ChangeLog b/c/src/lib/libbsp/i386/pc386/ChangeLog
index 4a49cb5bf7..70614bb9f8 100644
--- a/c/src/lib/libbsp/i386/pc386/ChangeLog
+++ b/c/src/lib/libbsp/i386/pc386/ChangeLog
@@ -1,3 +1,8 @@
+2011-04-11 Keith Robertson <kjrobert at alumni dot uwaterloo dot ca>
+
+ * ne2000/ne2000.c: Add multicast support. Patch submitted to mailing
+ list 2005-12-21.
+
2011-03-07 Joel Sherrill <joel.sherrilL@OARcorp.com>
PR 1756/bsps
diff --git a/c/src/lib/libbsp/i386/pc386/ne2000/ne2000.c b/c/src/lib/libbsp/i386/pc386/ne2000/ne2000.c
index 2c5484cd01..ba647fa464 100644
--- a/c/src/lib/libbsp/i386/pc386/ne2000/ne2000.c
+++ b/c/src/lib/libbsp/i386/pc386/ne2000/ne2000.c
@@ -504,6 +504,9 @@ ne_init_hardware (struct ne_softc *sc)
/* accept broadcast */
outport_byte (port + RCR, (sc->accept_broadcasts ? MSK_AB : 0));
+ /* accept multicast */
+ outport_byte (port + RCR, MSK_AM);
+
/* Start interface */
outport_byte (port + CMDR, MSK_PG0 | MSK_RD2 | MSK_STA);
@@ -1072,6 +1075,28 @@ ne_stats (struct ne_softc *sc)
printf (" Interrupts: %-8lu\n", sc->stats.interrupts);
}
+static int ne_set_multicast_filter(struct ne_softc* sc)
+{
+ int i=0;
+ unsigned int port = sc->port;
+ unsigned char cmd = 0;
+
+ /* Save CMDR settings */
+ inport_byte(port + CMDR, cmd);
+ /* Change to page 1 */
+ outport_byte(port + CMDR, cmd | MSK_PG1);
+
+ /* Set MAR to accept _all_ multicast packets */
+ for (i = 0; i < MARsize; ++i) {
+ outport_byte (port + MAR + i, 0xFF);
+ }
+
+ /* Revert to original CMDR settings */
+ outport_byte(port + CMDR, cmd);
+
+ return 0;
+}
+
/* NE2000 driver ioctl handler. */
static int
@@ -1105,13 +1130,25 @@ ne_ioctl (struct ifnet *ifp, ioctl_command_t command, caddr_t data)
break;
}
break;
+
+ case SIOCADDMULTI:
+ case SIOCDELMULTI:
+ {
+ struct ifreq* ifr = (struct ifreq*) data;
+ error = (command == SIOCADDMULTI ?
+ ether_addmulti(ifr, &(sc->arpcom)) :
+ ether_delmulti(ifr, &(sc->arpcom)) );
+ /* ENETRESET indicates that driver should update its multicast filters */
+ if(error == ENETRESET) {
+ error = ne_set_multicast_filter(sc);
+ }
+ break;
+ }
case SIO_RTEMS_SHOW_STATS:
ne_stats (sc);
break;
- /* FIXME: Multicast commands must be added here. */
-
default:
error = EINVAL;
break;
@@ -1247,7 +1284,7 @@ rtems_ne_driver_attach (struct rtems_bsdnet_ifconfig *config, int attach)
ifp->if_watchdog = ne_watchdog;
ifp->if_start = ne_start;
ifp->if_output = ether_output;
- ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX;
+ ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST;
if (ifp->if_snd.ifq_maxlen == 0)
ifp->if_snd.ifq_maxlen = ifqmaxlen;