aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/atl1
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/atl1')
-rw-r--r--drivers/net/atl1/atl1_main.c86
1 files changed, 50 insertions, 36 deletions
diff --git a/drivers/net/atl1/atl1_main.c b/drivers/net/atl1/atl1_main.c
index b40f1c7901fe..67ddf8dcf764 100644
--- a/drivers/net/atl1/atl1_main.c
+++ b/drivers/net/atl1/atl1_main.c
@@ -220,8 +220,6 @@ s32 atl1_setup_ring_resources(struct atl1_adapter *adapter)
220 tpd_ring->dma += offset; 220 tpd_ring->dma += offset;
221 tpd_ring->desc = (u8 *) ring_header->desc + offset; 221 tpd_ring->desc = (u8 *) ring_header->desc + offset;
222 tpd_ring->size = sizeof(struct tx_packet_desc) * tpd_ring->count; 222 tpd_ring->size = sizeof(struct tx_packet_desc) * tpd_ring->count;
223 atomic_set(&tpd_ring->next_to_use, 0);
224 atomic_set(&tpd_ring->next_to_clean, 0);
225 223
226 /* init RFD ring */ 224 /* init RFD ring */
227 rfd_ring->dma = tpd_ring->dma + tpd_ring->size; 225 rfd_ring->dma = tpd_ring->dma + tpd_ring->size;
@@ -229,8 +227,7 @@ s32 atl1_setup_ring_resources(struct atl1_adapter *adapter)
229 rfd_ring->dma += offset; 227 rfd_ring->dma += offset;
230 rfd_ring->desc = (u8 *) tpd_ring->desc + (tpd_ring->size + offset); 228 rfd_ring->desc = (u8 *) tpd_ring->desc + (tpd_ring->size + offset);
231 rfd_ring->size = sizeof(struct rx_free_desc) * rfd_ring->count; 229 rfd_ring->size = sizeof(struct rx_free_desc) * rfd_ring->count;
232 rfd_ring->next_to_clean = 0; 230
233 atomic_set(&rfd_ring->next_to_use, 0);
234 231
235 /* init RRD ring */ 232 /* init RRD ring */
236 rrd_ring->dma = rfd_ring->dma + rfd_ring->size; 233 rrd_ring->dma = rfd_ring->dma + rfd_ring->size;
@@ -238,8 +235,7 @@ s32 atl1_setup_ring_resources(struct atl1_adapter *adapter)
238 rrd_ring->dma += offset; 235 rrd_ring->dma += offset;
239 rrd_ring->desc = (u8 *) rfd_ring->desc + (rfd_ring->size + offset); 236 rrd_ring->desc = (u8 *) rfd_ring->desc + (rfd_ring->size + offset);
240 rrd_ring->size = sizeof(struct rx_return_desc) * rrd_ring->count; 237 rrd_ring->size = sizeof(struct rx_return_desc) * rrd_ring->count;
241 rrd_ring->next_to_use = 0; 238
242 atomic_set(&rrd_ring->next_to_clean, 0);
243 239
244 /* init CMB */ 240 /* init CMB */
245 adapter->cmb.dma = rrd_ring->dma + rrd_ring->size; 241 adapter->cmb.dma = rrd_ring->dma + rrd_ring->size;
@@ -263,6 +259,22 @@ err_nomem:
263 return -ENOMEM; 259 return -ENOMEM;
264} 260}
265 261
262void atl1_init_ring_ptrs(struct atl1_adapter *adapter)
263{
264 struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring;
265 struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring;
266 struct atl1_rrd_ring *rrd_ring = &adapter->rrd_ring;
267
268 atomic_set(&tpd_ring->next_to_use, 0);
269 atomic_set(&tpd_ring->next_to_clean, 0);
270
271 rfd_ring->next_to_clean = 0;
272 atomic_set(&rfd_ring->next_to_use, 0);
273
274 rrd_ring->next_to_use = 0;
275 atomic_set(&rrd_ring->next_to_clean, 0);
276}
277
266/* 278/*
267 * atl1_irq_enable - Enable default interrupt generation settings 279 * atl1_irq_enable - Enable default interrupt generation settings
268 * @adapter: board private structure 280 * @adapter: board private structure
@@ -472,6 +484,31 @@ next:
472 return num_alloc; 484 return num_alloc;
473} 485}
474 486
487static void atl1_clean_alloc_flag(struct atl1_adapter *adapter,
488 struct rx_return_desc *rrd, u16 offset)
489{
490 struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring;
491
492 while (rfd_ring->next_to_clean != (rrd->buf_indx + offset)) {
493 rfd_ring->buffer_info[rfd_ring->next_to_clean].alloced = 0;
494 if (++rfd_ring->next_to_clean == rfd_ring->count) {
495 rfd_ring->next_to_clean = 0;
496 }
497 }
498}
499
500static void atl1_update_rfd_index(struct atl1_adapter *adapter,
501 struct rx_return_desc *rrd)
502{
503 u16 num_buf;
504
505 num_buf = (rrd->xsz.xsum_sz.pkt_size + adapter->rx_buffer_len - 1) /
506 adapter->rx_buffer_len;
507 if (rrd->num_buf == num_buf)
508 /* clean alloc flag for bad rrd */
509 atl1_clean_alloc_flag(adapter, rrd, num_buf);
510}
511
475static void atl1_intr_rx(struct atl1_adapter *adapter) 512static void atl1_intr_rx(struct atl1_adapter *adapter)
476{ 513{
477 int i, count; 514 int i, count;
@@ -509,26 +546,8 @@ chk_rrd:
509 dev_printk(KERN_DEBUG, &adapter->pdev->dev, 546 dev_printk(KERN_DEBUG, &adapter->pdev->dev,
510 "bad RRD\n"); 547 "bad RRD\n");
511 /* see if update RFD index */ 548 /* see if update RFD index */
512 if (rrd->num_buf > 1) { 549 if (rrd->num_buf > 1)
513 u16 num_buf; 550 atl1_update_rfd_index(adapter, rrd);
514 num_buf =
515 (rrd->xsz.xsum_sz.pkt_size +
516 adapter->rx_buffer_len -
517 1) / adapter->rx_buffer_len;
518 if (rrd->num_buf == num_buf) {
519 /* clean alloc flag for bad rrd */
520 while (rfd_ring->next_to_clean !=
521 (rrd->buf_indx + num_buf)) {
522 rfd_ring->buffer_info[rfd_ring->
523 next_to_clean].alloced = 0;
524 if (++rfd_ring->next_to_clean ==
525 rfd_ring->count) {
526 rfd_ring->
527 next_to_clean = 0;
528 }
529 }
530 }
531 }
532 551
533 /* update rrd */ 552 /* update rrd */
534 rrd->xsz.valid = 0; 553 rrd->xsz.valid = 0;
@@ -542,12 +561,7 @@ chk_rrd:
542 } 561 }
543rrd_ok: 562rrd_ok:
544 /* clean alloc flag for bad rrd */ 563 /* clean alloc flag for bad rrd */
545 while (rfd_ring->next_to_clean != rrd->buf_indx) { 564 atl1_clean_alloc_flag(adapter, rrd, 0);
546 rfd_ring->buffer_info[rfd_ring->next_to_clean].alloced =
547 0;
548 if (++rfd_ring->next_to_clean == rfd_ring->count)
549 rfd_ring->next_to_clean = 0;
550 }
551 565
552 buffer_info = &rfd_ring->buffer_info[rrd->buf_indx]; 566 buffer_info = &rfd_ring->buffer_info[rrd->buf_indx];
553 if (++rfd_ring->next_to_clean == rfd_ring->count) 567 if (++rfd_ring->next_to_clean == rfd_ring->count)
@@ -1058,7 +1072,8 @@ static u32 atl1_configure(struct atl1_adapter *adapter)
1058 value <<= 16; 1072 value <<= 16;
1059 value += adapter->rfd_ring.count; 1073 value += adapter->rfd_ring.count;
1060 iowrite32(value, hw->hw_addr + REG_DESC_RFD_RRD_RING_SIZE); 1074 iowrite32(value, hw->hw_addr + REG_DESC_RFD_RRD_RING_SIZE);
1061 iowrite32(adapter->tpd_ring.count, hw->hw_addr + REG_DESC_TPD_RING_SIZE); 1075 iowrite32(adapter->tpd_ring.count, hw->hw_addr +
1076 REG_DESC_TPD_RING_SIZE);
1062 1077
1063 /* Load Ptr */ 1078 /* Load Ptr */
1064 iowrite32(1, hw->hw_addr + REG_LOAD_PTR); 1079 iowrite32(1, hw->hw_addr + REG_LOAD_PTR);
@@ -1258,9 +1273,7 @@ static int atl1_tso(struct atl1_adapter *adapter, struct sk_buff *skb,
1258 iph->tot_len = 0; 1273 iph->tot_len = 0;
1259 iph->check = 0; 1274 iph->check = 0;
1260 tcp_hdr(skb)->check = ~csum_tcpudp_magic(iph->saddr, 1275 tcp_hdr(skb)->check = ~csum_tcpudp_magic(iph->saddr,
1261 iph->daddr, 0, 1276 iph->daddr, 0, IPPROTO_TCP, 0);
1262 IPPROTO_TCP,
1263 0);
1264 ipofst = skb_network_offset(skb); 1277 ipofst = skb_network_offset(skb);
1265 if (ipofst != ENET_HEADER_SIZE) /* 802.3 frame */ 1278 if (ipofst != ENET_HEADER_SIZE) /* 802.3 frame */
1266 tso->tsopl |= 1 << TSO_PARAM_ETHTYPE_SHIFT; 1279 tso->tsopl |= 1 << TSO_PARAM_ETHTYPE_SHIFT;
@@ -1721,6 +1734,7 @@ s32 atl1_up(struct atl1_adapter *adapter)
1721 1734
1722 /* hardware has been reset, we need to reload some things */ 1735 /* hardware has been reset, we need to reload some things */
1723 atl1_set_multi(netdev); 1736 atl1_set_multi(netdev);
1737 atl1_init_ring_ptrs(adapter);
1724 atl1_restore_vlan(adapter); 1738 atl1_restore_vlan(adapter);
1725 err = atl1_alloc_rx_buffers(adapter); 1739 err = atl1_alloc_rx_buffers(adapter);
1726 if (unlikely(!err)) /* no RX BUFFER allocated */ 1740 if (unlikely(!err)) /* no RX BUFFER allocated */