diff options
author | Adrian Chadd <adrian@FreeBSD.org> | 2014-03-01 10:04:31 +0000 |
---|---|---|
committer | Adrian Chadd <adrian@FreeBSD.org> | 2014-03-01 10:04:31 +0000 |
commit | 03b5d8277b413df9579e7a5f275c1ad7a10626dd (patch) | |
tree | b3595fee7ddfe9e80cdc8eb866fcdfa60c23581a /sys/dev/etherswitch/arswitch | |
parent | 0f4d09728465f496a4d8d6de1875dc6c2dad11b3 (diff) |
(I think!) make the AR8327 switch correctly handle traffic.
This patch does four things:
* it globally disables mirroring;
* it globally sets the mirroring on each port to be disabled;
* the initial port setup now programs a portmask for the port to allow
transmission (forwarding) to all other ports bar itself;
* the vlan setup path now programs the portmask for the port to
allow transmission (forwarding) to all other ports bar itself.
Before this, I hard-coded the portmask to 0x3f which would mean all
ports (bar port 6, which currently isn't hooked up to anything.)
This means that traffic would be duplicated back out the port it
received it. I bet this wasn't .. optimal.
In any case, this _seems_ to make DHCP from my macosx laptop
work through this access point. I'll do some further testing
to ensure it's actually working correctly on all my devices.
Tested:
* DB120, AR8327 switch
Notes
Notes:
svn path=/head/; revision=262653
Diffstat (limited to 'sys/dev/etherswitch/arswitch')
-rw-r--r-- | sys/dev/etherswitch/arswitch/arswitch_8327.c | 32 |
1 files changed, 31 insertions, 1 deletions
diff --git a/sys/dev/etherswitch/arswitch/arswitch_8327.c b/sys/dev/etherswitch/arswitch/arswitch_8327.c index 5fc1d0f4c4dd..44dbb1e44f45 100644 --- a/sys/dev/etherswitch/arswitch/arswitch_8327.c +++ b/sys/dev/etherswitch/arswitch/arswitch_8327.c @@ -665,8 +665,19 @@ ar8327_port_init(struct arswitch_softc *sc, int port) t = AR8327_PORT_VLAN1_OUT_MODE_UNTOUCH << AR8327_PORT_VLAN1_OUT_MODE_S; arswitch_writereg(sc->sc_dev, AR8327_REG_PORT_VLAN1(port), t); + /* + * This doesn't configure any ports which this port can "see". + * bits 0-6 control which ports a frame coming into this port + * can be sent out to. + * + * So by doing this, we're making it impossible to send frames out + * to that port. + */ t = AR8327_PORT_LOOKUP_LEARN; t |= AR8X16_PORT_CTRL_STATE_FORWARD << AR8327_PORT_LOOKUP_STATE_S; + + /* So this allows traffic to any port except ourselves */ + t |= (0x3f & ~(1 << port)); arswitch_writereg(sc->sc_dev, AR8327_REG_PORT_LOOKUP(port), t); } @@ -695,6 +706,13 @@ ar8327_reset_vlans(struct arswitch_softc *sc) uint32_t mode, t; /* + * Disable mirroring. + */ + arswitch_modifyreg(sc->sc_dev, AR8327_REG_FWD_CTRL0, + AR8327_FWD_CTRL0_MIRROR_PORT, + (0xF << AR8327_FWD_CTRL0_MIRROR_PORT_S)); + + /* * For now, let's default to one portgroup, just so traffic * flows. All ports can see other ports. */ @@ -713,13 +731,25 @@ ar8327_reset_vlans(struct arswitch_softc *sc) /* Set ingress = out_keep; members = 0x3f for all ports */ - t = 0x3f; /* all ports */ + t = (0x3f & ~(1 << i)); /* all ports besides us */ t |= AR8327_PORT_LOOKUP_LEARN; /* in_port_only, forward */ t |= AR8X16_PORT_VLAN_MODE_PORT_ONLY << AR8327_PORT_LOOKUP_IN_MODE_S; t |= AR8X16_PORT_CTRL_STATE_FORWARD << AR8327_PORT_LOOKUP_STATE_S; arswitch_writereg(sc->sc_dev, AR8327_REG_PORT_LOOKUP(i), t); + + /* + * Disable port mirroring entirely. + */ + arswitch_modifyreg(sc->sc_dev, + AR8327_REG_PORT_LOOKUP(i), + AR8327_PORT_LOOKUP_ING_MIRROR_EN, + 0); + arswitch_modifyreg(sc->sc_dev, + AR8327_REG_PORT_HOL_CTRL1(i), + AR8327_PORT_HOL_CTRL1_EG_MIRROR_EN, + 0); } } |