aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/net/usb/usbnet.c45
-rw-r--r--include/linux/usb/usbnet.h6
2 files changed, 37 insertions, 14 deletions
diff --git a/drivers/net/usb/usbnet.c b/drivers/net/usb/usbnet.c
index c04d49e31f81..cff74b81a7d2 100644
--- a/drivers/net/usb/usbnet.c
+++ b/drivers/net/usb/usbnet.c
@@ -391,14 +391,19 @@ static inline void rx_process (struct usbnet *dev, struct sk_buff *skb)
391 goto error; 391 goto error;
392 // else network stack removes extra byte if we forced a short packet 392 // else network stack removes extra byte if we forced a short packet
393 393
394 if (skb->len) 394 if (skb->len) {
395 usbnet_skb_return (dev, skb); 395 /* all data was already cloned from skb inside the driver */
396 else { 396 if (dev->driver_info->flags & FLAG_MULTI_PACKET)
397 netif_dbg(dev, rx_err, dev->net, "drop\n"); 397 dev_kfree_skb_any(skb);
398error: 398 else
399 dev->net->stats.rx_errors++; 399 usbnet_skb_return(dev, skb);
400 skb_queue_tail (&dev->done, skb); 400 return;
401 } 401 }
402
403 netif_dbg(dev, rx_err, dev->net, "drop\n");
404error:
405 dev->net->stats.rx_errors++;
406 skb_queue_tail(&dev->done, skb);
402} 407}
403 408
404/*-------------------------------------------------------------------------*/ 409/*-------------------------------------------------------------------------*/
@@ -971,7 +976,8 @@ static void tx_complete (struct urb *urb)
971 struct usbnet *dev = entry->dev; 976 struct usbnet *dev = entry->dev;
972 977
973 if (urb->status == 0) { 978 if (urb->status == 0) {
974 dev->net->stats.tx_packets++; 979 if (!(dev->driver_info->flags & FLAG_MULTI_PACKET))
980 dev->net->stats.tx_packets++;
975 dev->net->stats.tx_bytes += entry->length; 981 dev->net->stats.tx_bytes += entry->length;
976 } else { 982 } else {
977 dev->net->stats.tx_errors++; 983 dev->net->stats.tx_errors++;
@@ -1044,8 +1050,13 @@ netdev_tx_t usbnet_start_xmit (struct sk_buff *skb,
1044 if (info->tx_fixup) { 1050 if (info->tx_fixup) {
1045 skb = info->tx_fixup (dev, skb, GFP_ATOMIC); 1051 skb = info->tx_fixup (dev, skb, GFP_ATOMIC);
1046 if (!skb) { 1052 if (!skb) {
1047 netif_dbg(dev, tx_err, dev->net, "can't tx_fixup skb\n"); 1053 if (netif_msg_tx_err(dev)) {
1048 goto drop; 1054 netif_dbg(dev, tx_err, dev->net, "can't tx_fixup skb\n");
1055 goto drop;
1056 } else {
1057 /* cdc_ncm collected packet; waits for more */
1058 goto not_drop;
1059 }
1049 } 1060 }
1050 } 1061 }
1051 length = skb->len; 1062 length = skb->len;
@@ -1067,13 +1078,18 @@ netdev_tx_t usbnet_start_xmit (struct sk_buff *skb,
1067 /* don't assume the hardware handles USB_ZERO_PACKET 1078 /* don't assume the hardware handles USB_ZERO_PACKET
1068 * NOTE: strictly conforming cdc-ether devices should expect 1079 * NOTE: strictly conforming cdc-ether devices should expect
1069 * the ZLP here, but ignore the one-byte packet. 1080 * the ZLP here, but ignore the one-byte packet.
1081 * NOTE2: CDC NCM specification is different from CDC ECM when
1082 * handling ZLP/short packets, so cdc_ncm driver will make short
1083 * packet itself if needed.
1070 */ 1084 */
1071 if (length % dev->maxpacket == 0) { 1085 if (length % dev->maxpacket == 0) {
1072 if (!(info->flags & FLAG_SEND_ZLP)) { 1086 if (!(info->flags & FLAG_SEND_ZLP)) {
1073 urb->transfer_buffer_length++; 1087 if (!(info->flags & FLAG_MULTI_PACKET)) {
1074 if (skb_tailroom(skb)) { 1088 urb->transfer_buffer_length++;
1075 skb->data[skb->len] = 0; 1089 if (skb_tailroom(skb)) {
1076 __skb_put(skb, 1); 1090 skb->data[skb->len] = 0;
1091 __skb_put(skb, 1);
1092 }
1077 } 1093 }
1078 } else 1094 } else
1079 urb->transfer_flags |= URB_ZERO_PACKET; 1095 urb->transfer_flags |= URB_ZERO_PACKET;
@@ -1122,6 +1138,7 @@ netdev_tx_t usbnet_start_xmit (struct sk_buff *skb,
1122 netif_dbg(dev, tx_err, dev->net, "drop, code %d\n", retval); 1138 netif_dbg(dev, tx_err, dev->net, "drop, code %d\n", retval);
1123drop: 1139drop:
1124 dev->net->stats.tx_dropped++; 1140 dev->net->stats.tx_dropped++;
1141not_drop:
1125 if (skb) 1142 if (skb)
1126 dev_kfree_skb_any (skb); 1143 dev_kfree_skb_any (skb);
1127 usb_free_urb (urb); 1144 usb_free_urb (urb);
diff --git a/include/linux/usb/usbnet.h b/include/linux/usb/usbnet.h
index 7ae27a473818..44842c8d38c0 100644
--- a/include/linux/usb/usbnet.h
+++ b/include/linux/usb/usbnet.h
@@ -97,6 +97,12 @@ struct driver_info {
97 97
98#define FLAG_LINK_INTR 0x0800 /* updates link (carrier) status */ 98#define FLAG_LINK_INTR 0x0800 /* updates link (carrier) status */
99 99
100/*
101 * Indicates to usbnet, that USB driver accumulates multiple IP packets.
102 * Affects statistic (counters) and short packet handling.
103 */
104#define FLAG_MULTI_PACKET 0x1000
105
100 /* init device ... can sleep, or cause probe() failure */ 106 /* init device ... can sleep, or cause probe() failure */
101 int (*bind)(struct usbnet *, struct usb_interface *); 107 int (*bind)(struct usbnet *, struct usb_interface *);
102 108