PERFORCE change 132649 for review
    Hans Petter Selasky 
    hselasky at FreeBSD.org
       
    Sun Jan  6 14:29:05 PST 2008
    
    
  
http://perforce.freebsd.org/chv.cgi?CH=132649
Change 132649 by hselasky at hselasky_laptop001 on 2008/01/06 22:28:36
	
	Upgrade "if_cdce" to support the new 512x4 protocol.
	
	o I should have started by reverting CDCE to get rid of
	the initial MF protocol, instead of working forward. Basically
	there are two new callbacks "cdce_bulk_write_512x4_callback" and
	"cdce_bulk_read_512x4_callback". The other callbacks do the
	same like before.
	
	o Added some code to set the correct alternate setting.
	
	o The protocol has been tested on a AT91 KB9202B board and
	there was a noticable increase in the number of frames
	transmitted per second when doing a ping flood !
Affected files ...
.. //depot/projects/usb/src/sys/dev/usb/if_cdce.c#45 edit
.. //depot/projects/usb/src/sys/dev/usb/if_cdcereg.h#16 edit
Differences ...
==== //depot/projects/usb/src/sys/dev/usb/if_cdce.c#45 (text+ko) ====
@@ -115,15 +115,15 @@
 		.type = UE_BULK,
 		.endpoint = UE_ADDR_ANY,
 		.direction = UE_DIR_OUT,
-		.frames = CDCE_ETH_FRAMES_MAX,
-		.bufsize = (MCLBYTES * CDCE_ETH_FRAMES_MAX),
+		.frames = CDCE_512X4_FRAGS_MAX + 1,
+		.bufsize = (CDCE_512X4_FRAMES_MAX * MCLBYTES) + sizeof(usb_cdc_mf_eth_512x4_header_t),
 		.if_index = 0,
 		/* Host Mode */
 		.mh.flags = {.pipe_bof = 1,.force_short_xfer = 1,.ext_buffer = 1,},
 		.mh.callback = &cdce_bulk_write_callback,
 		.mh.timeout = 10000,	/* 10 seconds */
 		/* Device Mode */
-		.md.flags = {.pipe_bof = 1,.short_xfer_ok = 1,.ext_buffer = 1,.short_frames_ok = 1,},
+		.md.flags = {.pipe_bof = 1,.short_xfer_ok = 1,.ext_buffer = 1,},
 		.md.callback = &cdce_bulk_read_callback,
 		.md.timeout = 0,	/* no timeout */
 	},
@@ -132,11 +132,11 @@
 		.type = UE_BULK,
 		.endpoint = UE_ADDR_ANY,
 		.direction = UE_DIR_IN,
-		.frames = CDCE_ETH_FRAMES_MAX,
-		.bufsize = (MCLBYTES * CDCE_ETH_FRAMES_MAX),
+		.frames = CDCE_512X4_FRAGS_MAX + 1,
+		.bufsize = (CDCE_512X4_FRAMES_MAX * MCLBYTES) + sizeof(usb_cdc_mf_eth_512x4_header_t),
 		.if_index = 0,
 		/* Host Mode */
-		.mh.flags = {.pipe_bof = 1,.short_xfer_ok = 1,.ext_buffer = 1,.short_frames_ok = 1,},
+		.mh.flags = {.pipe_bof = 1,.short_xfer_ok = 1,.ext_buffer = 1,},
 		.mh.callback = &cdce_bulk_read_callback,
 		.mh.timeout = 0,	/* no timeout */
 		/* Device Mode */
@@ -280,9 +280,11 @@
 	const usb_cdc_union_descriptor_t *ud;
 	const usb_cdc_ethernet_descriptor_t *ue;
 	const usb_interface_descriptor_t *id;
+	usb_descriptor_t *desc = NULL;
 	const struct cdce_type *t;
 	struct ifnet *ifp;
 	int error;
+	uint8_t alt_index;
 	uint8_t i;
 	uint8_t eaddr[ETHER_ADDR_LEN];
 	uint8_t eaddr_str[USB_STRING_DESC_LEN(ETHER_ADDR_LEN * 2) + 1];
@@ -298,6 +300,51 @@
 	if (t) {
 		sc->sc_flags = t->cdce_flags;
 	}
+	/* search for alternate settings */
+	if (uaa->usb_mode == USB_MODE_HOST) {
+
+		id = uaa->iface->idesc;
+		i = id->bInterfaceNumber;
+		alt_index = 0;
+		while ((desc = usbd_desc_foreach(
+			usbd_get_config_descriptor(uaa->device), desc))) {
+			id = (void *)desc;
+			if ((id->bDescriptorType == UDESC_INTERFACE) &&
+			    (id->bLength >= sizeof(*id))) {
+				if (id->bInterfaceNumber != i) {
+					alt_index = 0;
+					break;
+				}
+				if ((id->bInterfaceClass == UICLASS_CDC) &&
+				    (id->bInterfaceSubClass == 
+				     UISUBCLASS_ETHERNET_NETWORKING_CONTROL_MODEL) &&
+				    (id->bInterfaceProtocol == UIPROTO_CDC_ETH_512X4)) {
+
+					alt_index = id->bAlternateSetting;
+					/*
+					 * We want this alt setting hence
+					 * the protocol supports multi
+					 * sub-framing !
+					 */
+					break;
+				}
+			}
+		}
+
+		if (alt_index > 0) {
+
+			error = usbd_set_alt_interface_index(uaa->device,
+			    uaa->iface_index, alt_index);
+			if (error) {
+				device_printf(dev, "Could not set alternate "
+				    "setting, error = %s\n", usbd_errstr(error));
+				return (EINVAL);
+			}
+		}
+	}
+	/* get the interface subclass we are using */
+	sc->sc_iface_protocol = uaa->iface->idesc->bInterfaceProtocol;
+
 	usbd_set_device_desc(dev);
 
 	snprintf(sc->sc_name, sizeof(sc->sc_name), "%s",
@@ -459,8 +506,13 @@
 	ifp->if_start = cdce_start_cb;
 	ifp->if_init = cdce_init_cb;
 	ifp->if_baudrate = 11000000;
-	IFQ_SET_MAXLEN(&ifp->if_snd, CDCE_IFQ_MAXLEN);
-	ifp->if_snd.ifq_drv_maxlen = CDCE_IFQ_MAXLEN;
+	if (sc->sc_iface_protocol == UIPROTO_CDC_ETH_512X4) {
+	    IFQ_SET_MAXLEN(&ifp->if_snd, CDCE_512X4_IFQ_MAXLEN);
+	    ifp->if_snd.ifq_drv_maxlen = CDCE_512X4_IFQ_MAXLEN;
+	} else {
+	    IFQ_SET_MAXLEN(&ifp->if_snd, IFQ_MAXLEN);
+	    ifp->if_snd.ifq_drv_maxlen = IFQ_MAXLEN;
+	}
 	IFQ_SET_READY(&ifp->if_snd);
 
 	/* no IFM type for 11Mbps USB, so go with 10baseT */
@@ -547,7 +599,62 @@
 	return;
 }
 
+static uint32_t
+cdce_m_frags(struct mbuf *m)
+{
+  uint32_t temp = 1;
+  while ((m = m->m_next)) {
+	temp ++;
+  }
+  return (temp);
+}
+
+static void
+cdce_fwd_mq(struct cdce_softc *sc, struct cdce_mq *mq)
+{
+	struct mbuf *m;
+	struct ifnet *ifp = sc->sc_ifp;
+
+	if (mq->ifq_head) {
+
+		mtx_unlock(&(sc->sc_mtx));
+
+		while (1) {
+
+			_IF_DEQUEUE(mq, m);
+
+			if (m == NULL)
+				break;
+
+			(ifp->if_input) (ifp, m);
+		}
+
+		mtx_lock(&(sc->sc_mtx));
+	}
+	return;
+}
+
 static void
+cdce_free_mq(struct cdce_mq *mq)
+{
+	struct mbuf *m;
+
+	if (mq->ifq_head) {
+
+		while (1) {
+
+			_IF_DEQUEUE(mq, m);
+
+			if (m == NULL)
+				break;
+
+			m_freem(m);
+		}
+	}
+	return;
+}
+
+static void
 cdce_bulk_write_clear_stall_callback(struct usbd_xfer *xfer)
 {
 	struct cdce_softc *sc = xfer->priv_sc;
@@ -561,38 +668,140 @@
 }
 
 static void
-cdce_free_mbufs(struct mbuf **ppm)
+cdce_bulk_write_512x4_callback(struct usbd_xfer *xfer)
 {
+	struct cdce_softc *sc = xfer->priv_sc;
+	struct ifnet *ifp = sc->sc_ifp;
 	struct mbuf *m;
+	struct mbuf *mt;
 	uint16_t x;
+	uint16_t y;
+	uint16_t flen;
+
+	switch (USBD_GET_STATE(xfer)) {
+	case USBD_ST_TRANSFERRED:
+		DPRINTF(sc, 10, "transfer complete: "
+		    "%u bytes in %u fragments and %u frames\n",
+		xfer->actlen, xfer->nframes, sc->sc_tx_mq.ifq_len);
+
+		/* update packet counter */
+		ifp->if_opackets += sc->sc_tx_mq.ifq_len;
+
+		/* free all previous mbufs */
+		cdce_free_mq(&(sc->sc_tx_mq));
+
+	case USBD_ST_SETUP:
+tr_setup:
+		if (xfer->flags.stall_pipe &&
+		    (xfer->flags_int.usb_mode == USB_MODE_HOST)) {
+			/* try to clear stall */
+			usbd_transfer_start(sc->sc_xfer[2]);
+			break;
+		}
+
+		x = 0; /* number of frames */
+		y = 1; /* number of fragments */
+
+		while (x != CDCE_512X4_FRAMES_MAX) {
 
-	/* free all previous mbufs */
-	for (x = 0; x != CDCE_ETH_FRAMES_MAX; x++) {
-		m = ppm[x];
-		if (m) {
-			m_freem(m);
-			ppm[x] = NULL;
-		} else {
-			if (x != 0) {
+			IFQ_DRV_DEQUEUE(&(ifp->if_snd), m);
+
+			if (m == NULL) {
 				break;
 			}
+			if (m->m_pkthdr.len > MCLBYTES) {
+				m_freem(m);
+				ifp->if_oerrors++;
+				continue;
+			}
+			if (cdce_m_frags(m) > CDCE_512X4_FRAME_FRAG_MAX) {
+				mt = m_defrag(m, M_DONTWAIT);
+				if (mt == NULL) {
+					m_freem(m);
+					ifp->if_oerrors++;
+					continue;
+				}
+				m = mt;
+			}
+
+			_IF_ENQUEUE(&(sc->sc_tx_mq), m);
+
+			/*
+			 * if there's a BPF listener, bounce a copy
+			 * of this frame to him:
+			 */
+			BPF_MTAP(ifp, m);
+
+#if (CDCE_512X4_FRAG_LENGTH_MASK < MCLBYTES)
+#error "(CDCE_512X4_FRAG_LENGTH_MASK < MCLBYTES)"
+#endif
+			do {
+
+			    flen = m->m_len & CDCE_512X4_FRAG_LENGTH_MASK;
+			    xfer->frlengths[y] = m->m_len;
+			    usbd_set_frame_data(xfer, m->m_data, y);
+
+			    if (m->m_next == NULL) {
+				flen |= CDCE_512X4_FRAG_LAST_MASK;
+			    }
+
+			    USETW(sc->sc_tx.hdr.wFragLength[y-1], flen);
+
+			    y++;
+
+			} while ((m = m->m_next));
+
+			x++;
+		}
+
+		if (y == 1) {
+		    /* no data to transmit */
+		    break;
+		}
+
+		/* fill in Signature */
+		sc->sc_tx.hdr.bSig[0] = 'F';
+		sc->sc_tx.hdr.bSig[1] = 'L';
+
+		/* 
+		 * We ensure that the header results in a short packet
+		 * by making the length odd !
+		 */
+		USETW(sc->sc_tx.hdr.wFragLength[y-1], 0);
+		xfer->frlengths[0] = CDCE_512X4_FRAG_LENGTH_OFFSET + ((y-1) * 2) + 1;
+		usbd_set_frame_data(xfer, &(sc->sc_tx.hdr), 0);
+		xfer->nframes = y;
+		usbd_start_hardware(xfer);
+		break;
+
+	default:			/* Error */
+		DPRINTF(sc, 10, "transfer error, %s\n",
+		    usbd_errstr(xfer->error));
+
+		/* update error counter */
+		ifp->if_oerrors += sc->sc_tx_mq.ifq_len;
+
+		/* free all previous mbufs */
+		cdce_free_mq(&(sc->sc_tx_mq));
+
+		if (xfer->error != USBD_CANCELLED) {
+			/* try to clear stall first */
+			xfer->flags.stall_pipe = 1;
+			goto tr_setup;
 		}
+		break;
 	}
 	return;
 }
 
 static void
-cdce_bulk_write_callback(struct usbd_xfer *xfer)
+cdce_bulk_write_std_callback(struct usbd_xfer *xfer)
 {
 	struct cdce_softc *sc = xfer->priv_sc;
 	struct ifnet *ifp = sc->sc_ifp;
 	struct mbuf *m;
 	struct mbuf *mt;
 	uint32_t crc;
-	uint32_t x;
-
-	/* free all previous mbufs */
-	cdce_free_mbufs(sc->sc_tx_mbufs);
 
 	switch (USBD_GET_STATE(xfer)) {
 	case USBD_ST_TRANSFERRED:
@@ -602,44 +811,22 @@
 
 		ifp->if_opackets++;
 
+		/* free all previous mbufs */
+		cdce_free_mq(&(sc->sc_tx_mq));
+
 	case USBD_ST_SETUP:
 tr_setup:
 		if (xfer->flags.stall_pipe &&
 		    (xfer->flags_int.usb_mode == USB_MODE_HOST)) {
 			usbd_transfer_start(sc->sc_xfer[2]);
-			goto done;
+			break;
 		}
-#ifdef CDCE_MF_ENABLE
-		x = 1;
-#else
-		x = 0;
-#endif
-		while (x != CDCE_ETH_FRAMES_MAX) {
 
 			IFQ_DRV_DEQUEUE(&(ifp->if_snd), m);
 
 			if (m == NULL) {
-#ifdef CDCE_DO_BENCHMARK
-				/* send dummy ethernet frames */
-				usbd_set_frame_data(xfer,
-				    &(sc->sc_rx_dump), x);
-				xfer->frlengths[x] =
-				    (sizeof(sc->sc_rx_dump) / 2) + (x & 63) - 1;
-				x++;
-				continue;
-#else
 				break;
-#endif
 			}
-			if (m->m_pkthdr.len < sizeof(struct ether_header)) {
-				/*
-				 * frames of this size have special meaning
-				 * - filter away
-				 */
-				m_freem(m);
-				ifp->if_oerrors++;
-				continue;
-			}
 			if (sc->sc_flags & CDCE_FLAG_ZAURUS) {
 				/*
 				 * Zaurus wants a 32-bit CRC appended to
@@ -652,27 +839,27 @@
 				if (!m_append(m, 4, (void *)&crc)) {
 					m_freem(m);
 					ifp->if_oerrors++;
-					continue;
+					goto tr_setup;
 				}
-				m->m_pkthdr.len += 4;
 			}
 			if (m->m_len != m->m_pkthdr.len) {
 				mt = m_defrag(m, M_DONTWAIT);
 				if (mt == NULL) {
 					m_freem(m);
 					ifp->if_oerrors++;
-					continue;
+					goto tr_setup;
 				}
 				m = mt;
 			}
 			if (m->m_pkthdr.len > MCLBYTES) {
 				m->m_pkthdr.len = MCLBYTES;
 			}
-			sc->sc_tx_mbufs[x] = m;
 
-			xfer->frlengths[x] = m->m_len;
+			_IF_ENQUEUE(&(sc->sc_tx_mq), m);
 
-			usbd_set_frame_data(xfer, m->m_data, x);
+			xfer->frlengths[0] = m->m_len;
+			usbd_set_frame_data(xfer, m->m_data, 0);
+			xfer->nframes = 1;
 
 			/*
 			 * if there's a BPF listener, bounce a copy
@@ -680,51 +867,41 @@
 			 */
 			BPF_MTAP(ifp, m);
 
-			x++;
-		}
-
-		xfer->nframes = x;
-
-#ifdef CDCE_MF_ENABLE
-		if (x == 1) {
-			/* nothing to do */
-			goto done;
-		}
-		/* fill out the Multi Frame header */
-		usbd_set_frame_data(xfer, &(sc->sc_tx_eth.hdr), 0);
-		xfer->frlengths[0] = sizeof(sc->sc_tx_eth.hdr);
-		sc->sc_tx_eth.hdr.bSig0[0] = 'M';
-		sc->sc_tx_eth.hdr.bSig0[1] = 'F';
-		x--;
-		/* tell the peer how many frames are coming */
-		x += ifp->if_snd.ifq_drv_len;
-		USETDW(sc->sc_tx_eth.hdr.dwFramesAhead, x);
-		x = ~x;
-		USETDW(sc->sc_tx_eth.hdr.dwFramesAheadInverse, x);
-#else
-		if (x == 0) {
-			/* nothing to do */
-			goto done;
-		}
-#endif
 		usbd_start_hardware(xfer);
+		break;
 
-done:
-		return;
-
 	default:			/* Error */
 		DPRINTF(sc, 10, "transfer error, %s\n",
 		    usbd_errstr(xfer->error));
 
+		/* free all previous mbufs */
+		cdce_free_mq(&(sc->sc_tx_mq));
+		ifp->if_oerrors++;
+
 		if (xfer->error != USBD_CANCELLED) {
 			/* try to clear stall first */
 			xfer->flags.stall_pipe = 1;
 			goto tr_setup;
 		}
-		ifp->if_oerrors++;
-		return;
+		break;
+	}
+	return;
+}
+
+static void
+cdce_bulk_write_callback(struct usbd_xfer *xfer)
+{
+	struct cdce_softc *sc = xfer->priv_sc;
 
+	/* first call - set the correct callback */
+	if (sc->sc_iface_protocol == UIPROTO_CDC_ETH_512X4) {
+		xfer->flags.force_short_xfer = 0;
+		xfer->callback = &cdce_bulk_write_512x4_callback;
+	} else {
+		xfer->callback = &cdce_bulk_write_std_callback;
 	}
+	(xfer->callback) (xfer);
+	return;
 }
 
 static int32_t
@@ -883,16 +1060,24 @@
 }
 
 static void
-cdce_bulk_read_callback(struct usbd_xfer *xfer)
+cdce_bulk_read_512x4_callback(struct usbd_xfer *xfer)
 {
-	struct cdce_mq mq = {NULL, NULL, 0};
 	struct cdce_softc *sc = xfer->priv_sc;
 	struct ifnet *ifp = sc->sc_ifp;
 	struct mbuf *m;
-	usb_cdc_mf_eth_header_t *mf_hdr;
-	uint32_t x;
-	uint32_t ta;
-	uint32_t tb;
+	void *data_ptr;
+	uint32_t offset;
+	uint16_t x;
+	uint16_t y;
+	uint16_t z;
+	uint16_t rx_frags;
+	uint16_t flen;
+	uint8_t fwd_mq;
+	uint8_t free_mq;
+
+	fwd_mq = 0;
+	free_mq = 0;
+	rx_frags = 0;
 
 	switch (USBD_GET_STATE(xfer)) {
 	case USBD_ST_TRANSFERRED:
@@ -900,172 +1085,312 @@
 		DPRINTF(sc, 0, "received %u bytes in %u frames\n",
 		    xfer->actlen, xfer->aframes);
 
-		for (x = 0; x != xfer->nframes; x++) {
+		/* check state */
+		if (!(sc->sc_flags & CDCE_FLAG_RX_DATA)) {
 
-			if (sc->sc_flags & CDCE_FLAG_ZAURUS) {
-
-				/* Strip off CRC added by Zaurus */
-				if (xfer->frlengths[x] >= MAX(4, 14)) {
-					xfer->frlengths[x] -= 4;
-				}
+			/* verify the header */
+			if ((xfer->actlen < CDCE_512X4_FRAG_LENGTH_OFFSET) ||
+			    (sc->sc_rx.hdr.bSig[0] != 'F') ||
+			    (sc->sc_rx.hdr.bSig[1] != 'L')) {
+				/* try to clear stall first */
+				xfer->flags.stall_pipe = 1;
+				goto tr_setup;
 			}
-			m = sc->sc_rx_mbufs[x];
-			sc->sc_rx_mbufs[x] = NULL;
-			if (m == NULL) {
-				continue;
+			rx_frags = (xfer->actlen - 
+			    CDCE_512X4_FRAG_LENGTH_OFFSET) / 2;
+			if (rx_frags != 0) {
+				/* start receiving data */
+				sc->sc_flags |= CDCE_FLAG_RX_DATA;
 			}
-			if (xfer->frlengths[x] < sizeof(struct ether_header)) {
 
-#ifdef CDCE_MF_ENABLE
-				if (xfer->frlengths[x] >=
-				    CDC_MF_ETH_HEADER_SIZE) {
+			DPRINTF(sc, 0, "doing %u fragments\n", rx_frags);
 
-					mf_hdr = (void *)(m->m_data);
+		} else {
+			/* we are done receiving data */
+			sc->sc_flags &= ~CDCE_FLAG_RX_DATA;
+			fwd_mq = 1;
+		}
 
-					/*
-					 * decode and verify the multi
-					 * frame header
-					 */
+	case USBD_ST_SETUP:
+tr_setup:
+		if (xfer->flags.stall_pipe) {
 
-					ta = UGETDW(mf_hdr->dwFramesAhead);
-					tb = UGETDW(mf_hdr->dwFramesAheadInverse);
-					tb ^= ta;
+			/* we are done */
+			sc->sc_flags &= ~CDCE_FLAG_RX_DATA;
 
-					DPRINTF(sc, 0, "ta = 0x%08x, "
-					    "tb = 0x%08x\n", ta, tb);
+			if (xfer->flags_int.usb_mode == USB_MODE_HOST) {
+				usbd_transfer_start(sc->sc_xfer[3]);
+				free_mq = 1;
+				break;
+			}
+		}
+		/* we expect a Multi Frame Ethernet Header */
+		if (!(sc->sc_flags & CDCE_FLAG_RX_DATA)) {
+			DPRINTF(sc, 0, "expecting length header\n");
+			usbd_set_frame_data(xfer, &(sc->sc_rx.hdr), 0);
+			xfer->frlengths[0] = sizeof(sc->sc_rx.hdr);
+			xfer->nframes = 1;
+			xfer->flags.short_xfer_ok = 1;
+			usbd_start_hardware(xfer);
+			free_mq = 1;
+			break;
+		}
+		/* verify number of fragments */
+		if (rx_frags > CDCE_512X4_FRAGS_MAX) {
+			/* try to clear stall first */
+			xfer->flags.stall_pipe = 1;
+			goto tr_setup;
+		}
 
-					/*
-					 * check if we have a multi frame
-					 * header
-					 */
-					if ((tb == 0xFFFFFFFF) &&
-					    (mf_hdr->bSig0[0] == 'M') &&
-					    (mf_hdr->bSig0[1] == 'F')) {
+		/* check if the last fragment does not complete a frame */
+		x = rx_frags - 1;
+		flen = UGETW(sc->sc_rx.hdr.wFragLength[x]);
+		if (!(flen & CDCE_512X4_FRAG_LAST_MASK)) {
+			DPRINTF(sc, 0, "no last frag mask\n");
+			/* try to clear stall first */
+			xfer->flags.stall_pipe = 1;
+			goto tr_setup;
+		}
+		/*
+		 * Setup a new USB transfer chain to receive all the
+		 * IP-frame fragments, automagically defragged :
+		 */
+		x = 0;
+		y = 0;
+		while (1) {
 
-						DPRINTF(sc, 0, "frames ahead "
-						    "= %u\n", ta);
+			z = x;
+			offset = 0;
 
-						sc->sc_rx_frames_ahead = ta;
-					}
+			/* precompute the frame length */
+			while (1) {
+				flen = UGETW(sc->sc_rx.hdr.wFragLength[z]);
+				offset += (flen & CDCE_512X4_FRAG_LENGTH_MASK);
+				if (flen & CDCE_512X4_FRAG_LAST_MASK) {
+					break;
 				}
-#endif
+				z++;
+			}
 
-				m_freem(m);
-				continue;
-			} else {
+			if (offset >= sizeof(struct ether_header)) {
 				/*
-				 * we received a frame - decrement frames
-				 * ahead
+				 * allocate a suitable memory buffer, if
+				 * possible
 				 */
-				if (sc->sc_rx_frames_ahead) {
-					sc->sc_rx_frames_ahead--;
+				if (offset > (MCLBYTES - ETHER_ALIGN)) {
+					/* try to clear stall first */
+					xfer->flags.stall_pipe = 1;
+					goto tr_setup;
+				} if (offset > (MHLEN - ETHER_ALIGN)) {
+					m = m_getcl(M_DONTWAIT, MT_DATA, M_PKTHDR);
+				} else {
+					m = m_gethdr(M_DONTWAIT, MT_DATA);
 				}
+			} else {
+				m = NULL;	/* dump it */
 			}
 
-			ifp->if_ipackets++;
-			m->m_pkthdr.rcvif = ifp;
-			m->m_pkthdr.len = m->m_len = xfer->frlengths[x];
+			DPRINTF(sc, 16, "frame %u, length = %u \n", y, offset);
+
+			/* check if we have a buffer */
+			if (m) {
+				m->m_data = USBD_ADD_BYTES(m->m_data, ETHER_ALIGN);
+				m->m_pkthdr.rcvif = ifp;
+				m->m_pkthdr.len = m->m_len = offset;
+
+				/* enqueue */
+				_IF_ENQUEUE(&(sc->sc_rx_mq), m);
+
+				data_ptr = m->m_data;
+				ifp->if_ipackets++;
+			} else {
+				data_ptr = sc->sc_rx.data;
+				ifp->if_ierrors++;
+			}
+
+			/* setup the RX chain */
+			offset = 0;
+			while (1) {
+
+				flen = UGETW(sc->sc_rx.hdr.wFragLength[x]);
+
+				usbd_set_frame_data(xfer,
+				    USBD_ADD_BYTES(data_ptr, offset), x);
+
+				xfer->frlengths[x] =
+				    (flen & CDCE_512X4_FRAG_LENGTH_MASK);
 
-			/* enqueue */
-			_IF_ENQUEUE(&(mq), m);
-		}
+				DPRINTF(sc, 16, "length[%u] = %u\n", 
+					x, xfer->frlengths[x]);
 
-	case USBD_ST_SETUP:
-tr_setup:
-		if (xfer->flags.stall_pipe) {
+				offset += xfer->frlengths[x];
 
-			/* reset number of frames ahead */
-			sc->sc_rx_frames_ahead = 0;
+				x++;
 
-			if (xfer->flags_int.usb_mode == USB_MODE_HOST) {
-				usbd_transfer_start(sc->sc_xfer[3]);
-				/*
-				 * In case the "stall_pipe" flag was set
-				 * while transferring data, we need to go to
-				 * the "tr_if_input" afterwards!
-				 */
-				goto tr_if_input;
+				if (flen & CDCE_512X4_FRAG_LAST_MASK) {
+					break;
+				}
 			}
-		}
-		/* setup a new USB transfer chain */
 
-		ta = sc->sc_rx_frames_ahead;
+			y++;
 
-		if (ta == 0) {
-			/* always receive at least one frame */
-			ta = 1;
-		}
-		if (ta > CDCE_ETH_FRAMES_MAX) {
-			ta = CDCE_ETH_FRAMES_MAX;
-		}
-		for (x = 0; x != ta; x++) {
-			m = usbd_ether_get_mbuf();
-			if (m == NULL) {
+			if (x == rx_frags) {
 				break;
 			}
-			usbd_set_frame_data(xfer, m->m_data, x);
+			if (y == CDCE_512X4_FRAMES_MAX) {
+				/* try to clear stall first */
+				xfer->flags.stall_pipe = 1;
+				goto tr_setup;
+			}
+		}
+
+		DPRINTF(sc, 0, "nframes = %u\n", x);
+
+		xfer->nframes = x;
+		xfer->flags.short_xfer_ok = 0;
+		usbd_start_hardware(xfer);
+		break;
+
+	default:			/* Error */
+		DPRINTF(sc, 0, "error = %s\n",
+		    usbd_errstr(xfer->error));
 
-			xfer->frlengths[x] = m->m_len;
-			sc->sc_rx_mbufs[x] = m;
+		if (xfer->error != USBD_CANCELLED) {
+			/* try to clear stall first */
+			xfer->flags.stall_pipe = 1;
+			goto tr_setup;
 		}
+		free_mq = 1;
+		break;
+	}
+
+	/*
+	 * At the end of a USB callback it is always safe to unlock
+	 * the private mutex of a device! 
+	 *
+	 *
+	 * By safe we mean that if "usbd_transfer_stop()" is called,
+	 * we will get a callback having the error code
+	 * USBD_CANCELLED.
+	 */
+	if (fwd_mq) {
+		cdce_fwd_mq(sc, &(sc->sc_rx_mq));
+	}
+
+	if (free_mq) {
+		cdce_free_mq(&(sc->sc_rx_mq));
+	}
+	return;
+}
+
+static void
+cdce_bulk_read_std_callback(struct usbd_xfer *xfer)
+{
+	struct cdce_softc *sc = xfer->priv_sc;
+	struct ifnet *ifp = sc->sc_ifp;
+	struct mbuf *m;
+	struct mbuf *m_rx = NULL;
 
-		for (; x != ta; x++) {
+	switch (USBD_GET_STATE(xfer)) {
+	case USBD_ST_TRANSFERRED:
 
-			/*
-			 * We are out of mbufs and need to dump all the
-			 * received data !
-			 */
-			usbd_set_frame_data(xfer, &(sc->sc_rx_dump), x);
-			xfer->frlengths[x] = sizeof(sc->sc_rx_dump);
-		}
-		xfer->nframes = ta;
-		usbd_start_hardware(xfer);
+		DPRINTF(sc, 0, "received %u bytes in %u frames\n",
+		    xfer->actlen, xfer->aframes);
 
-		/*
-		 * At the end of a USB callback it is always safe to
-		 * unlock the private mutex of a device! That is why
-		 * we do the "if_input" here, and not some lines up!
-		 *
-		 * By safe we mean that if "usbd_transfer_stop()" is
-		 * called, we will get a callback having the error
-		 * code USBD_CANCELLED.
-		 */
-tr_if_input:
-		if (mq.ifq_head) {
+		if (sc->sc_flags & CDCE_FLAG_ZAURUS) {
 
-			mtx_unlock(&(sc->sc_mtx));
+			/* Strip off CRC added by Zaurus */
+			if (xfer->frlengths[0] >= MAX(4, 14)) {
+				xfer->frlengths[0] -= 4;
+			}
+		}
 
-			while (1) {
+		_IF_DEQUEUE(&(sc->sc_rx_mq), m);
 
-				_IF_DEQUEUE(&(mq), m);
+		if (m) {
 
-				if (m == NULL)
-					break;
+			if (xfer->frlengths[0] < sizeof(struct ether_header)) {
+				m_freem(m);
+				goto tr_setup;
+			}
+			ifp->if_ipackets++;
+			m->m_pkthdr.rcvif = ifp;
+			m->m_pkthdr.len = m->m_len = xfer->frlengths[0];
+			m_rx = m;
+		}
+	case USBD_ST_SETUP:
+tr_setup:
+		if (xfer->flags.stall_pipe) {
 
-				(ifp->if_input) (ifp, m);
+			if (xfer->flags_int.usb_mode == USB_MODE_HOST) {
+				usbd_transfer_start(sc->sc_xfer[3]);
+				break;
 			}
+		}
+		m = usbd_ether_get_mbuf();
+		if (m == NULL) {
+
+			/*
+			 * We are out of mbufs and need to dump all the
+			 * received data !
+			 */
+			usbd_set_frame_data(xfer, &(sc->sc_rx.data), 0);
+			xfer->frlengths[0] = sizeof(sc->sc_rx.data);
 
-			mtx_lock(&(sc->sc_mtx));
+		} else {
+			usbd_set_frame_data(xfer, m->m_data, 0);
+			xfer->frlengths[0] = m->m_len;
+			_IF_ENQUEUE(&(sc->sc_rx_mq), m);
 		}
-		return;
+		xfer->nframes = 1;
+		usbd_start_hardware(xfer);
+		break;
 
 	default:			/* Error */
-		/* free all received data, if any */
-		cdce_free_mbufs(sc->sc_rx_mbufs);
+		DPRINTF(sc, 0, "error = %s\n",
+		    usbd_errstr(xfer->error));
 
-		/* reset number of frames ahead */
-		sc->sc_rx_frames_ahead = 0;
+		/* free all mbufs */
+		cdce_free_mq(&(sc->sc_rx_mq));
 
 		if (xfer->error != USBD_CANCELLED) {
 			/* try to clear stall first */
 			xfer->flags.stall_pipe = 1;
 			goto tr_setup;
 		}
-		DPRINTF(sc, 0, "bulk read error, %s\n",
-		    usbd_errstr(xfer->error));
 		return;
+	}
+
+	/*
+	 * At the end of a USB callback it is always safe to unlock
+	 * the private mutex of a device! That is why we do the
+	 * "if_input" here, and not some lines up!
+	 *
+	 * By safe we mean that if "usbd_transfer_stop()" is called,
+	 * we will get a callback having the error code
+	 * USBD_CANCELLED.
+	 */
+	if (m_rx) {
+		mtx_unlock(&(sc->sc_mtx));
+		(ifp->if_input) (ifp, m_rx);
+		mtx_lock(&(sc->sc_mtx));
+	}
+	return;
+}
+
+static void
+cdce_bulk_read_callback(struct usbd_xfer *xfer)
+{
+	struct cdce_softc *sc = xfer->priv_sc;
 
+	/* first call - set the correct callback */
+	if (sc->sc_iface_protocol == UIPROTO_CDC_ETH_512X4) {
+		xfer->callback = &cdce_bulk_read_512x4_callback;
+	} else {
+		xfer->callback = &cdce_bulk_read_std_callback;
 	}
+	(xfer->callback) (xfer);
+	return;
 }
 
 static int
@@ -1166,13 +1491,9 @@
 }
 
 static int
-cdce_handle_request(device_t dev, const void *req, void **pptr,
-    uint16_t *plen, uint16_t offset, uint8_t is_complete)
+cdce_handle_request(device_t dev,
+    const void *req, void **pptr, uint16_t *plen,
+    uint16_t offset, uint8_t is_complete)
 {
-#ifdef USB_DEBUG
-	struct cdce_softc *sc = device_get_softc(dev);
-
-	DPRINTF(sc, 0, "\n");
-#endif
-	return (ENXIO);			/* use builtin handling */
+	return (ENXIO); /* use builtin handler */
 }
==== //depot/projects/usb/src/sys/dev/usb/if_cdcereg.h#16 (text+ko) ====
@@ -37,21 +37,20 @@
 
 #define	CDCE_N_TRANSFER	6		/* units */
 #define	CDCE_IND_SIZE_MAX 32		/* bytes */
-#define	CDCE_ETH_FRAMES_MAX 64		/* USB ethernet acceleration factor */
-#define	CDCE_IFQ_MAXLEN MAX((2*CDCE_ETH_FRAMES_MAX), IFQ_MAXLEN)
>>> TRUNCATED FOR MAIL (1000 lines) <<<
    
    
More information about the p4-projects
mailing list