summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
authorJonathan Matthew <jmatthew@cvs.openbsd.org>2022-03-13 10:13:55 +0000
committerJonathan Matthew <jmatthew@cvs.openbsd.org>2022-03-13 10:13:55 +0000
commitdb680cb4015bcdae7dfa2ffb8530121a276ee4ac (patch)
tree289567c7870bae7b615c9ef9d09a386a51fddca8 /sys
parentecd34776fe2c5a8a2e74ded8ea4c93337cf01d30 (diff)
Enable vlan promisc and header stripping, and use vlan rx/tx offloads.
The vlan tx information takes up an extra slot on the tx ring, so this affects the point at which we stop putting packets on the ring too. ok dlg@
Diffstat (limited to 'sys')
-rw-r--r--sys/dev/pci/if_aq_pci.c46
1 files changed, 41 insertions, 5 deletions
diff --git a/sys/dev/pci/if_aq_pci.c b/sys/dev/pci/if_aq_pci.c
index 25c8f4ce72a..d7791b06ddc 100644
--- a/sys/dev/pci/if_aq_pci.c
+++ b/sys/dev/pci/if_aq_pci.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_aq_pci.c,v 1.8 2022/03/12 23:54:53 jmatthew Exp $ */
+/* $OpenBSD: if_aq_pci.c,v 1.9 2022/03/13 10:13:54 jmatthew Exp $ */
/* $NetBSD: if_aq.c,v 1.27 2021/06/16 00:21:18 riastradh Exp $ */
/*
@@ -79,6 +79,7 @@
* POSSIBILITY OF SUCH DAMAGE.
*/
#include "bpfilter.h"
+#include "vlan.h"
#include <sys/types.h>
#include <sys/device.h>
@@ -654,6 +655,7 @@ struct aq_tx_desc {
#define AQ_TXDESC_CTL1_TYPE_TXD 0x00000001
#define AQ_TXDESC_CTL1_TYPE_TXC 0x00000002
#define AQ_TXDESC_CTL1_BLEN_SHIFT 4
+#define AQ_TXDESC_CTL1_VLAN_SHIFT 4
#define AQ_TXDESC_CTL1_DD (1 << 20)
#define AQ_TXDESC_CTL1_CMD_EOP (1 << 21)
#define AQ_TXDESC_CTL1_CMD_VLAN (1 << 22)
@@ -1035,6 +1037,9 @@ aq_attach(struct device *parent, struct device *self, void *aux)
ifp->if_watchdog = aq_watchdog;
ifp->if_hardmtu = 9000;
ifp->if_capabilities = IFCAP_VLAN_MTU;
+#if NVLAN > 0
+ ifp->if_capabilities |= IFCAP_VLAN_HWTAGGING;
+#endif
ifq_set_maxlen(&ifp->if_snd, AQ_TXD_NUM);
ifmedia_init(&sc->sc_media, IFM_IMASK, aq_ifmedia_change,
@@ -1799,7 +1804,7 @@ aq_hw_init_rx_path(struct aq_softc *sc)
ETHERTYPE_QINQ);
AQ_WRITE_REG_BIT(sc, RPF_VLAN_TPID_REG, RPF_VLAN_TPID_INNER,
ETHERTYPE_VLAN);
- AQ_WRITE_REG_BIT(sc, RPF_VLAN_MODE_REG, RPF_VLAN_MODE_PROMISC, 0);
+ AQ_WRITE_REG_BIT(sc, RPF_VLAN_MODE_REG, RPF_VLAN_MODE_PROMISC, 1);
if (sc->sc_features & FEATURES_REV_B) {
AQ_WRITE_REG_BIT(sc, RPF_VLAN_MODE_REG,
@@ -2022,6 +2027,7 @@ void
aq_rxring_reset(struct aq_softc *sc, struct aq_rxring *rx, int start)
{
daddr_t paddr;
+ int strip;
AQ_WRITE_REG_BIT(sc, RX_DMA_DESC_REG(rx->rx_q), RX_DMA_DESC_EN, 0);
/* drain */
@@ -2043,8 +2049,14 @@ aq_rxring_reset(struct aq_softc *sc, struct aq_rxring *rx, int start)
RX_DMA_DESC_BUFSIZE_HDR, 0);
AQ_WRITE_REG_BIT(sc, RX_DMA_DESC_REG(rx->rx_q),
RX_DMA_DESC_HEADER_SPLIT, 0);
+
+#if NVLAN > 0
+ strip = 1;
+#else
+ strip = 0;
+#endif
AQ_WRITE_REG_BIT(sc, RX_DMA_DESC_REG(rx->rx_q),
- RX_DMA_DESC_VLAN_STRIP, 0);
+ RX_DMA_DESC_VLAN_STRIP, strip);
rx->rx_cons = AQ_READ_REG(sc, RX_DMA_DESC_HEAD_PTR_REG(rx->rx_q)) &
RX_DMA_DESC_HEAD_PTR;
@@ -2186,7 +2198,14 @@ aq_rxeof(struct aq_softc *sc, struct aq_rxring *rx)
pktlen = lemtoh16(&rxd->pkt_len);
rxd_type = lemtoh32(&rxd->type);
- /* rss hash, vlan */
+ /* rss hash */
+
+#if NVLAN > 0
+ if (rxd_type & (AQ_RXDESC_TYPE_VLAN | AQ_RXDESC_TYPE_VLAN2)) {
+ m->m_pkthdr.ether_vtag = lemtoh16(&rxd->vlan);
+ m->m_flags |= M_VLANTAG;
+ }
+#endif
if ((status & AQ_RXDESC_STATUS_MACERR) ||
(rxd_type & AQ_RXDESC_TYPE_DMA_ERR)) {
@@ -2278,7 +2297,7 @@ aq_start(struct ifqueue *ifq)
ring = (struct aq_tx_desc *)AQ_DMA_KVA(&tx->tx_mem);
for (;;) {
- if (used + AQ_TX_MAX_SEGMENTS >= free) {
+ if (used + AQ_TX_MAX_SEGMENTS + 1 >= free) {
ifq_set_oactive(ifq);
break;
}
@@ -2316,6 +2335,23 @@ aq_start(struct ifqueue *ifq)
ctl2 = m->m_pkthdr.len << AQ_TXDESC_CTL2_LEN_SHIFT;
ctl1 = AQ_TXDESC_CTL1_TYPE_TXD | AQ_TXDESC_CTL1_CMD_FCS;
+#if NVLAN > 0
+ if (m->m_flags & M_VLANTAG) {
+ txd = ring + idx;
+ txd->buf_addr = 0;
+ txd->ctl1 = htole32(AQ_TXDESC_CTL1_TYPE_TXC |
+ (m->m_pkthdr.ether_vtag << AQ_TXDESC_CTL1_VLAN_SHIFT));
+ txd->ctl2 = 0;
+
+ ctl1 |= AQ_TXDESC_CTL1_CMD_VLAN;
+ ctl2 |= AQ_TXDESC_CTL2_CTX_EN;
+
+ idx++;
+ if (idx == AQ_TXD_NUM)
+ idx = 0;
+ used++;
+ }
+#endif
for (i = 0; i < as->as_map->dm_nsegs; i++) {