Commit 3f703186 authored by Mengyuan Lou's avatar Mengyuan Lou Committed by David S. Miller

net: libwx: Add irq flow functions

Add irq flow functions for ngbe and txgbe.
Alloc pcie msix irqs for drivers, otherwise fall back to msi/legacy.
Signed-off-by: default avatarMengyuan Lou <mengyuanlou@net-swift.com>
Signed-off-by: default avatarJiawen Wu <jiawenwu@trustnetic.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 542bcea4
...@@ -4,4 +4,4 @@ ...@@ -4,4 +4,4 @@
obj-$(CONFIG_LIBWX) += libwx.o obj-$(CONFIG_LIBWX) += libwx.o
libwx-objs := wx_hw.o libwx-objs := wx_hw.o wx_lib.o
...@@ -14,7 +14,7 @@ static void wx_intr_disable(struct wx *wx, u64 qmask) ...@@ -14,7 +14,7 @@ static void wx_intr_disable(struct wx *wx, u64 qmask)
{ {
u32 mask; u32 mask;
mask = (qmask & 0xFFFFFFFF); mask = (qmask & U32_MAX);
if (mask) if (mask)
wr32(wx, WX_PX_IMS(0), mask); wr32(wx, WX_PX_IMS(0), mask);
...@@ -25,6 +25,45 @@ static void wx_intr_disable(struct wx *wx, u64 qmask) ...@@ -25,6 +25,45 @@ static void wx_intr_disable(struct wx *wx, u64 qmask)
} }
} }
void wx_intr_enable(struct wx *wx, u64 qmask)
{
u32 mask;
mask = (qmask & U32_MAX);
if (mask)
wr32(wx, WX_PX_IMC(0), mask);
if (wx->mac.type == wx_mac_sp) {
mask = (qmask >> 32);
if (mask)
wr32(wx, WX_PX_IMC(1), mask);
}
}
EXPORT_SYMBOL(wx_intr_enable);
/**
* wx_irq_disable - Mask off interrupt generation on the NIC
* @wx: board private structure
**/
void wx_irq_disable(struct wx *wx)
{
struct pci_dev *pdev = wx->pdev;
wr32(wx, WX_PX_MISC_IEN, 0);
wx_intr_disable(wx, WX_INTR_ALL);
if (pdev->msix_enabled) {
int vector;
for (vector = 0; vector < wx->num_q_vectors; vector++)
synchronize_irq(wx->msix_entries[vector].vector);
synchronize_irq(wx->msix_entries[vector].vector);
} else {
synchronize_irq(pdev->irq);
}
}
EXPORT_SYMBOL(wx_irq_disable);
/* cmd_addr is used for some special command: /* cmd_addr is used for some special command:
* 1. to be sector address, when implemented erase sector command * 1. to be sector address, when implemented erase sector command
* 2. to be flash address when implemented read, write flash address * 2. to be flash address when implemented read, write flash address
...@@ -844,6 +883,20 @@ void wx_disable_rx(struct wx *wx) ...@@ -844,6 +883,20 @@ void wx_disable_rx(struct wx *wx)
} }
EXPORT_SYMBOL(wx_disable_rx); EXPORT_SYMBOL(wx_disable_rx);
static void wx_configure_isb(struct wx *wx)
{
/* set ISB Address */
wr32(wx, WX_PX_ISB_ADDR_L, wx->isb_dma & DMA_BIT_MASK(32));
if (IS_ENABLED(CONFIG_ARCH_DMA_ADDR_T_64BIT))
wr32(wx, WX_PX_ISB_ADDR_H, upper_32_bits(wx->isb_dma));
}
void wx_configure(struct wx *wx)
{
wx_configure_isb(wx);
}
EXPORT_SYMBOL(wx_configure);
/** /**
* wx_disable_pcie_master - Disable PCI-express master access * wx_disable_pcie_master - Disable PCI-express master access
* @wx: pointer to hardware structure * @wx: pointer to hardware structure
......
...@@ -4,6 +4,8 @@ ...@@ -4,6 +4,8 @@
#ifndef _WX_HW_H_ #ifndef _WX_HW_H_
#define _WX_HW_H_ #define _WX_HW_H_
void wx_intr_enable(struct wx *wx, u64 qmask);
void wx_irq_disable(struct wx *wx);
int wx_check_flash_load(struct wx *wx, u32 check_bit); int wx_check_flash_load(struct wx *wx, u32 check_bit);
void wx_control_hw(struct wx *wx, bool drv); void wx_control_hw(struct wx *wx, bool drv);
int wx_mng_present(struct wx *wx); int wx_mng_present(struct wx *wx);
...@@ -20,6 +22,7 @@ void wx_mac_set_default_filter(struct wx *wx, u8 *addr); ...@@ -20,6 +22,7 @@ void wx_mac_set_default_filter(struct wx *wx, u8 *addr);
void wx_flush_sw_mac_table(struct wx *wx); void wx_flush_sw_mac_table(struct wx *wx);
int wx_set_mac(struct net_device *netdev, void *p); int wx_set_mac(struct net_device *netdev, void *p);
void wx_disable_rx(struct wx *wx); void wx_disable_rx(struct wx *wx);
void wx_configure(struct wx *wx);
int wx_disable_pcie_master(struct wx *wx); int wx_disable_pcie_master(struct wx *wx);
int wx_stop_adapter(struct wx *wx); int wx_stop_adapter(struct wx *wx);
void wx_reset_misc(struct wx *wx); void wx_reset_misc(struct wx *wx);
......
This diff is collapsed.
/* SPDX-License-Identifier: GPL-2.0 */
/*
* WangXun Gigabit PCI Express Linux driver
* Copyright (c) 2019 - 2022 Beijing WangXun Technology Co., Ltd.
*/
#ifndef _WX_LIB_H_
#define _WX_LIB_H_
void wx_reset_interrupt_capability(struct wx *wx);
void wx_clear_interrupt_scheme(struct wx *wx);
int wx_init_interrupt_scheme(struct wx *wx);
irqreturn_t wx_msix_clean_rings(int __always_unused irq, void *data);
void wx_free_irq(struct wx *wx);
int wx_setup_isb_resources(struct wx *wx);
void wx_free_isb_resources(struct wx *wx);
u32 wx_misc_isb(struct wx *wx, enum wx_isb_idx idx);
void wx_configure_vectors(struct wx *wx);
#endif /* _NGBE_LIB_H_ */
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
#define _WX_TYPE_H_ #define _WX_TYPE_H_
#include <linux/bitfield.h> #include <linux/bitfield.h>
#include <linux/netdevice.h>
/* Vendor ID */ /* Vendor ID */
#ifndef PCI_VENDOR_ID_WANGXUN #ifndef PCI_VENDOR_ID_WANGXUN
...@@ -66,6 +67,22 @@ ...@@ -66,6 +67,22 @@
#define WX_CFG_PORT_CTL 0x14400 #define WX_CFG_PORT_CTL 0x14400
#define WX_CFG_PORT_CTL_DRV_LOAD BIT(3) #define WX_CFG_PORT_CTL_DRV_LOAD BIT(3)
/* GPIO Registers */
#define WX_GPIO_DR 0x14800
#define WX_GPIO_DR_0 BIT(0) /* SDP0 Data Value */
#define WX_GPIO_DR_1 BIT(1) /* SDP1 Data Value */
#define WX_GPIO_DDR 0x14804
#define WX_GPIO_DDR_0 BIT(0) /* SDP0 IO direction */
#define WX_GPIO_DDR_1 BIT(1) /* SDP1 IO direction */
#define WX_GPIO_CTL 0x14808
#define WX_GPIO_INTEN 0x14830
#define WX_GPIO_INTEN_0 BIT(0)
#define WX_GPIO_INTEN_1 BIT(1)
#define WX_GPIO_INTMASK 0x14834
#define WX_GPIO_INTTYPE_LEVEL 0x14838
#define WX_GPIO_POLARITY 0x1483C
#define WX_GPIO_EOI 0x1484C
/*********************** Transmit DMA registers **************************/ /*********************** Transmit DMA registers **************************/
/* transmit global control */ /* transmit global control */
#define WX_TDM_CTL 0x18000 #define WX_TDM_CTL 0x18000
...@@ -151,8 +168,28 @@ ...@@ -151,8 +168,28 @@
/* Interrupt Registers */ /* Interrupt Registers */
#define WX_BME_CTL 0x12020 #define WX_BME_CTL 0x12020
#define WX_PX_MISC_IC 0x100 #define WX_PX_MISC_IC 0x100
#define WX_PX_MISC_ICS 0x104
#define WX_PX_MISC_IEN 0x108
#define WX_PX_INTA 0x110
#define WX_PX_GPIE 0x118
#define WX_PX_GPIE_MODEL BIT(0)
#define WX_PX_IC 0x120
#define WX_PX_IMS(_i) (0x140 + (_i) * 4) #define WX_PX_IMS(_i) (0x140 + (_i) * 4)
#define WX_PX_IMC(_i) (0x150 + (_i) * 4)
#define WX_PX_ISB_ADDR_L 0x160
#define WX_PX_ISB_ADDR_H 0x164
#define WX_PX_TRANSACTION_PENDING 0x168 #define WX_PX_TRANSACTION_PENDING 0x168
#define WX_PX_ITRSEL 0x180
#define WX_PX_ITR(_i) (0x200 + (_i) * 4)
#define WX_PX_ITR_CNT_WDIS BIT(31)
#define WX_PX_MISC_IVAR 0x4FC
#define WX_PX_IVAR(_i) (0x500 + (_i) * 4)
#define WX_PX_IVAR_ALLOC_VAL 0x80 /* Interrupt Allocation valid */
#define WX_7K_ITR 595
#define WX_12K_ITR 336
#define WX_SP_MAX_EITR 0x00000FF8U
#define WX_EM_MAX_EITR 0x00007FFCU
/* transmit DMA Registers */ /* transmit DMA Registers */
#define WX_PX_TR_CFG(_i) (0x03010 + ((_i) * 0x40)) #define WX_PX_TR_CFG(_i) (0x03010 + ((_i) * 0x40))
...@@ -312,6 +349,58 @@ enum wx_reset_type { ...@@ -312,6 +349,58 @@ enum wx_reset_type {
WX_GLOBAL_RESET WX_GLOBAL_RESET
}; };
/* iterator for handling rings in ring container */
#define wx_for_each_ring(posm, headm) \
for (posm = (headm).ring; posm; posm = posm->next)
struct wx_ring_container {
struct wx_ring *ring; /* pointer to linked list of rings */
u8 count; /* total number of rings in vector */
u8 itr; /* current ITR setting for ring */
};
struct wx_ring {
struct wx_ring *next; /* pointer to next ring in q_vector */
struct wx_q_vector *q_vector; /* backpointer to host q_vector */
struct net_device *netdev; /* netdev ring belongs to */
struct device *dev; /* device for DMA mapping */
u16 count; /* amount of descriptors */
u8 queue_index; /* needed for multiqueue queue management */
u8 reg_idx; /* holds the special value that gets
* the hardware register offset
* associated with this ring, which is
* different for DCB and RSS modes
*/
} ____cacheline_internodealigned_in_smp;
struct wx_q_vector {
struct wx *wx;
int cpu; /* CPU for DCA */
u16 v_idx; /* index of q_vector within array, also used for
* finding the bit in EICR and friends that
* represents the vector for this ring
*/
u16 itr; /* Interrupt throttle rate written to EITR */
struct wx_ring_container rx, tx;
struct napi_struct napi;
struct rcu_head rcu; /* to avoid race with update stats on free */
char name[IFNAMSIZ + 17];
/* for dynamic allocation of rings associated with this q_vector */
struct wx_ring ring[0] ____cacheline_internodealigned_in_smp;
};
enum wx_isb_idx {
WX_ISB_HEADER,
WX_ISB_MISC,
WX_ISB_VEC0,
WX_ISB_VEC1,
WX_ISB_MAX
};
struct wx { struct wx {
u8 __iomem *hw_addr; u8 __iomem *hw_addr;
struct pci_dev *pdev; struct pci_dev *pdev;
...@@ -360,6 +449,18 @@ struct wx { ...@@ -360,6 +449,18 @@ struct wx {
u32 tx_ring_count; u32 tx_ring_count;
u32 rx_ring_count; u32 rx_ring_count;
struct wx_ring *tx_ring[64] ____cacheline_aligned_in_smp;
struct wx_ring *rx_ring[64];
struct wx_q_vector *q_vector[64];
unsigned int queues_per_pool;
struct msix_entry *msix_entries;
/* misc interrupt status block */
dma_addr_t isb_dma;
u32 *isb_mem;
u32 isb_tag[WX_ISB_MAX];
#define WX_MAX_RETA_ENTRIES 128 #define WX_MAX_RETA_ENTRIES 128
u8 rss_indir_tbl[WX_MAX_RETA_ENTRIES]; u8 rss_indir_tbl[WX_MAX_RETA_ENTRIES];
......
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