summaryrefslogtreecommitdiff
path: root/sys/dev/pcmcia
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pcmcia')
-rw-r--r--sys/dev/pcmcia/if_xe.c33
1 files changed, 24 insertions, 9 deletions
diff --git a/sys/dev/pcmcia/if_xe.c b/sys/dev/pcmcia/if_xe.c
index a92ff94e315..76b6ff95216 100644
--- a/sys/dev/pcmcia/if_xe.c
+++ b/sys/dev/pcmcia/if_xe.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_xe.c,v 1.8 1999/08/13 20:36:38 fgsch Exp $ */
+/* $OpenBSD: if_xe.c,v 1.9 1999/09/16 11:28:42 niklas Exp $ */
/*
* Copyright (c) 1999 Niklas Hallqvist, C Stone, Job de Haas
@@ -112,6 +112,7 @@
#define XED_CONFIG 0x1
#define XED_MII 0x2
#define XED_INTR 0x4
+#define XED_FIFO 0x8
#ifndef XEDEBUG_DEF
#define XEDEBUG_DEF (XED_CONFIG|XED_INTR)
@@ -1111,6 +1112,7 @@ xe_start(ifp)
bus_addr_t offset = sc->sc_offset;
unsigned int s, len, pad = 0;
struct mbuf *m0, *m;
+ u_int16_t space;
/* Don't transmit if interface is busy or not running. */
if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING)
@@ -1125,18 +1127,27 @@ xe_start(ifp)
if (!(m0->m_flags & M_PKTHDR))
panic("xe_start: no header mbuf");
- IF_DEQUEUE(&ifp->if_snd, m0);
len = m0->m_pkthdr.len;
+ /* Pad to ETHER_MIN_LEN - ETHER_CRC_LEN. */
+ if (len < ETHER_MIN_LEN - ETHER_CRC_LEN)
+ pad = ETHER_MIN_LEN - ETHER_CRC_LEN - len;
+
+ space = bus_space_read_2(bst, bsh, offset + TSO0) & 0x7fff;
+ if (len + pad + 2 > space) {
+ DPRINTF(XED_FIFO,
+ ("%s: not enough space in output FIFO (%d > %d)\n",
+ sc->sc_dev.dv_xname, len + pad + 2, space));
+ return;
+ }
+
+ IF_DEQUEUE(&ifp->if_snd, m0);
+
#if NBPFILTER > 0
if (ifp->if_bpf)
bpf_mtap(ifp->if_bpf, m0);
#endif
- /* Pad to ETHER_MIN_LEN - ETHER_CRC_LEN. */
- if (len < ETHER_MIN_LEN - ETHER_CRC_LEN)
- pad = ETHER_MIN_LEN - ETHER_CRC_LEN - len;
-
/*
* Do the output at splhigh() so that an interrupt from another device
* won't cause a FIFO underrun.
@@ -1156,11 +1167,15 @@ xe_start(ifp)
MFREE(m, m0);
m = m0;
}
- for (; pad > 0; pad--)
- bus_space_write_1(bst, bsh, offset + EDP, 0);
-
if (sc->sc_flags & XEF_MOHAWK)
bus_space_write_1(bst, bsh, offset + CR, TX_PKT | ENABLE_INT);
+ else {
+ for (; pad > 1; pad -= 2)
+ bus_space_write_2(bst, bsh, offset + EDP, 0);
+ if (pad == 1)
+ bus_space_write_1(bst, bsh, offset + EDP, 0);
+ }
+
splx(s);
ifp->if_timer = 5;