Commit 8042824a authored by Gatis Peisenieks's avatar Gatis Peisenieks Committed by Jakub Kicinski

atl1c: prepare for multiple rx queues

Move napi and other per queue members into per rx queue struct.
Allocate max rx queues that any hw supported by the driver might have.
Patch that actually enables multiple rx queues will follow.
Signed-off-by: default avatarGatis Peisenieks <gatis@mikrotik.com>
Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent 20a1b6bd
...@@ -500,21 +500,23 @@ struct atl1c_rfd_ring { ...@@ -500,21 +500,23 @@ struct atl1c_rfd_ring {
/* receive return descriptor (rrd) ring */ /* receive return descriptor (rrd) ring */
struct atl1c_rrd_ring { struct atl1c_rrd_ring {
struct atl1c_adapter *adapter;
void *desc; /* descriptor ring virtual address */ void *desc; /* descriptor ring virtual address */
dma_addr_t dma; /* descriptor ring physical address */ dma_addr_t dma; /* descriptor ring physical address */
u16 num;
u16 size; /* descriptor ring length in bytes */ u16 size; /* descriptor ring length in bytes */
u16 count; /* number of descriptors in the ring */ u16 count; /* number of descriptors in the ring */
u16 next_to_use; u16 next_to_use;
u16 next_to_clean; u16 next_to_clean;
struct napi_struct napi;
struct page *rx_page;
unsigned int rx_page_offset;
}; };
/* board specific private data structure */ /* board specific private data structure */
struct atl1c_adapter { struct atl1c_adapter {
struct net_device *netdev; struct net_device *netdev;
struct pci_dev *pdev; struct pci_dev *pdev;
struct napi_struct napi;
struct page *rx_page;
unsigned int rx_page_offset;
unsigned int rx_frag_size; unsigned int rx_frag_size;
struct atl1c_hw hw; struct atl1c_hw hw;
struct atl1c_hw_stats hw_stats; struct atl1c_hw_stats hw_stats;
...@@ -545,8 +547,8 @@ struct atl1c_adapter { ...@@ -545,8 +547,8 @@ struct atl1c_adapter {
/* All Descriptor memory */ /* All Descriptor memory */
struct atl1c_ring_header ring_header; struct atl1c_ring_header ring_header;
struct atl1c_tpd_ring tpd_ring[AT_MAX_TRANSMIT_QUEUE]; struct atl1c_tpd_ring tpd_ring[AT_MAX_TRANSMIT_QUEUE];
struct atl1c_rfd_ring rfd_ring; struct atl1c_rfd_ring rfd_ring[AT_MAX_RECEIVE_QUEUE];
struct atl1c_rrd_ring rrd_ring; struct atl1c_rrd_ring rrd_ring[AT_MAX_RECEIVE_QUEUE];
u32 bd_number; /* board number;*/ u32 bd_number; /* board number;*/
}; };
......
...@@ -40,8 +40,6 @@ static int atl1c_stop_mac(struct atl1c_hw *hw); ...@@ -40,8 +40,6 @@ static int atl1c_stop_mac(struct atl1c_hw *hw);
static void atl1c_disable_l0s_l1(struct atl1c_hw *hw); static void atl1c_disable_l0s_l1(struct atl1c_hw *hw);
static void atl1c_set_aspm(struct atl1c_hw *hw, u16 link_speed); static void atl1c_set_aspm(struct atl1c_hw *hw, u16 link_speed);
static void atl1c_start_mac(struct atl1c_adapter *adapter); static void atl1c_start_mac(struct atl1c_adapter *adapter);
static void atl1c_clean_rx_irq(struct atl1c_adapter *adapter,
int *work_done, int work_to_do);
static int atl1c_up(struct atl1c_adapter *adapter); static int atl1c_up(struct atl1c_adapter *adapter);
static void atl1c_down(struct atl1c_adapter *adapter); static void atl1c_down(struct atl1c_adapter *adapter);
static int atl1c_reset_mac(struct atl1c_hw *hw); static int atl1c_reset_mac(struct atl1c_hw *hw);
...@@ -770,7 +768,7 @@ static int atl1c_sw_init(struct atl1c_adapter *adapter) ...@@ -770,7 +768,7 @@ static int atl1c_sw_init(struct atl1c_adapter *adapter)
adapter->link_speed = SPEED_0; adapter->link_speed = SPEED_0;
adapter->link_duplex = FULL_DUPLEX; adapter->link_duplex = FULL_DUPLEX;
adapter->tpd_ring[0].count = 1024; adapter->tpd_ring[0].count = 1024;
adapter->rfd_ring.count = 512; adapter->rfd_ring[0].count = 512;
hw->vendor_id = pdev->vendor; hw->vendor_id = pdev->vendor;
hw->device_id = pdev->device; hw->device_id = pdev->device;
...@@ -878,8 +876,8 @@ static void atl1c_clean_tx_ring(struct atl1c_adapter *adapter, ...@@ -878,8 +876,8 @@ static void atl1c_clean_tx_ring(struct atl1c_adapter *adapter,
*/ */
static void atl1c_clean_rx_ring(struct atl1c_adapter *adapter) static void atl1c_clean_rx_ring(struct atl1c_adapter *adapter)
{ {
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring; struct atl1c_rfd_ring *rfd_ring = adapter->rfd_ring;
struct atl1c_rrd_ring *rrd_ring = &adapter->rrd_ring; struct atl1c_rrd_ring *rrd_ring = adapter->rrd_ring;
struct atl1c_buffer *buffer_info; struct atl1c_buffer *buffer_info;
struct pci_dev *pdev = adapter->pdev; struct pci_dev *pdev = adapter->pdev;
int j; int j;
...@@ -902,8 +900,8 @@ static void atl1c_clean_rx_ring(struct atl1c_adapter *adapter) ...@@ -902,8 +900,8 @@ static void atl1c_clean_rx_ring(struct atl1c_adapter *adapter)
static void atl1c_init_ring_ptrs(struct atl1c_adapter *adapter) static void atl1c_init_ring_ptrs(struct atl1c_adapter *adapter)
{ {
struct atl1c_tpd_ring *tpd_ring = adapter->tpd_ring; struct atl1c_tpd_ring *tpd_ring = adapter->tpd_ring;
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring; struct atl1c_rfd_ring *rfd_ring = adapter->rfd_ring;
struct atl1c_rrd_ring *rrd_ring = &adapter->rrd_ring; struct atl1c_rrd_ring *rrd_ring = adapter->rrd_ring;
struct atl1c_buffer *buffer_info; struct atl1c_buffer *buffer_info;
int i, j; int i, j;
...@@ -945,9 +943,9 @@ static void atl1c_free_ring_resources(struct atl1c_adapter *adapter) ...@@ -945,9 +943,9 @@ static void atl1c_free_ring_resources(struct atl1c_adapter *adapter)
kfree(adapter->tpd_ring[0].buffer_info); kfree(adapter->tpd_ring[0].buffer_info);
adapter->tpd_ring[0].buffer_info = NULL; adapter->tpd_ring[0].buffer_info = NULL;
} }
if (adapter->rx_page) { if (adapter->rrd_ring[0].rx_page) {
put_page(adapter->rx_page); put_page(adapter->rrd_ring[0].rx_page);
adapter->rx_page = NULL; adapter->rrd_ring[0].rx_page = NULL;
} }
} }
...@@ -961,8 +959,8 @@ static int atl1c_setup_ring_resources(struct atl1c_adapter *adapter) ...@@ -961,8 +959,8 @@ static int atl1c_setup_ring_resources(struct atl1c_adapter *adapter)
{ {
struct pci_dev *pdev = adapter->pdev; struct pci_dev *pdev = adapter->pdev;
struct atl1c_tpd_ring *tpd_ring = adapter->tpd_ring; struct atl1c_tpd_ring *tpd_ring = adapter->tpd_ring;
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring; struct atl1c_rfd_ring *rfd_ring = adapter->rfd_ring;
struct atl1c_rrd_ring *rrd_ring = &adapter->rrd_ring; struct atl1c_rrd_ring *rrd_ring = adapter->rrd_ring;
struct atl1c_ring_header *ring_header = &adapter->ring_header; struct atl1c_ring_header *ring_header = &adapter->ring_header;
int size; int size;
int i; int i;
...@@ -1030,6 +1028,8 @@ static int atl1c_setup_ring_resources(struct atl1c_adapter *adapter) ...@@ -1030,6 +1028,8 @@ static int atl1c_setup_ring_resources(struct atl1c_adapter *adapter)
offset += roundup(rfd_ring->size, 8); offset += roundup(rfd_ring->size, 8);
/* init RRD ring */ /* init RRD ring */
rrd_ring->adapter = adapter;
rrd_ring->num = 0;
rrd_ring->dma = ring_header->dma + offset; rrd_ring->dma = ring_header->dma + offset;
rrd_ring->desc = (u8 *) ring_header->desc + offset; rrd_ring->desc = (u8 *) ring_header->desc + offset;
rrd_ring->size = sizeof(struct atl1c_recv_ret_status) * rrd_ring->size = sizeof(struct atl1c_recv_ret_status) *
...@@ -1046,10 +1046,9 @@ static int atl1c_setup_ring_resources(struct atl1c_adapter *adapter) ...@@ -1046,10 +1046,9 @@ static int atl1c_setup_ring_resources(struct atl1c_adapter *adapter)
static void atl1c_configure_des_ring(struct atl1c_adapter *adapter) static void atl1c_configure_des_ring(struct atl1c_adapter *adapter)
{ {
struct atl1c_hw *hw = &adapter->hw; struct atl1c_hw *hw = &adapter->hw;
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring; struct atl1c_rfd_ring *rfd_ring = adapter->rfd_ring;
struct atl1c_rrd_ring *rrd_ring = &adapter->rrd_ring; struct atl1c_rrd_ring *rrd_ring = adapter->rrd_ring;
struct atl1c_tpd_ring *tpd_ring = (struct atl1c_tpd_ring *) struct atl1c_tpd_ring *tpd_ring = adapter->tpd_ring;
adapter->tpd_ring;
/* TPD */ /* TPD */
AT_WRITE_REG(hw, REG_TX_BASE_ADDR_HI, AT_WRITE_REG(hw, REG_TX_BASE_ADDR_HI,
...@@ -1608,12 +1607,12 @@ static irqreturn_t atl1c_intr(int irq, void *data) ...@@ -1608,12 +1607,12 @@ static irqreturn_t atl1c_intr(int irq, void *data)
/* Ack ISR */ /* Ack ISR */
AT_WRITE_REG(hw, REG_ISR, status | ISR_DIS_INT); AT_WRITE_REG(hw, REG_ISR, status | ISR_DIS_INT);
if (status & ISR_RX_PKT) { if (status & ISR_RX_PKT) {
if (likely(napi_schedule_prep(&adapter->napi))) { if (napi_schedule_prep(&adapter->rrd_ring[0].napi)) {
spin_lock(&hw->intr_mask_lock); spin_lock(&hw->intr_mask_lock);
hw->intr_mask &= ~ISR_RX_PKT; hw->intr_mask &= ~ISR_RX_PKT;
AT_WRITE_REG(hw, REG_IMR, hw->intr_mask); AT_WRITE_REG(hw, REG_IMR, hw->intr_mask);
spin_unlock(&hw->intr_mask_lock); spin_unlock(&hw->intr_mask_lock);
__napi_schedule(&adapter->napi); __napi_schedule(&adapter->rrd_ring[0].napi);
} }
} }
if (status & ISR_TX_PKT) { if (status & ISR_TX_PKT) {
...@@ -1677,33 +1676,35 @@ static inline void atl1c_rx_checksum(struct atl1c_adapter *adapter, ...@@ -1677,33 +1676,35 @@ static inline void atl1c_rx_checksum(struct atl1c_adapter *adapter,
static struct sk_buff *atl1c_alloc_skb(struct atl1c_adapter *adapter, static struct sk_buff *atl1c_alloc_skb(struct atl1c_adapter *adapter,
bool napi_mode) bool napi_mode)
{ {
struct atl1c_rrd_ring *rrd_ring = &adapter->rrd_ring[0];
struct sk_buff *skb; struct sk_buff *skb;
struct page *page; struct page *page;
if (adapter->rx_frag_size > PAGE_SIZE) { if (adapter->rx_frag_size > PAGE_SIZE) {
if (likely(napi_mode)) if (likely(napi_mode))
return napi_alloc_skb(&adapter->napi, return napi_alloc_skb(&rrd_ring->napi,
adapter->rx_buffer_len); adapter->rx_buffer_len);
else else
return netdev_alloc_skb_ip_align(adapter->netdev, return netdev_alloc_skb_ip_align(adapter->netdev,
adapter->rx_buffer_len); adapter->rx_buffer_len);
} }
page = adapter->rx_page; page = rrd_ring->rx_page;
if (!page) { if (!page) {
adapter->rx_page = page = alloc_page(GFP_ATOMIC); page = alloc_page(GFP_ATOMIC);
if (unlikely(!page)) if (unlikely(!page))
return NULL; return NULL;
adapter->rx_page_offset = 0; rrd_ring->rx_page = page;
rrd_ring->rx_page_offset = 0;
} }
skb = build_skb(page_address(page) + adapter->rx_page_offset, skb = build_skb(page_address(page) + rrd_ring->rx_page_offset,
adapter->rx_frag_size); adapter->rx_frag_size);
if (likely(skb)) { if (likely(skb)) {
skb_reserve(skb, NET_SKB_PAD + NET_IP_ALIGN); skb_reserve(skb, NET_SKB_PAD + NET_IP_ALIGN);
adapter->rx_page_offset += adapter->rx_frag_size; rrd_ring->rx_page_offset += adapter->rx_frag_size;
if (adapter->rx_page_offset >= PAGE_SIZE) if (rrd_ring->rx_page_offset >= PAGE_SIZE)
adapter->rx_page = NULL; rrd_ring->rx_page = NULL;
else else
get_page(page); get_page(page);
} }
...@@ -1712,7 +1713,7 @@ static struct sk_buff *atl1c_alloc_skb(struct atl1c_adapter *adapter, ...@@ -1712,7 +1713,7 @@ static struct sk_buff *atl1c_alloc_skb(struct atl1c_adapter *adapter,
static int atl1c_alloc_rx_buffer(struct atl1c_adapter *adapter, bool napi_mode) static int atl1c_alloc_rx_buffer(struct atl1c_adapter *adapter, bool napi_mode)
{ {
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring; struct atl1c_rfd_ring *rfd_ring = adapter->rfd_ring;
struct pci_dev *pdev = adapter->pdev; struct pci_dev *pdev = adapter->pdev;
struct atl1c_buffer *buffer_info, *next_info; struct atl1c_buffer *buffer_info, *next_info;
struct sk_buff *skb; struct sk_buff *skb;
...@@ -1812,22 +1813,34 @@ static void atl1c_clean_rfd(struct atl1c_rfd_ring *rfd_ring, ...@@ -1812,22 +1813,34 @@ static void atl1c_clean_rfd(struct atl1c_rfd_ring *rfd_ring,
rfd_ring->next_to_clean = rfd_index; rfd_ring->next_to_clean = rfd_index;
} }
static void atl1c_clean_rx_irq(struct atl1c_adapter *adapter, /**
int *work_done, int work_to_do) * atl1c_clean_rx - NAPI Rx polling callback
* @napi: napi info
* @budget: limit of packets to clean
*/
static int atl1c_clean_rx(struct napi_struct *napi, int budget)
{ {
struct atl1c_rrd_ring *rrd_ring =
container_of(napi, struct atl1c_rrd_ring, napi);
struct atl1c_adapter *adapter = rrd_ring->adapter;
u16 rfd_num, rfd_index; u16 rfd_num, rfd_index;
u16 count = 0; u16 count = 0;
u16 length; u16 length;
struct pci_dev *pdev = adapter->pdev; struct pci_dev *pdev = adapter->pdev;
struct net_device *netdev = adapter->netdev; struct net_device *netdev = adapter->netdev;
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring; struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring[rrd_ring->num];
struct atl1c_rrd_ring *rrd_ring = &adapter->rrd_ring;
struct sk_buff *skb; struct sk_buff *skb;
struct atl1c_recv_ret_status *rrs; struct atl1c_recv_ret_status *rrs;
struct atl1c_buffer *buffer_info; struct atl1c_buffer *buffer_info;
int work_done = 0;
unsigned long flags;
/* Keep link state information with original netdev */
if (!netif_carrier_ok(adapter->netdev))
goto quit_polling;
while (1) { while (1) {
if (*work_done >= work_to_do) if (work_done >= budget)
break; break;
rrs = ATL1C_RRD_DESC(rrd_ring, rrd_ring->next_to_clean); rrs = ATL1C_RRD_DESC(rrd_ring, rrd_ring->next_to_clean);
if (likely(RRS_RXD_IS_VALID(rrs->word3))) { if (likely(RRS_RXD_IS_VALID(rrs->word3))) {
...@@ -1881,32 +1894,13 @@ static void atl1c_clean_rx_irq(struct atl1c_adapter *adapter, ...@@ -1881,32 +1894,13 @@ static void atl1c_clean_rx_irq(struct atl1c_adapter *adapter,
vlan = le16_to_cpu(vlan); vlan = le16_to_cpu(vlan);
__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q), vlan); __vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q), vlan);
} }
napi_gro_receive(&adapter->napi, skb); napi_gro_receive(napi, skb);
(*work_done)++; work_done++;
count++; count++;
} }
if (count) if (count)
atl1c_alloc_rx_buffer(adapter, true); atl1c_alloc_rx_buffer(adapter, true);
}
/**
* atl1c_clean - NAPI Rx polling callback
* @napi: napi info
* @budget: limit of packets to clean
*/
static int atl1c_clean(struct napi_struct *napi, int budget)
{
struct atl1c_adapter *adapter =
container_of(napi, struct atl1c_adapter, napi);
int work_done = 0;
unsigned long flags;
/* Keep link state information with original netdev */
if (!netif_carrier_ok(adapter->netdev))
goto quit_polling;
/* just enable one RXQ */
atl1c_clean_rx_irq(adapter, &work_done, budget);
if (work_done < budget) { if (work_done < budget) {
quit_polling: quit_polling:
...@@ -2355,7 +2349,7 @@ static int atl1c_up(struct atl1c_adapter *adapter) ...@@ -2355,7 +2349,7 @@ static int atl1c_up(struct atl1c_adapter *adapter)
atl1c_check_link_status(adapter); atl1c_check_link_status(adapter);
clear_bit(__AT_DOWN, &adapter->flags); clear_bit(__AT_DOWN, &adapter->flags);
napi_enable(&adapter->napi); napi_enable(&adapter->rrd_ring[0].napi);
napi_enable(&adapter->tpd_ring[0].napi); napi_enable(&adapter->tpd_ring[0].napi);
atl1c_irq_enable(adapter); atl1c_irq_enable(adapter);
netif_start_queue(netdev); netif_start_queue(netdev);
...@@ -2376,7 +2370,7 @@ static void atl1c_down(struct atl1c_adapter *adapter) ...@@ -2376,7 +2370,7 @@ static void atl1c_down(struct atl1c_adapter *adapter)
* reschedule our watchdog timer */ * reschedule our watchdog timer */
set_bit(__AT_DOWN, &adapter->flags); set_bit(__AT_DOWN, &adapter->flags);
netif_carrier_off(netdev); netif_carrier_off(netdev);
napi_disable(&adapter->napi); napi_disable(&adapter->rrd_ring[0].napi);
napi_disable(&adapter->tpd_ring[0].napi); napi_disable(&adapter->tpd_ring[0].napi);
atl1c_irq_disable(adapter); atl1c_irq_disable(adapter);
atl1c_free_irq(adapter); atl1c_free_irq(adapter);
...@@ -2633,7 +2627,7 @@ static int atl1c_probe(struct pci_dev *pdev, const struct pci_device_id *ent) ...@@ -2633,7 +2627,7 @@ static int atl1c_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
adapter->mii.phy_id_mask = 0x1f; adapter->mii.phy_id_mask = 0x1f;
adapter->mii.reg_num_mask = MDIO_CTRL_REG_MASK; adapter->mii.reg_num_mask = MDIO_CTRL_REG_MASK;
dev_set_threaded(netdev, true); dev_set_threaded(netdev, true);
netif_napi_add(netdev, &adapter->napi, atl1c_clean, 64); netif_napi_add(netdev, &adapter->rrd_ring[0].napi, atl1c_clean_rx, 64);
netif_napi_add(netdev, &adapter->tpd_ring[0].napi, atl1c_clean_tx, 64); netif_napi_add(netdev, &adapter->tpd_ring[0].napi, atl1c_clean_tx, 64);
timer_setup(&adapter->phy_config_timer, atl1c_phy_config, 0); timer_setup(&adapter->phy_config_timer, atl1c_phy_config, 0);
/* setup the private structure */ /* setup the private structure */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment