sky2: large memory workaround.

This patch might fix problems with 4G or more of memory.
It stops the driver from doing a small optimization for Tx and Rx,
and instead always sets the high-page on tx/rx descriptors.

Fixes-bug: http://bugzilla.kernel.org/show_bug.cgi?id=9725

Signed-off-by: Stephen Hemminger <shemminger@linux-foundation.org>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
This commit is contained in:
Stephen Hemminger 2008-01-10 16:14:12 -08:00 committed by Jeff Garzik
parent dfd9a421be
commit 86c6887e6f
2 changed files with 10 additions and 18 deletions

View File

@ -944,7 +944,6 @@ static void tx_init(struct sky2_port *sky2)
le = get_tx_le(sky2); le = get_tx_le(sky2);
le->addr = 0; le->addr = 0;
le->opcode = OP_ADDR64 | HW_OWNER; le->opcode = OP_ADDR64 | HW_OWNER;
sky2->tx_addr64 = 0;
} }
static inline struct tx_ring_info *tx_le_re(struct sky2_port *sky2, static inline struct tx_ring_info *tx_le_re(struct sky2_port *sky2,
@ -978,13 +977,11 @@ static void sky2_rx_add(struct sky2_port *sky2, u8 op,
dma_addr_t map, unsigned len) dma_addr_t map, unsigned len)
{ {
struct sky2_rx_le *le; struct sky2_rx_le *le;
u32 hi = upper_32_bits(map);
if (sky2->rx_addr64 != hi) { if (sizeof(dma_addr_t) > sizeof(u32)) {
le = sky2_next_rx(sky2); le = sky2_next_rx(sky2);
le->addr = cpu_to_le32(hi); le->addr = cpu_to_le32(upper_32_bits(map));
le->opcode = OP_ADDR64 | HW_OWNER; le->opcode = OP_ADDR64 | HW_OWNER;
sky2->rx_addr64 = upper_32_bits(map + len);
} }
le = sky2_next_rx(sky2); le = sky2_next_rx(sky2);
@ -1480,7 +1477,6 @@ static int sky2_xmit_frame(struct sk_buff *skb, struct net_device *dev)
struct tx_ring_info *re; struct tx_ring_info *re;
unsigned i, len; unsigned i, len;
dma_addr_t mapping; dma_addr_t mapping;
u32 addr64;
u16 mss; u16 mss;
u8 ctrl; u8 ctrl;
@ -1493,15 +1489,12 @@ static int sky2_xmit_frame(struct sk_buff *skb, struct net_device *dev)
len = skb_headlen(skb); len = skb_headlen(skb);
mapping = pci_map_single(hw->pdev, skb->data, len, PCI_DMA_TODEVICE); mapping = pci_map_single(hw->pdev, skb->data, len, PCI_DMA_TODEVICE);
addr64 = upper_32_bits(mapping);
/* Send high bits if changed or crosses boundary */ /* Send high bits if needed */
if (addr64 != sky2->tx_addr64 || if (sizeof(dma_addr_t) > sizeof(u32)) {
upper_32_bits(mapping + len) != sky2->tx_addr64) {
le = get_tx_le(sky2); le = get_tx_le(sky2);
le->addr = cpu_to_le32(addr64); le->addr = cpu_to_le32(upper_32_bits(mapping));
le->opcode = OP_ADDR64 | HW_OWNER; le->opcode = OP_ADDR64 | HW_OWNER;
sky2->tx_addr64 = upper_32_bits(mapping + len);
} }
/* Check for TCP Segmentation Offload */ /* Check for TCP Segmentation Offload */
@ -1582,13 +1575,12 @@ static int sky2_xmit_frame(struct sk_buff *skb, struct net_device *dev)
mapping = pci_map_page(hw->pdev, frag->page, frag->page_offset, mapping = pci_map_page(hw->pdev, frag->page, frag->page_offset,
frag->size, PCI_DMA_TODEVICE); frag->size, PCI_DMA_TODEVICE);
addr64 = upper_32_bits(mapping);
if (addr64 != sky2->tx_addr64) { if (sizeof(dma_addr_t) > sizeof(u32)) {
le = get_tx_le(sky2); le = get_tx_le(sky2);
le->addr = cpu_to_le32(addr64); le->addr = cpu_to_le32(upper_32_bits(mapping));
le->ctrl = 0; le->ctrl = 0;
le->opcode = OP_ADDR64 | HW_OWNER; le->opcode = OP_ADDR64 | HW_OWNER;
sky2->tx_addr64 = addr64;
} }
le = get_tx_le(sky2); le = get_tx_le(sky2);

View File

@ -1991,14 +1991,14 @@ struct sky2_port {
u16 tx_cons; /* next le to check */ u16 tx_cons; /* next le to check */
u16 tx_prod; /* next le to use */ u16 tx_prod; /* next le to use */
u16 tx_next; /* debug only */ u16 tx_next; /* debug only */
u32 tx_addr64;
u16 tx_pending; u16 tx_pending;
u16 tx_last_mss; u16 tx_last_mss;
u32 tx_tcpsum; u32 tx_tcpsum;
struct rx_ring_info *rx_ring ____cacheline_aligned_in_smp; struct rx_ring_info *rx_ring ____cacheline_aligned_in_smp;
struct sky2_rx_le *rx_le; struct sky2_rx_le *rx_le;
u32 rx_addr64;
u16 rx_next; /* next re to check */ u16 rx_next; /* next re to check */
u16 rx_put; /* next le index to use */ u16 rx_put; /* next le index to use */
u16 rx_pending; u16 rx_pending;