netdev
[Top] [All Lists]

Re: [PATCH] fix natsemi PCI mapping

To: Jeff Garzik <jgarzik@xxxxxxxxx>
Subject: Re: [PATCH] fix natsemi PCI mapping
From: "Steven J. Hill" <Steve.Hill@xxxxxxxxxxx>
Date: Tue, 16 Mar 2004 14:15:42 -0500
Cc: Netdev <netdev@xxxxxxxxxxx>, Tim Hockin <thockin@xxxxxxx>, Linux Kernel <linux-kernel@xxxxxxxxxxxxxxx>, ralf@xxxxxxxxxxxxxx
In-reply-to: <40574227.8020302@xxxxxxxxx>
References: <40574227.8020302@xxxxxxxxx>
Sender: netdev-bounce@xxxxxxxxxxx
User-agent: Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.6) Gecko/20040122 Debian/1.6-1
Jeff Garzik wrote:

Somebody wanna review and/or test?

Hey Jeff.

I have tested this on 2.4 and it works great on MIPS with one
minor change below. Remove the 16 byte alignment of the IP
header. I discovered this when trying to do a BOOTP and mount
my NFS root filesystem. The BOOTP never succeeds. Patch against
latest 2.4.25 attached.

-Steve

@@ -1467,13 +1469,16 @@
                struct sk_buff *skb;
                int entry = np->dirty_rx % RX_RING_SIZE;
                if (np->rx_skbuff[entry] == NULL) {
-                       skb = dev_alloc_skb(np->rx_buf_sz);
+                       unsigned int buflen = np->rx_buf_sz + RX_OFFSET;
+                       skb = dev_alloc_skb(buflen);
                        np->rx_skbuff[entry] = skb;
                        if (skb == NULL)
                                break; /* Better luck next round. */
                        skb->dev = dev; /* Mark as being used by this device. */
+                       /* 16 byte align the IP header */
+                       skb_reserve(skb, RX_OFFSET);
                        np->rx_dma[entry] = pci_map_single(np->pci_dev,
-                               skb->data, skb->len, PCI_DMA_FROMDEVICE);
+                               skb->tail, buflen, PCI_DMA_FROMDEVICE);
                        np->rx_ring[entry].addr = 
cpu_to_le32(np->rx_dma[entry]);
                }
diff -urN -X /home/sjhill/diff-exc linux-2.4.25/drivers/net/natsemi.c 
linux-2.4.25-patched/drivers/net/natsemi.c
--- linux-2.4.25/drivers/net/natsemi.c  Tue Mar 16 14:05:21 2004
+++ linux-2.4.25-patched/drivers/net/natsemi.c  Tue Mar 16 13:58:15 2004
@@ -175,6 +175,8 @@
 #define DRV_VERSION    "1.07+LK1.0.17"
 #define DRV_RELDATE    "Sep 27, 2002"
 
+#define RX_OFFSET      2
+
 /* Updated to recommendations in pci-skeleton v2.03. */
 
 /* The user-configurable values.
@@ -1466,13 +1467,14 @@
                struct sk_buff *skb;
                int entry = np->dirty_rx % RX_RING_SIZE;
                if (np->rx_skbuff[entry] == NULL) {
-                       skb = dev_alloc_skb(np->rx_buf_sz);
+                       unsigned int buflen = np->rx_buf_sz + RX_OFFSET;
+                       skb = dev_alloc_skb(buflen);
                        np->rx_skbuff[entry] = skb;
                        if (skb == NULL)
                                break; /* Better luck next round. */
                        skb->dev = dev; /* Mark as being used by this device. */
                        np->rx_dma[entry] = pci_map_single(np->pci_dev,
-                               skb->data, skb->len, PCI_DMA_FROMDEVICE);
+                               skb->tail, buflen, PCI_DMA_FROMDEVICE);
                        np->rx_ring[entry].addr = 
cpu_to_le32(np->rx_dma[entry]);
                }
                np->rx_ring[entry].cmd_status = cpu_to_le32(np->rx_buf_sz);
@@ -1542,6 +1544,7 @@
 static void drain_ring(struct net_device *dev)
 {
        struct netdev_private *np = dev->priv;
+       unsigned int buflen = np->rx_buf_sz + RX_OFFSET;
        int i;
 
        /* Free all the skbuffs in the Rx queue. */
@@ -1550,7 +1553,7 @@
                np->rx_ring[i].addr = 0xBADF00D0; /* An invalid address. */
                if (np->rx_skbuff[i]) {
                        pci_unmap_single(np->pci_dev,
-                               np->rx_dma[i], np->rx_skbuff[i]->len,
+                               np->rx_dma[i], buflen,
                                PCI_DMA_FROMDEVICE);
                        dev_kfree_skb(np->rx_skbuff[i]);
                }
@@ -1746,6 +1745,7 @@
        int entry = np->cur_rx % RX_RING_SIZE;
        int boguscnt = np->dirty_rx + RX_RING_SIZE - np->cur_rx;
        s32 desc_status = le32_to_cpu(np->rx_head_desc->cmd_status);
+       unsigned int buflen = np->rx_buf_sz + RX_OFFSET;
 
        /* If the driver owns the next entry it's a new packet. Send it up. */
        while (desc_status < 0) { /* e.g. & DescOwn */
@@ -1784,13 +1784,13 @@
                        /* Check if the packet is long enough to accept
                         * without copying to a minimally-sized skbuff. */
                        if (pkt_len < rx_copybreak
-                           && (skb = dev_alloc_skb(pkt_len + 2)) != NULL) {
+                           && (skb = dev_alloc_skb(pkt_len + RX_OFFSET)) != 
NULL) {
                                skb->dev = dev;
                                /* 16 byte align the IP header */
-                               skb_reserve(skb, 2);
+                               skb_reserve(skb, RX_OFFSET);
                                pci_dma_sync_single(np->pci_dev,
                                        np->rx_dma[entry],
-                                       np->rx_skbuff[entry]->len,
+                                       buflen,
                                        PCI_DMA_FROMDEVICE);
 #if HAS_IP_COPYSUM
                                eth_copy_and_sum(skb,
@@ -1802,8 +1802,7 @@
 #endif
                        } else {
                                pci_unmap_single(np->pci_dev, np->rx_dma[entry],
-                                       np->rx_skbuff[entry]->len,
-                                       PCI_DMA_FROMDEVICE);
+                                       buflen, PCI_DMA_FROMDEVICE);
                                skb_put(skb = np->rx_skbuff[entry], pkt_len);
                                np->rx_skbuff[entry] = NULL;
                        }
<Prev in Thread] Current Thread [Next in Thread>