mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-09 23:00:21 +00:00
Current release - regressions:
- tcp: fix tcp_disordered_ack() vs usec TS resolution Current release - new code bugs: - dpll: sanitize possible null pointer dereference in dpll_pin_parent_pin_set() - eth: octeon_ep: initialise control mbox tasks before using APIs Previous releases - regressions: - io_uring/af_unix: disable sending io_uring over sockets - eth: mlx5e: - TC, don't offload post action rule if not supported - fix possible deadlock on mlx5e_tx_timeout_work - eth: iavf: fix iavf_shutdown to call iavf_remove instead iavf_close - eth: bnxt_en: fix skb recycling logic in bnxt_deliver_skb() - eth: ena: fix DMA syncing in XDP path when SWIOTLB is on - eth: team: fix use-after-free when an option instance allocation fails Previous releases - always broken: - neighbour: don't let neigh_forced_gc() disable preemption for long - net: prevent mss overflow in skb_segment() - ipv6: support reporting otherwise unknown prefix flags in RTM_NEWPREFIX - tcp: remove acked SYN flag from packet in the transmit queue correctly - eth: octeontx2-af: - fix a use-after-free in rvu_nix_register_reporters - fix promisc mcam entry action - eth: dwmac-loongson: make sure MDIO is initialized before use - eth: atlantic: fix double free in ring reinit logic Signed-off-by: Paolo Abeni <pabeni@redhat.com> -----BEGIN PGP SIGNATURE----- iQJGBAABCAAwFiEEg1AjqC77wbdLX2LbKSR5jcyPE6QFAmV6/E4SHHBhYmVuaUBy ZWRoYXQuY29tAAoJECkkeY3MjxOkas8P/if7c+MUxkegwRbO0vOObG/B/QXJ+dR8 UcqPYnroF0u7s2KhDqbj/h9msbNhAmWzrhzk4c086hpIkq34piiS+W319K/tia6u H1fRbVfBAo/mcQ8eG7EPiDYrNKDhuiGL6Gsd/Fdl9om1CMjW4fAFWY1F79OoL7F5 mDTiVdnHik06CGgic6zRdp4xy6zHZ5oBanS60VNjLa4sb69g1Z1fjLQoJt4qXYbJ jWZ9QkJ1t/98MOca6mFIZNJY+f3doYMRv5dP1oUSJmbFGfCYjbMcdpa3BQlTiDdu 96xWF01p5uJ2UBib0nKiGSZmg1Xz1xal9V+ahApmTe8BpZAn6PJeXYbtMQO2SXYf VW3V7rSkCB482UPN3siubhtZnOE5oYixM/5OL/UGZv113ShF8HNjj4AAZOeXtJPc 75QeQOSRy+vhopEexCZ+21Zou+Ao3MjEFlVMCfTJ7couvjFg9LNkazHTXfAkwe0J QaLYpbbaXwS3lOspwWFK2rV/G+3fpJZBrW2WRwlLBMMg3lXLuo2OdqrewV9GoI36 ksqv2c5mMtLwomdM2QfK0zeUc6kDeqlpEcjMzfapn/92A+pcAmcBpT2FfFDR4QUz nhoULC2XvTdlri7nxxp/9AYbQK0DFXqChPPV3NdcN/HPI7fYFHTv387ZkLU5zDlN nwnXj8rbA0d5 =84lK -----END PGP SIGNATURE----- Merge tag 'net-6.7-rc6' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net Pull networking fixes from Paolo Abeni: "Current release - regressions: - tcp: fix tcp_disordered_ack() vs usec TS resolution Current release - new code bugs: - dpll: sanitize possible null pointer dereference in dpll_pin_parent_pin_set() - eth: octeon_ep: initialise control mbox tasks before using APIs Previous releases - regressions: - io_uring/af_unix: disable sending io_uring over sockets - eth: mlx5e: - TC, don't offload post action rule if not supported - fix possible deadlock on mlx5e_tx_timeout_work - eth: iavf: fix iavf_shutdown to call iavf_remove instead iavf_close - eth: bnxt_en: fix skb recycling logic in bnxt_deliver_skb() - eth: ena: fix DMA syncing in XDP path when SWIOTLB is on - eth: team: fix use-after-free when an option instance allocation fails Previous releases - always broken: - neighbour: don't let neigh_forced_gc() disable preemption for long - net: prevent mss overflow in skb_segment() - ipv6: support reporting otherwise unknown prefix flags in RTM_NEWPREFIX - tcp: remove acked SYN flag from packet in the transmit queue correctly - eth: octeontx2-af: - fix a use-after-free in rvu_nix_register_reporters - fix promisc mcam entry action - eth: dwmac-loongson: make sure MDIO is initialized before use - eth: atlantic: fix double free in ring reinit logic" * tag 'net-6.7-rc6' of git://git.kernel.org/pub/scm/linux/kernel/git/netdev/net: (62 commits) net: atlantic: fix double free in ring reinit logic appletalk: Fix Use-After-Free in atalk_ioctl net: stmmac: Handle disabled MDIO busses from devicetree net: stmmac: dwmac-qcom-ethqos: Fix drops in 10M SGMII RX dpaa2-switch: do not ask for MDB, VLAN and FDB replay dpaa2-switch: fix size of the dma_unmap net: prevent mss overflow in skb_segment() vsock/virtio: Fix unsigned integer wrap around in virtio_transport_has_space() Revert "tcp: disable tcp_autocorking for socket when TCP_NODELAY flag is set" MIPS: dts: loongson: drop incorrect dwmac fallback compatible stmmac: dwmac-loongson: drop useless check for compatible fallback stmmac: dwmac-loongson: Make sure MDIO is initialized before use tcp: disable tcp_autocorking for socket when TCP_NODELAY flag is set dpll: sanitize possible null pointer dereference in dpll_pin_parent_pin_set() net: ena: Fix XDP redirection error net: ena: Fix DMA syncing in XDP path when SWIOTLB is on net: ena: Fix xdp drops handling due to multibuf packets net: ena: Destroy correct number of xdp queues upon failure net: Remove acked SYN flag from packet in the transmit queue correctly qed: Fix a potential use-after-free in qed_cxt_tables_alloc ...
This commit is contained in:
commit
c7402612e2
@ -19564,7 +19564,6 @@ S: Maintained
|
||||
F: drivers/misc/sgi-xp/
|
||||
|
||||
SHARED MEMORY COMMUNICATIONS (SMC) SOCKETS
|
||||
M: Karsten Graul <kgraul@linux.ibm.com>
|
||||
M: Wenjia Zhang <wenjia@linux.ibm.com>
|
||||
M: Jan Karcher <jaka@linux.ibm.com>
|
||||
R: D. Wythe <alibuda@linux.alibaba.com>
|
||||
|
@ -130,8 +130,7 @@
|
||||
compatible = "pci0014,7a03.0",
|
||||
"pci0014,7a03",
|
||||
"pciclass0c0320",
|
||||
"pciclass0c03",
|
||||
"loongson, pci-gmac";
|
||||
"pciclass0c03";
|
||||
|
||||
reg = <0x1800 0x0 0x0 0x0 0x0>;
|
||||
interrupts = <12 IRQ_TYPE_LEVEL_LOW>,
|
||||
|
@ -193,8 +193,7 @@
|
||||
compatible = "pci0014,7a03.0",
|
||||
"pci0014,7a03",
|
||||
"pciclass020000",
|
||||
"pciclass0200",
|
||||
"loongson, pci-gmac";
|
||||
"pciclass0200";
|
||||
|
||||
reg = <0x1800 0x0 0x0 0x0 0x0>;
|
||||
interrupts = <12 IRQ_TYPE_LEVEL_HIGH>,
|
||||
|
@ -449,9 +449,9 @@ static ssize_t console_show(struct device *dev, struct device_attribute *attr,
|
||||
struct sk_buff *skb;
|
||||
unsigned int len;
|
||||
|
||||
spin_lock(&card->cli_queue_lock);
|
||||
spin_lock_bh(&card->cli_queue_lock);
|
||||
skb = skb_dequeue(&card->cli_queue[SOLOS_CHAN(atmdev)]);
|
||||
spin_unlock(&card->cli_queue_lock);
|
||||
spin_unlock_bh(&card->cli_queue_lock);
|
||||
if(skb == NULL)
|
||||
return sprintf(buf, "No data.\n");
|
||||
|
||||
@ -956,14 +956,14 @@ static void pclose(struct atm_vcc *vcc)
|
||||
struct pkt_hdr *header;
|
||||
|
||||
/* Remove any yet-to-be-transmitted packets from the pending queue */
|
||||
spin_lock(&card->tx_queue_lock);
|
||||
spin_lock_bh(&card->tx_queue_lock);
|
||||
skb_queue_walk_safe(&card->tx_queue[port], skb, tmpskb) {
|
||||
if (SKB_CB(skb)->vcc == vcc) {
|
||||
skb_unlink(skb, &card->tx_queue[port]);
|
||||
solos_pop(vcc, skb);
|
||||
}
|
||||
}
|
||||
spin_unlock(&card->tx_queue_lock);
|
||||
spin_unlock_bh(&card->tx_queue_lock);
|
||||
|
||||
skb = alloc_skb(sizeof(*header), GFP_KERNEL);
|
||||
if (!skb) {
|
||||
|
@ -925,7 +925,6 @@ dpll_pin_parent_pin_set(struct dpll_pin *pin, struct nlattr *parent_nest,
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
struct nlattr *tb[DPLL_A_PIN_MAX + 1];
|
||||
enum dpll_pin_state state;
|
||||
u32 ppin_idx;
|
||||
int ret;
|
||||
|
||||
@ -936,10 +935,14 @@ dpll_pin_parent_pin_set(struct dpll_pin *pin, struct nlattr *parent_nest,
|
||||
return -EINVAL;
|
||||
}
|
||||
ppin_idx = nla_get_u32(tb[DPLL_A_PIN_PARENT_ID]);
|
||||
state = nla_get_u32(tb[DPLL_A_PIN_STATE]);
|
||||
ret = dpll_pin_on_pin_state_set(pin, ppin_idx, state, extack);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (tb[DPLL_A_PIN_STATE]) {
|
||||
enum dpll_pin_state state = nla_get_u32(tb[DPLL_A_PIN_STATE]);
|
||||
|
||||
ret = dpll_pin_on_pin_state_set(pin, ppin_idx, state, extack);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -328,9 +328,6 @@ static int ena_com_create_and_store_tx_meta_desc(struct ena_com_io_sq *io_sq,
|
||||
* compare it to the stored version, just create the meta
|
||||
*/
|
||||
if (io_sq->disable_meta_caching) {
|
||||
if (unlikely(!ena_tx_ctx->meta_valid))
|
||||
return -EINVAL;
|
||||
|
||||
*have_meta = true;
|
||||
return ena_com_create_meta(io_sq, ena_meta);
|
||||
}
|
||||
|
@ -74,6 +74,8 @@ static void ena_unmap_tx_buff(struct ena_ring *tx_ring,
|
||||
struct ena_tx_buffer *tx_info);
|
||||
static int ena_create_io_tx_queues_in_range(struct ena_adapter *adapter,
|
||||
int first_index, int count);
|
||||
static void ena_free_all_io_tx_resources_in_range(struct ena_adapter *adapter,
|
||||
int first_index, int count);
|
||||
|
||||
/* Increase a stat by cnt while holding syncp seqlock on 32bit machines */
|
||||
static void ena_increase_stat(u64 *statp, u64 cnt,
|
||||
@ -457,23 +459,22 @@ static void ena_init_all_xdp_queues(struct ena_adapter *adapter)
|
||||
|
||||
static int ena_setup_and_create_all_xdp_queues(struct ena_adapter *adapter)
|
||||
{
|
||||
u32 xdp_first_ring = adapter->xdp_first_ring;
|
||||
u32 xdp_num_queues = adapter->xdp_num_queues;
|
||||
int rc = 0;
|
||||
|
||||
rc = ena_setup_tx_resources_in_range(adapter, adapter->xdp_first_ring,
|
||||
adapter->xdp_num_queues);
|
||||
rc = ena_setup_tx_resources_in_range(adapter, xdp_first_ring, xdp_num_queues);
|
||||
if (rc)
|
||||
goto setup_err;
|
||||
|
||||
rc = ena_create_io_tx_queues_in_range(adapter,
|
||||
adapter->xdp_first_ring,
|
||||
adapter->xdp_num_queues);
|
||||
rc = ena_create_io_tx_queues_in_range(adapter, xdp_first_ring, xdp_num_queues);
|
||||
if (rc)
|
||||
goto create_err;
|
||||
|
||||
return 0;
|
||||
|
||||
create_err:
|
||||
ena_free_all_io_tx_resources(adapter);
|
||||
ena_free_all_io_tx_resources_in_range(adapter, xdp_first_ring, xdp_num_queues);
|
||||
setup_err:
|
||||
return rc;
|
||||
}
|
||||
@ -1492,11 +1493,6 @@ static struct sk_buff *ena_rx_skb(struct ena_ring *rx_ring,
|
||||
if (unlikely(!skb))
|
||||
return NULL;
|
||||
|
||||
/* sync this buffer for CPU use */
|
||||
dma_sync_single_for_cpu(rx_ring->dev,
|
||||
dma_unmap_addr(&rx_info->ena_buf, paddr) + pkt_offset,
|
||||
len,
|
||||
DMA_FROM_DEVICE);
|
||||
skb_copy_to_linear_data(skb, buf_addr + buf_offset, len);
|
||||
dma_sync_single_for_device(rx_ring->dev,
|
||||
dma_unmap_addr(&rx_info->ena_buf, paddr) + pkt_offset,
|
||||
@ -1515,17 +1511,10 @@ static struct sk_buff *ena_rx_skb(struct ena_ring *rx_ring,
|
||||
|
||||
buf_len = SKB_DATA_ALIGN(len + buf_offset + tailroom);
|
||||
|
||||
pre_reuse_paddr = dma_unmap_addr(&rx_info->ena_buf, paddr);
|
||||
|
||||
/* If XDP isn't loaded try to reuse part of the RX buffer */
|
||||
reuse_rx_buf_page = !is_xdp_loaded &&
|
||||
ena_try_rx_buf_page_reuse(rx_info, buf_len, len, pkt_offset);
|
||||
|
||||
dma_sync_single_for_cpu(rx_ring->dev,
|
||||
pre_reuse_paddr + pkt_offset,
|
||||
len,
|
||||
DMA_FROM_DEVICE);
|
||||
|
||||
if (!reuse_rx_buf_page)
|
||||
ena_unmap_rx_buff_attrs(rx_ring, rx_info, DMA_ATTR_SKIP_CPU_SYNC);
|
||||
|
||||
@ -1671,20 +1660,23 @@ static void ena_set_rx_hash(struct ena_ring *rx_ring,
|
||||
}
|
||||
}
|
||||
|
||||
static int ena_xdp_handle_buff(struct ena_ring *rx_ring, struct xdp_buff *xdp)
|
||||
static int ena_xdp_handle_buff(struct ena_ring *rx_ring, struct xdp_buff *xdp, u16 num_descs)
|
||||
{
|
||||
struct ena_rx_buffer *rx_info;
|
||||
int ret;
|
||||
|
||||
/* XDP multi-buffer packets not supported */
|
||||
if (unlikely(num_descs > 1)) {
|
||||
netdev_err_once(rx_ring->adapter->netdev,
|
||||
"xdp: dropped unsupported multi-buffer packets\n");
|
||||
ena_increase_stat(&rx_ring->rx_stats.xdp_drop, 1, &rx_ring->syncp);
|
||||
return ENA_XDP_DROP;
|
||||
}
|
||||
|
||||
rx_info = &rx_ring->rx_buffer_info[rx_ring->ena_bufs[0].req_id];
|
||||
xdp_prepare_buff(xdp, page_address(rx_info->page),
|
||||
rx_info->buf_offset,
|
||||
rx_ring->ena_bufs[0].len, false);
|
||||
/* If for some reason we received a bigger packet than
|
||||
* we expect, then we simply drop it
|
||||
*/
|
||||
if (unlikely(rx_ring->ena_bufs[0].len > ENA_XDP_MAX_MTU))
|
||||
return ENA_XDP_DROP;
|
||||
|
||||
ret = ena_xdp_execute(rx_ring, xdp);
|
||||
|
||||
@ -1719,6 +1711,7 @@ static int ena_clean_rx_irq(struct ena_ring *rx_ring, struct napi_struct *napi,
|
||||
int xdp_flags = 0;
|
||||
int total_len = 0;
|
||||
int xdp_verdict;
|
||||
u8 pkt_offset;
|
||||
int rc = 0;
|
||||
int i;
|
||||
|
||||
@ -1745,15 +1738,21 @@ static int ena_clean_rx_irq(struct ena_ring *rx_ring, struct napi_struct *napi,
|
||||
|
||||
/* First descriptor might have an offset set by the device */
|
||||
rx_info = &rx_ring->rx_buffer_info[rx_ring->ena_bufs[0].req_id];
|
||||
rx_info->buf_offset += ena_rx_ctx.pkt_offset;
|
||||
pkt_offset = ena_rx_ctx.pkt_offset;
|
||||
rx_info->buf_offset += pkt_offset;
|
||||
|
||||
netif_dbg(rx_ring->adapter, rx_status, rx_ring->netdev,
|
||||
"rx_poll: q %d got packet from ena. descs #: %d l3 proto %d l4 proto %d hash: %x\n",
|
||||
rx_ring->qid, ena_rx_ctx.descs, ena_rx_ctx.l3_proto,
|
||||
ena_rx_ctx.l4_proto, ena_rx_ctx.hash);
|
||||
|
||||
dma_sync_single_for_cpu(rx_ring->dev,
|
||||
dma_unmap_addr(&rx_info->ena_buf, paddr) + pkt_offset,
|
||||
rx_ring->ena_bufs[0].len,
|
||||
DMA_FROM_DEVICE);
|
||||
|
||||
if (ena_xdp_present_ring(rx_ring))
|
||||
xdp_verdict = ena_xdp_handle_buff(rx_ring, &xdp);
|
||||
xdp_verdict = ena_xdp_handle_buff(rx_ring, &xdp, ena_rx_ctx.descs);
|
||||
|
||||
/* allocate skb and fill it */
|
||||
if (xdp_verdict == ENA_XDP_PASS)
|
||||
@ -1777,7 +1776,7 @@ static int ena_clean_rx_irq(struct ena_ring *rx_ring, struct napi_struct *napi,
|
||||
if (xdp_verdict & ENA_XDP_FORWARDED) {
|
||||
ena_unmap_rx_buff_attrs(rx_ring,
|
||||
&rx_ring->rx_buffer_info[req_id],
|
||||
0);
|
||||
DMA_ATTR_SKIP_CPU_SYNC);
|
||||
rx_ring->rx_buffer_info[req_id].page = NULL;
|
||||
}
|
||||
}
|
||||
|
@ -938,11 +938,14 @@ void aq_ring_free(struct aq_ring_s *self)
|
||||
return;
|
||||
|
||||
kfree(self->buff_ring);
|
||||
self->buff_ring = NULL;
|
||||
|
||||
if (self->dx_ring)
|
||||
if (self->dx_ring) {
|
||||
dma_free_coherent(aq_nic_get_dev(self->aq_nic),
|
||||
self->size * self->dx_size, self->dx_ring,
|
||||
self->dx_ring_pa);
|
||||
self->dx_ring = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
unsigned int aq_ring_fill_stats_data(struct aq_ring_s *self, u64 *data)
|
||||
|
@ -1748,16 +1748,32 @@ static void bnxt_tpa_agg(struct bnxt *bp, struct bnxt_rx_ring_info *rxr,
|
||||
static void bnxt_deliver_skb(struct bnxt *bp, struct bnxt_napi *bnapi,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
skb_mark_for_recycle(skb);
|
||||
|
||||
if (skb->dev != bp->dev) {
|
||||
/* this packet belongs to a vf-rep */
|
||||
bnxt_vf_rep_rx(bp, skb);
|
||||
return;
|
||||
}
|
||||
skb_record_rx_queue(skb, bnapi->index);
|
||||
skb_mark_for_recycle(skb);
|
||||
napi_gro_receive(&bnapi->napi, skb);
|
||||
}
|
||||
|
||||
static bool bnxt_rx_ts_valid(struct bnxt *bp, u32 flags,
|
||||
struct rx_cmp_ext *rxcmp1, u32 *cmpl_ts)
|
||||
{
|
||||
u32 ts = le32_to_cpu(rxcmp1->rx_cmp_timestamp);
|
||||
|
||||
if (BNXT_PTP_RX_TS_VALID(flags))
|
||||
goto ts_valid;
|
||||
if (!bp->ptp_all_rx_tstamp || !ts || !BNXT_ALL_RX_TS_VALID(flags))
|
||||
return false;
|
||||
|
||||
ts_valid:
|
||||
*cmpl_ts = ts;
|
||||
return true;
|
||||
}
|
||||
|
||||
/* returns the following:
|
||||
* 1 - 1 packet successfully received
|
||||
* 0 - successful TPA_START, packet not completed yet
|
||||
@ -1783,6 +1799,7 @@ static int bnxt_rx_pkt(struct bnxt *bp, struct bnxt_cp_ring_info *cpr,
|
||||
struct sk_buff *skb;
|
||||
struct xdp_buff xdp;
|
||||
u32 flags, misc;
|
||||
u32 cmpl_ts;
|
||||
void *data;
|
||||
int rc = 0;
|
||||
|
||||
@ -2005,10 +2022,8 @@ static int bnxt_rx_pkt(struct bnxt *bp, struct bnxt_cp_ring_info *cpr,
|
||||
}
|
||||
}
|
||||
|
||||
if (unlikely((flags & RX_CMP_FLAGS_ITYPES_MASK) ==
|
||||
RX_CMP_FLAGS_ITYPE_PTP_W_TS) || bp->ptp_all_rx_tstamp) {
|
||||
if (bnxt_rx_ts_valid(bp, flags, rxcmp1, &cmpl_ts)) {
|
||||
if (bp->flags & BNXT_FLAG_CHIP_P5) {
|
||||
u32 cmpl_ts = le32_to_cpu(rxcmp1->rx_cmp_timestamp);
|
||||
u64 ns, ts;
|
||||
|
||||
if (!bnxt_get_rx_ts_p5(bp, &ts, cmpl_ts)) {
|
||||
@ -10731,10 +10746,8 @@ static void __bnxt_close_nic(struct bnxt *bp, bool irq_re_init,
|
||||
bnxt_free_mem(bp, irq_re_init);
|
||||
}
|
||||
|
||||
int bnxt_close_nic(struct bnxt *bp, bool irq_re_init, bool link_re_init)
|
||||
void bnxt_close_nic(struct bnxt *bp, bool irq_re_init, bool link_re_init)
|
||||
{
|
||||
int rc = 0;
|
||||
|
||||
if (test_bit(BNXT_STATE_IN_FW_RESET, &bp->state)) {
|
||||
/* If we get here, it means firmware reset is in progress
|
||||
* while we are trying to close. We can safely proceed with
|
||||
@ -10749,15 +10762,18 @@ int bnxt_close_nic(struct bnxt *bp, bool irq_re_init, bool link_re_init)
|
||||
|
||||
#ifdef CONFIG_BNXT_SRIOV
|
||||
if (bp->sriov_cfg) {
|
||||
int rc;
|
||||
|
||||
rc = wait_event_interruptible_timeout(bp->sriov_cfg_wait,
|
||||
!bp->sriov_cfg,
|
||||
BNXT_SRIOV_CFG_WAIT_TMO);
|
||||
if (rc)
|
||||
netdev_warn(bp->dev, "timeout waiting for SRIOV config operation to complete!\n");
|
||||
if (!rc)
|
||||
netdev_warn(bp->dev, "timeout waiting for SRIOV config operation to complete, proceeding to close!\n");
|
||||
else if (rc < 0)
|
||||
netdev_warn(bp->dev, "SRIOV config operation interrupted, proceeding to close!\n");
|
||||
}
|
||||
#endif
|
||||
__bnxt_close_nic(bp, irq_re_init, link_re_init);
|
||||
return rc;
|
||||
}
|
||||
|
||||
static int bnxt_close(struct net_device *dev)
|
||||
@ -13940,6 +13956,8 @@ static int bnxt_resume(struct device *device)
|
||||
if (rc)
|
||||
goto resume_exit;
|
||||
|
||||
bnxt_clear_reservations(bp, true);
|
||||
|
||||
if (bnxt_hwrm_func_drv_rgtr(bp, NULL, 0, false)) {
|
||||
rc = -ENODEV;
|
||||
goto resume_exit;
|
||||
|
@ -161,7 +161,7 @@ struct rx_cmp {
|
||||
#define RX_CMP_FLAGS_ERROR (1 << 6)
|
||||
#define RX_CMP_FLAGS_PLACEMENT (7 << 7)
|
||||
#define RX_CMP_FLAGS_RSS_VALID (1 << 10)
|
||||
#define RX_CMP_FLAGS_UNUSED (1 << 11)
|
||||
#define RX_CMP_FLAGS_PKT_METADATA_PRESENT (1 << 11)
|
||||
#define RX_CMP_FLAGS_ITYPES_SHIFT 12
|
||||
#define RX_CMP_FLAGS_ITYPES_MASK 0xf000
|
||||
#define RX_CMP_FLAGS_ITYPE_UNKNOWN (0 << 12)
|
||||
@ -188,6 +188,12 @@ struct rx_cmp {
|
||||
__le32 rx_cmp_rss_hash;
|
||||
};
|
||||
|
||||
#define BNXT_PTP_RX_TS_VALID(flags) \
|
||||
(((flags) & RX_CMP_FLAGS_ITYPES_MASK) == RX_CMP_FLAGS_ITYPE_PTP_W_TS)
|
||||
|
||||
#define BNXT_ALL_RX_TS_VALID(flags) \
|
||||
!((flags) & RX_CMP_FLAGS_PKT_METADATA_PRESENT)
|
||||
|
||||
#define RX_CMP_HASH_VALID(rxcmp) \
|
||||
((rxcmp)->rx_cmp_len_flags_type & cpu_to_le32(RX_CMP_FLAGS_RSS_VALID))
|
||||
|
||||
@ -2375,7 +2381,7 @@ int bnxt_open_nic(struct bnxt *, bool, bool);
|
||||
int bnxt_half_open_nic(struct bnxt *bp);
|
||||
void bnxt_half_close_nic(struct bnxt *bp);
|
||||
void bnxt_reenable_sriov(struct bnxt *bp);
|
||||
int bnxt_close_nic(struct bnxt *, bool, bool);
|
||||
void bnxt_close_nic(struct bnxt *, bool, bool);
|
||||
void bnxt_get_ring_err_stats(struct bnxt *bp,
|
||||
struct bnxt_total_ring_err_stats *stats);
|
||||
int bnxt_dbg_hwrm_rd_reg(struct bnxt *bp, u32 reg_off, u16 num_words,
|
||||
|
@ -449,15 +449,8 @@ static int bnxt_dl_reload_down(struct devlink *dl, bool netns_change,
|
||||
return -ENODEV;
|
||||
}
|
||||
bnxt_ulp_stop(bp);
|
||||
if (netif_running(bp->dev)) {
|
||||
rc = bnxt_close_nic(bp, true, true);
|
||||
if (rc) {
|
||||
NL_SET_ERR_MSG_MOD(extack, "Failed to close");
|
||||
dev_close(bp->dev);
|
||||
rtnl_unlock();
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (netif_running(bp->dev))
|
||||
bnxt_close_nic(bp, true, true);
|
||||
bnxt_vf_reps_free(bp);
|
||||
rc = bnxt_hwrm_func_drv_unrgtr(bp);
|
||||
if (rc) {
|
||||
|
@ -165,9 +165,8 @@ static int bnxt_set_coalesce(struct net_device *dev,
|
||||
reset_coalesce:
|
||||
if (test_bit(BNXT_STATE_OPEN, &bp->state)) {
|
||||
if (update_stats) {
|
||||
rc = bnxt_close_nic(bp, true, false);
|
||||
if (!rc)
|
||||
rc = bnxt_open_nic(bp, true, false);
|
||||
bnxt_close_nic(bp, true, false);
|
||||
rc = bnxt_open_nic(bp, true, false);
|
||||
} else {
|
||||
rc = bnxt_hwrm_set_coal(bp);
|
||||
}
|
||||
@ -972,12 +971,7 @@ static int bnxt_set_channels(struct net_device *dev,
|
||||
* before PF unload
|
||||
*/
|
||||
}
|
||||
rc = bnxt_close_nic(bp, true, false);
|
||||
if (rc) {
|
||||
netdev_err(bp->dev, "Set channel failure rc :%x\n",
|
||||
rc);
|
||||
return rc;
|
||||
}
|
||||
bnxt_close_nic(bp, true, false);
|
||||
}
|
||||
|
||||
if (sh) {
|
||||
@ -4042,12 +4036,7 @@ static void bnxt_self_test(struct net_device *dev, struct ethtool_test *etest,
|
||||
bnxt_run_fw_tests(bp, test_mask, &test_results);
|
||||
} else {
|
||||
bnxt_ulp_stop(bp);
|
||||
rc = bnxt_close_nic(bp, true, false);
|
||||
if (rc) {
|
||||
etest->flags |= ETH_TEST_FL_FAILED;
|
||||
bnxt_ulp_start(bp, rc);
|
||||
return;
|
||||
}
|
||||
bnxt_close_nic(bp, true, false);
|
||||
bnxt_run_fw_tests(bp, test_mask, &test_results);
|
||||
|
||||
buf[BNXT_MACLPBK_TEST_IDX] = 1;
|
||||
|
@ -521,9 +521,8 @@ static int bnxt_hwrm_ptp_cfg(struct bnxt *bp)
|
||||
|
||||
if (netif_running(bp->dev)) {
|
||||
if (ptp->rx_filter == HWTSTAMP_FILTER_ALL) {
|
||||
rc = bnxt_close_nic(bp, false, false);
|
||||
if (!rc)
|
||||
rc = bnxt_open_nic(bp, false, false);
|
||||
bnxt_close_nic(bp, false, false);
|
||||
rc = bnxt_open_nic(bp, false, false);
|
||||
} else {
|
||||
bnxt_ptp_cfg_tstamp_filters(bp);
|
||||
}
|
||||
|
@ -139,7 +139,8 @@ int dpaa2_switch_acl_entry_add(struct dpaa2_switch_filter_block *filter_block,
|
||||
err = dpsw_acl_add_entry(ethsw->mc_io, 0, ethsw->dpsw_handle,
|
||||
filter_block->acl_id, acl_entry_cfg);
|
||||
|
||||
dma_unmap_single(dev, acl_entry_cfg->key_iova, sizeof(cmd_buff),
|
||||
dma_unmap_single(dev, acl_entry_cfg->key_iova,
|
||||
DPAA2_ETHSW_PORT_ACL_CMD_BUF_SIZE,
|
||||
DMA_TO_DEVICE);
|
||||
if (err) {
|
||||
dev_err(dev, "dpsw_acl_add_entry() failed %d\n", err);
|
||||
@ -181,8 +182,8 @@ dpaa2_switch_acl_entry_remove(struct dpaa2_switch_filter_block *block,
|
||||
err = dpsw_acl_remove_entry(ethsw->mc_io, 0, ethsw->dpsw_handle,
|
||||
block->acl_id, acl_entry_cfg);
|
||||
|
||||
dma_unmap_single(dev, acl_entry_cfg->key_iova, sizeof(cmd_buff),
|
||||
DMA_TO_DEVICE);
|
||||
dma_unmap_single(dev, acl_entry_cfg->key_iova,
|
||||
DPAA2_ETHSW_PORT_ACL_CMD_BUF_SIZE, DMA_TO_DEVICE);
|
||||
if (err) {
|
||||
dev_err(dev, "dpsw_acl_remove_entry() failed %d\n", err);
|
||||
kfree(cmd_buff);
|
||||
|
@ -1998,9 +1998,6 @@ static int dpaa2_switch_port_attr_set_event(struct net_device *netdev,
|
||||
return notifier_from_errno(err);
|
||||
}
|
||||
|
||||
static struct notifier_block dpaa2_switch_port_switchdev_nb;
|
||||
static struct notifier_block dpaa2_switch_port_switchdev_blocking_nb;
|
||||
|
||||
static int dpaa2_switch_port_bridge_join(struct net_device *netdev,
|
||||
struct net_device *upper_dev,
|
||||
struct netlink_ext_ack *extack)
|
||||
@ -2043,9 +2040,7 @@ static int dpaa2_switch_port_bridge_join(struct net_device *netdev,
|
||||
goto err_egress_flood;
|
||||
|
||||
err = switchdev_bridge_port_offload(netdev, netdev, NULL,
|
||||
&dpaa2_switch_port_switchdev_nb,
|
||||
&dpaa2_switch_port_switchdev_blocking_nb,
|
||||
false, extack);
|
||||
NULL, NULL, false, extack);
|
||||
if (err)
|
||||
goto err_switchdev_offload;
|
||||
|
||||
@ -2079,9 +2074,7 @@ static int dpaa2_switch_port_restore_rxvlan(struct net_device *vdev, int vid, vo
|
||||
|
||||
static void dpaa2_switch_port_pre_bridge_leave(struct net_device *netdev)
|
||||
{
|
||||
switchdev_bridge_port_unoffload(netdev, NULL,
|
||||
&dpaa2_switch_port_switchdev_nb,
|
||||
&dpaa2_switch_port_switchdev_blocking_nb);
|
||||
switchdev_bridge_port_unoffload(netdev, NULL, NULL, NULL);
|
||||
}
|
||||
|
||||
static int dpaa2_switch_port_bridge_leave(struct net_device *netdev)
|
||||
|
@ -3731,31 +3731,26 @@ static int fec_set_features(struct net_device *netdev,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static u16 fec_enet_get_raw_vlan_tci(struct sk_buff *skb)
|
||||
{
|
||||
struct vlan_ethhdr *vhdr;
|
||||
unsigned short vlan_TCI = 0;
|
||||
|
||||
if (skb->protocol == htons(ETH_P_ALL)) {
|
||||
vhdr = (struct vlan_ethhdr *)(skb->data);
|
||||
vlan_TCI = ntohs(vhdr->h_vlan_TCI);
|
||||
}
|
||||
|
||||
return vlan_TCI;
|
||||
}
|
||||
|
||||
static u16 fec_enet_select_queue(struct net_device *ndev, struct sk_buff *skb,
|
||||
struct net_device *sb_dev)
|
||||
{
|
||||
struct fec_enet_private *fep = netdev_priv(ndev);
|
||||
u16 vlan_tag;
|
||||
u16 vlan_tag = 0;
|
||||
|
||||
if (!(fep->quirks & FEC_QUIRK_HAS_AVB))
|
||||
return netdev_pick_tx(ndev, skb, NULL);
|
||||
|
||||
vlan_tag = fec_enet_get_raw_vlan_tci(skb);
|
||||
if (!vlan_tag)
|
||||
/* VLAN is present in the payload.*/
|
||||
if (eth_type_vlan(skb->protocol)) {
|
||||
struct vlan_ethhdr *vhdr = skb_vlan_eth_hdr(skb);
|
||||
|
||||
vlan_tag = ntohs(vhdr->h_vlan_TCI);
|
||||
/* VLAN is present in the skb but not yet pushed in the payload.*/
|
||||
} else if (skb_vlan_tag_present(skb)) {
|
||||
vlan_tag = skb->vlan_tci;
|
||||
} else {
|
||||
return vlan_tag;
|
||||
}
|
||||
|
||||
return fec_enet_vlan_pri_to_queue[vlan_tag >> 13];
|
||||
}
|
||||
|
@ -292,6 +292,7 @@ struct iavf_adapter {
|
||||
#define IAVF_FLAG_QUEUES_DISABLED BIT(17)
|
||||
#define IAVF_FLAG_SETUP_NETDEV_FEATURES BIT(18)
|
||||
#define IAVF_FLAG_REINIT_MSIX_NEEDED BIT(20)
|
||||
#define IAVF_FLAG_FDIR_ENABLED BIT(21)
|
||||
/* duplicates for common code */
|
||||
#define IAVF_FLAG_DCB_ENABLED 0
|
||||
/* flags for admin queue service task */
|
||||
|
@ -1061,7 +1061,7 @@ iavf_get_ethtool_fdir_entry(struct iavf_adapter *adapter,
|
||||
struct iavf_fdir_fltr *rule = NULL;
|
||||
int ret = 0;
|
||||
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
if (!(adapter->flags & IAVF_FLAG_FDIR_ENABLED))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
@ -1203,7 +1203,7 @@ iavf_get_fdir_fltr_ids(struct iavf_adapter *adapter, struct ethtool_rxnfc *cmd,
|
||||
unsigned int cnt = 0;
|
||||
int val = 0;
|
||||
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
if (!(adapter->flags & IAVF_FLAG_FDIR_ENABLED))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
cmd->data = IAVF_MAX_FDIR_FILTERS;
|
||||
@ -1395,7 +1395,7 @@ static int iavf_add_fdir_ethtool(struct iavf_adapter *adapter, struct ethtool_rx
|
||||
int count = 50;
|
||||
int err;
|
||||
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
if (!(adapter->flags & IAVF_FLAG_FDIR_ENABLED))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
if (fsp->flow_type & FLOW_MAC_EXT)
|
||||
@ -1436,12 +1436,16 @@ static int iavf_add_fdir_ethtool(struct iavf_adapter *adapter, struct ethtool_rx
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
iavf_fdir_list_add_fltr(adapter, fltr);
|
||||
adapter->fdir_active_fltr++;
|
||||
fltr->state = IAVF_FDIR_FLTR_ADD_REQUEST;
|
||||
adapter->aq_required |= IAVF_FLAG_AQ_ADD_FDIR_FILTER;
|
||||
if (adapter->link_up) {
|
||||
fltr->state = IAVF_FDIR_FLTR_ADD_REQUEST;
|
||||
adapter->aq_required |= IAVF_FLAG_AQ_ADD_FDIR_FILTER;
|
||||
} else {
|
||||
fltr->state = IAVF_FDIR_FLTR_INACTIVE;
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
|
||||
mod_delayed_work(adapter->wq, &adapter->watchdog_task, 0);
|
||||
|
||||
if (adapter->link_up)
|
||||
mod_delayed_work(adapter->wq, &adapter->watchdog_task, 0);
|
||||
ret:
|
||||
if (err && fltr)
|
||||
kfree(fltr);
|
||||
@ -1463,7 +1467,7 @@ static int iavf_del_fdir_ethtool(struct iavf_adapter *adapter, struct ethtool_rx
|
||||
struct iavf_fdir_fltr *fltr = NULL;
|
||||
int err = 0;
|
||||
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
if (!(adapter->flags & IAVF_FLAG_FDIR_ENABLED))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
@ -1472,6 +1476,11 @@ static int iavf_del_fdir_ethtool(struct iavf_adapter *adapter, struct ethtool_rx
|
||||
if (fltr->state == IAVF_FDIR_FLTR_ACTIVE) {
|
||||
fltr->state = IAVF_FDIR_FLTR_DEL_REQUEST;
|
||||
adapter->aq_required |= IAVF_FLAG_AQ_DEL_FDIR_FILTER;
|
||||
} else if (fltr->state == IAVF_FDIR_FLTR_INACTIVE) {
|
||||
list_del(&fltr->list);
|
||||
kfree(fltr);
|
||||
adapter->fdir_active_fltr--;
|
||||
fltr = NULL;
|
||||
} else {
|
||||
err = -EBUSY;
|
||||
}
|
||||
@ -1780,7 +1789,7 @@ static int iavf_get_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *cmd,
|
||||
ret = 0;
|
||||
break;
|
||||
case ETHTOOL_GRXCLSRLCNT:
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
if (!(adapter->flags & IAVF_FLAG_FDIR_ENABLED))
|
||||
break;
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
cmd->rule_cnt = adapter->fdir_active_fltr;
|
||||
|
@ -6,12 +6,25 @@
|
||||
|
||||
struct iavf_adapter;
|
||||
|
||||
/* State of Flow Director filter */
|
||||
/* State of Flow Director filter
|
||||
*
|
||||
* *_REQUEST states are used to mark filter to be sent to PF driver to perform
|
||||
* an action (either add or delete filter). *_PENDING states are an indication
|
||||
* that request was sent to PF and the driver is waiting for response.
|
||||
*
|
||||
* Both DELETE and DISABLE states are being used to delete a filter in PF.
|
||||
* The difference is that after a successful response filter in DEL_PENDING
|
||||
* state is being deleted from VF driver as well and filter in DIS_PENDING state
|
||||
* is being changed to INACTIVE state.
|
||||
*/
|
||||
enum iavf_fdir_fltr_state_t {
|
||||
IAVF_FDIR_FLTR_ADD_REQUEST, /* User requests to add filter */
|
||||
IAVF_FDIR_FLTR_ADD_PENDING, /* Filter pending add by the PF */
|
||||
IAVF_FDIR_FLTR_DEL_REQUEST, /* User requests to delete filter */
|
||||
IAVF_FDIR_FLTR_DEL_PENDING, /* Filter pending delete by the PF */
|
||||
IAVF_FDIR_FLTR_DIS_REQUEST, /* Filter scheduled to be disabled */
|
||||
IAVF_FDIR_FLTR_DIS_PENDING, /* Filter pending disable by the PF */
|
||||
IAVF_FDIR_FLTR_INACTIVE, /* Filter inactive on link down */
|
||||
IAVF_FDIR_FLTR_ACTIVE, /* Filter is active */
|
||||
};
|
||||
|
||||
|
@ -276,27 +276,6 @@ void iavf_free_virt_mem(struct iavf_hw *hw, struct iavf_virt_mem *mem)
|
||||
kfree(mem->va);
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_lock_timeout - try to lock mutex but give up after timeout
|
||||
* @lock: mutex that should be locked
|
||||
* @msecs: timeout in msecs
|
||||
*
|
||||
* Returns 0 on success, negative on failure
|
||||
**/
|
||||
static int iavf_lock_timeout(struct mutex *lock, unsigned int msecs)
|
||||
{
|
||||
unsigned int wait, delay = 10;
|
||||
|
||||
for (wait = 0; wait < msecs; wait += delay) {
|
||||
if (mutex_trylock(lock))
|
||||
return 0;
|
||||
|
||||
msleep(delay);
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_schedule_reset - Set the flags and schedule a reset event
|
||||
* @adapter: board private structure
|
||||
@ -1353,18 +1332,20 @@ static void iavf_clear_cloud_filters(struct iavf_adapter *adapter)
|
||||
**/
|
||||
static void iavf_clear_fdir_filters(struct iavf_adapter *adapter)
|
||||
{
|
||||
struct iavf_fdir_fltr *fdir, *fdirtmp;
|
||||
struct iavf_fdir_fltr *fdir;
|
||||
|
||||
/* remove all Flow Director filters */
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
list_for_each_entry_safe(fdir, fdirtmp, &adapter->fdir_list_head,
|
||||
list) {
|
||||
list_for_each_entry(fdir, &adapter->fdir_list_head, list) {
|
||||
if (fdir->state == IAVF_FDIR_FLTR_ADD_REQUEST) {
|
||||
list_del(&fdir->list);
|
||||
kfree(fdir);
|
||||
adapter->fdir_active_fltr--;
|
||||
} else {
|
||||
fdir->state = IAVF_FDIR_FLTR_DEL_REQUEST;
|
||||
/* Cancel a request, keep filter as inactive */
|
||||
fdir->state = IAVF_FDIR_FLTR_INACTIVE;
|
||||
} else if (fdir->state == IAVF_FDIR_FLTR_ADD_PENDING ||
|
||||
fdir->state == IAVF_FDIR_FLTR_ACTIVE) {
|
||||
/* Disable filters which are active or have a pending
|
||||
* request to PF to be added
|
||||
*/
|
||||
fdir->state = IAVF_FDIR_FLTR_DIS_REQUEST;
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
@ -4112,6 +4093,33 @@ static int iavf_setup_tc(struct net_device *netdev, enum tc_setup_type type,
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_restore_fdir_filters
|
||||
* @adapter: board private structure
|
||||
*
|
||||
* Restore existing FDIR filters when VF netdev comes back up.
|
||||
**/
|
||||
static void iavf_restore_fdir_filters(struct iavf_adapter *adapter)
|
||||
{
|
||||
struct iavf_fdir_fltr *f;
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
list_for_each_entry(f, &adapter->fdir_list_head, list) {
|
||||
if (f->state == IAVF_FDIR_FLTR_DIS_REQUEST) {
|
||||
/* Cancel a request, keep filter as active */
|
||||
f->state = IAVF_FDIR_FLTR_ACTIVE;
|
||||
} else if (f->state == IAVF_FDIR_FLTR_DIS_PENDING ||
|
||||
f->state == IAVF_FDIR_FLTR_INACTIVE) {
|
||||
/* Add filters which are inactive or have a pending
|
||||
* request to PF to be deleted
|
||||
*/
|
||||
f->state = IAVF_FDIR_FLTR_ADD_REQUEST;
|
||||
adapter->aq_required |= IAVF_FLAG_AQ_ADD_FDIR_FILTER;
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_open - Called when a network interface is made active
|
||||
* @netdev: network interface device structure
|
||||
@ -4179,8 +4187,9 @@ static int iavf_open(struct net_device *netdev)
|
||||
|
||||
spin_unlock_bh(&adapter->mac_vlan_list_lock);
|
||||
|
||||
/* Restore VLAN filters that were removed with IFF_DOWN */
|
||||
/* Restore filters that were removed with IFF_DOWN */
|
||||
iavf_restore_filters(adapter);
|
||||
iavf_restore_fdir_filters(adapter);
|
||||
|
||||
iavf_configure(adapter);
|
||||
|
||||
@ -4311,6 +4320,49 @@ static int iavf_change_mtu(struct net_device *netdev, int new_mtu)
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_disable_fdir - disable Flow Director and clear existing filters
|
||||
* @adapter: board private structure
|
||||
**/
|
||||
static void iavf_disable_fdir(struct iavf_adapter *adapter)
|
||||
{
|
||||
struct iavf_fdir_fltr *fdir, *fdirtmp;
|
||||
bool del_filters = false;
|
||||
|
||||
adapter->flags &= ~IAVF_FLAG_FDIR_ENABLED;
|
||||
|
||||
/* remove all Flow Director filters */
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
list_for_each_entry_safe(fdir, fdirtmp, &adapter->fdir_list_head,
|
||||
list) {
|
||||
if (fdir->state == IAVF_FDIR_FLTR_ADD_REQUEST ||
|
||||
fdir->state == IAVF_FDIR_FLTR_INACTIVE) {
|
||||
/* Delete filters not registered in PF */
|
||||
list_del(&fdir->list);
|
||||
kfree(fdir);
|
||||
adapter->fdir_active_fltr--;
|
||||
} else if (fdir->state == IAVF_FDIR_FLTR_ADD_PENDING ||
|
||||
fdir->state == IAVF_FDIR_FLTR_DIS_REQUEST ||
|
||||
fdir->state == IAVF_FDIR_FLTR_ACTIVE) {
|
||||
/* Filters registered in PF, schedule their deletion */
|
||||
fdir->state = IAVF_FDIR_FLTR_DEL_REQUEST;
|
||||
del_filters = true;
|
||||
} else if (fdir->state == IAVF_FDIR_FLTR_DIS_PENDING) {
|
||||
/* Request to delete filter already sent to PF, change
|
||||
* state to DEL_PENDING to delete filter after PF's
|
||||
* response, not set as INACTIVE
|
||||
*/
|
||||
fdir->state = IAVF_FDIR_FLTR_DEL_PENDING;
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
|
||||
if (del_filters) {
|
||||
adapter->aq_required |= IAVF_FLAG_AQ_DEL_FDIR_FILTER;
|
||||
mod_delayed_work(adapter->wq, &adapter->watchdog_task, 0);
|
||||
}
|
||||
}
|
||||
|
||||
#define NETIF_VLAN_OFFLOAD_FEATURES (NETIF_F_HW_VLAN_CTAG_RX | \
|
||||
NETIF_F_HW_VLAN_CTAG_TX | \
|
||||
NETIF_F_HW_VLAN_STAG_RX | \
|
||||
@ -4336,6 +4388,13 @@ static int iavf_set_features(struct net_device *netdev,
|
||||
((netdev->features & NETIF_F_RXFCS) ^ (features & NETIF_F_RXFCS)))
|
||||
iavf_schedule_reset(adapter, IAVF_FLAG_RESET_NEEDED);
|
||||
|
||||
if ((netdev->features & NETIF_F_NTUPLE) ^ (features & NETIF_F_NTUPLE)) {
|
||||
if (features & NETIF_F_NTUPLE)
|
||||
adapter->flags |= IAVF_FLAG_FDIR_ENABLED;
|
||||
else
|
||||
iavf_disable_fdir(adapter);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -4685,6 +4744,9 @@ static netdev_features_t iavf_fix_features(struct net_device *netdev,
|
||||
|
||||
features = iavf_fix_netdev_vlan_features(adapter, features);
|
||||
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
features &= ~NETIF_F_NTUPLE;
|
||||
|
||||
return iavf_fix_strip_features(adapter, features);
|
||||
}
|
||||
|
||||
@ -4802,6 +4864,12 @@ int iavf_process_config(struct iavf_adapter *adapter)
|
||||
if (vfres->vf_cap_flags & VIRTCHNL_VF_OFFLOAD_VLAN)
|
||||
netdev->features |= NETIF_F_HW_VLAN_CTAG_FILTER;
|
||||
|
||||
if (FDIR_FLTR_SUPPORT(adapter)) {
|
||||
netdev->hw_features |= NETIF_F_NTUPLE;
|
||||
netdev->features |= NETIF_F_NTUPLE;
|
||||
adapter->flags |= IAVF_FLAG_FDIR_ENABLED;
|
||||
}
|
||||
|
||||
netdev->priv_flags |= IFF_UNICAST_FLT;
|
||||
|
||||
/* Do not turn on offloads when they are requested to be turned off.
|
||||
@ -4825,34 +4893,6 @@ int iavf_process_config(struct iavf_adapter *adapter)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_shutdown - Shutdown the device in preparation for a reboot
|
||||
* @pdev: pci device structure
|
||||
**/
|
||||
static void iavf_shutdown(struct pci_dev *pdev)
|
||||
{
|
||||
struct iavf_adapter *adapter = iavf_pdev_to_adapter(pdev);
|
||||
struct net_device *netdev = adapter->netdev;
|
||||
|
||||
netif_device_detach(netdev);
|
||||
|
||||
if (netif_running(netdev))
|
||||
iavf_close(netdev);
|
||||
|
||||
if (iavf_lock_timeout(&adapter->crit_lock, 5000))
|
||||
dev_warn(&adapter->pdev->dev, "%s: failed to acquire crit_lock\n", __func__);
|
||||
/* Prevent the watchdog from running. */
|
||||
iavf_change_state(adapter, __IAVF_REMOVE);
|
||||
adapter->aq_required = 0;
|
||||
mutex_unlock(&adapter->crit_lock);
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
pci_save_state(pdev);
|
||||
|
||||
#endif
|
||||
pci_disable_device(pdev);
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_probe - Device Initialization Routine
|
||||
* @pdev: PCI device information struct
|
||||
@ -5063,16 +5103,21 @@ static int __maybe_unused iavf_resume(struct device *dev_d)
|
||||
**/
|
||||
static void iavf_remove(struct pci_dev *pdev)
|
||||
{
|
||||
struct iavf_adapter *adapter = iavf_pdev_to_adapter(pdev);
|
||||
struct iavf_fdir_fltr *fdir, *fdirtmp;
|
||||
struct iavf_vlan_filter *vlf, *vlftmp;
|
||||
struct iavf_cloud_filter *cf, *cftmp;
|
||||
struct iavf_adv_rss *rss, *rsstmp;
|
||||
struct iavf_mac_filter *f, *ftmp;
|
||||
struct iavf_adapter *adapter;
|
||||
struct net_device *netdev;
|
||||
struct iavf_hw *hw;
|
||||
|
||||
netdev = adapter->netdev;
|
||||
/* Don't proceed with remove if netdev is already freed */
|
||||
netdev = pci_get_drvdata(pdev);
|
||||
if (!netdev)
|
||||
return;
|
||||
|
||||
adapter = iavf_pdev_to_adapter(pdev);
|
||||
hw = &adapter->hw;
|
||||
|
||||
if (test_and_set_bit(__IAVF_IN_REMOVE_TASK, &adapter->crit_section))
|
||||
@ -5184,11 +5229,25 @@ static void iavf_remove(struct pci_dev *pdev)
|
||||
|
||||
destroy_workqueue(adapter->wq);
|
||||
|
||||
pci_set_drvdata(pdev, NULL);
|
||||
|
||||
free_netdev(netdev);
|
||||
|
||||
pci_disable_device(pdev);
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_shutdown - Shutdown the device in preparation for a reboot
|
||||
* @pdev: pci device structure
|
||||
**/
|
||||
static void iavf_shutdown(struct pci_dev *pdev)
|
||||
{
|
||||
iavf_remove(pdev);
|
||||
|
||||
if (system_state == SYSTEM_POWER_OFF)
|
||||
pci_set_power_state(pdev, PCI_D3hot);
|
||||
}
|
||||
|
||||
static SIMPLE_DEV_PM_OPS(iavf_pm_ops, iavf_suspend, iavf_resume);
|
||||
|
||||
static struct pci_driver iavf_driver = {
|
||||
|
@ -1735,8 +1735,8 @@ void iavf_add_fdir_filter(struct iavf_adapter *adapter)
|
||||
**/
|
||||
void iavf_del_fdir_filter(struct iavf_adapter *adapter)
|
||||
{
|
||||
struct virtchnl_fdir_del f = {};
|
||||
struct iavf_fdir_fltr *fdir;
|
||||
struct virtchnl_fdir_del f;
|
||||
bool process_fltr = false;
|
||||
int len;
|
||||
|
||||
@ -1753,11 +1753,16 @@ void iavf_del_fdir_filter(struct iavf_adapter *adapter)
|
||||
list_for_each_entry(fdir, &adapter->fdir_list_head, list) {
|
||||
if (fdir->state == IAVF_FDIR_FLTR_DEL_REQUEST) {
|
||||
process_fltr = true;
|
||||
memset(&f, 0, len);
|
||||
f.vsi_id = fdir->vc_add_msg.vsi_id;
|
||||
f.flow_id = fdir->flow_id;
|
||||
fdir->state = IAVF_FDIR_FLTR_DEL_PENDING;
|
||||
break;
|
||||
} else if (fdir->state == IAVF_FDIR_FLTR_DIS_REQUEST) {
|
||||
process_fltr = true;
|
||||
f.vsi_id = fdir->vc_add_msg.vsi_id;
|
||||
f.flow_id = fdir->flow_id;
|
||||
fdir->state = IAVF_FDIR_FLTR_DIS_PENDING;
|
||||
break;
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
@ -1901,6 +1906,48 @@ static void iavf_netdev_features_vlan_strip_set(struct net_device *netdev,
|
||||
netdev->features &= ~NETIF_F_HW_VLAN_CTAG_RX;
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_activate_fdir_filters - Reactivate all FDIR filters after a reset
|
||||
* @adapter: private adapter structure
|
||||
*
|
||||
* Called after a reset to re-add all FDIR filters and delete some of them
|
||||
* if they were pending to be deleted.
|
||||
*/
|
||||
static void iavf_activate_fdir_filters(struct iavf_adapter *adapter)
|
||||
{
|
||||
struct iavf_fdir_fltr *f, *ftmp;
|
||||
bool add_filters = false;
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
list_for_each_entry_safe(f, ftmp, &adapter->fdir_list_head, list) {
|
||||
if (f->state == IAVF_FDIR_FLTR_ADD_REQUEST ||
|
||||
f->state == IAVF_FDIR_FLTR_ADD_PENDING ||
|
||||
f->state == IAVF_FDIR_FLTR_ACTIVE) {
|
||||
/* All filters and requests have been removed in PF,
|
||||
* restore them
|
||||
*/
|
||||
f->state = IAVF_FDIR_FLTR_ADD_REQUEST;
|
||||
add_filters = true;
|
||||
} else if (f->state == IAVF_FDIR_FLTR_DIS_REQUEST ||
|
||||
f->state == IAVF_FDIR_FLTR_DIS_PENDING) {
|
||||
/* Link down state, leave filters as inactive */
|
||||
f->state = IAVF_FDIR_FLTR_INACTIVE;
|
||||
} else if (f->state == IAVF_FDIR_FLTR_DEL_REQUEST ||
|
||||
f->state == IAVF_FDIR_FLTR_DEL_PENDING) {
|
||||
/* Delete filters that were pending to be deleted, the
|
||||
* list on PF is already cleared after a reset
|
||||
*/
|
||||
list_del(&f->list);
|
||||
kfree(f);
|
||||
adapter->fdir_active_fltr--;
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
|
||||
if (add_filters)
|
||||
adapter->aq_required |= IAVF_FLAG_AQ_ADD_FDIR_FILTER;
|
||||
}
|
||||
|
||||
/**
|
||||
* iavf_virtchnl_completion
|
||||
* @adapter: adapter structure
|
||||
@ -2078,7 +2125,8 @@ void iavf_virtchnl_completion(struct iavf_adapter *adapter,
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
list_for_each_entry(fdir, &adapter->fdir_list_head,
|
||||
list) {
|
||||
if (fdir->state == IAVF_FDIR_FLTR_DEL_PENDING) {
|
||||
if (fdir->state == IAVF_FDIR_FLTR_DEL_PENDING ||
|
||||
fdir->state == IAVF_FDIR_FLTR_DIS_PENDING) {
|
||||
fdir->state = IAVF_FDIR_FLTR_ACTIVE;
|
||||
dev_info(&adapter->pdev->dev, "Failed to del Flow Director filter, error %s\n",
|
||||
iavf_stat_str(&adapter->hw,
|
||||
@ -2214,6 +2262,8 @@ void iavf_virtchnl_completion(struct iavf_adapter *adapter,
|
||||
|
||||
spin_unlock_bh(&adapter->mac_vlan_list_lock);
|
||||
|
||||
iavf_activate_fdir_filters(adapter);
|
||||
|
||||
iavf_parse_vf_resource_msg(adapter);
|
||||
|
||||
/* negotiated VIRTCHNL_VF_OFFLOAD_VLAN_V2, so wait for the
|
||||
@ -2390,7 +2440,9 @@ void iavf_virtchnl_completion(struct iavf_adapter *adapter,
|
||||
list_for_each_entry_safe(fdir, fdir_tmp, &adapter->fdir_list_head,
|
||||
list) {
|
||||
if (fdir->state == IAVF_FDIR_FLTR_DEL_PENDING) {
|
||||
if (del_fltr->status == VIRTCHNL_FDIR_SUCCESS) {
|
||||
if (del_fltr->status == VIRTCHNL_FDIR_SUCCESS ||
|
||||
del_fltr->status ==
|
||||
VIRTCHNL_FDIR_FAILURE_RULE_NONEXIST) {
|
||||
dev_info(&adapter->pdev->dev, "Flow Director filter with location %u is deleted\n",
|
||||
fdir->loc);
|
||||
list_del(&fdir->list);
|
||||
@ -2402,6 +2454,17 @@ void iavf_virtchnl_completion(struct iavf_adapter *adapter,
|
||||
del_fltr->status);
|
||||
iavf_print_fdir_fltr(adapter, fdir);
|
||||
}
|
||||
} else if (fdir->state == IAVF_FDIR_FLTR_DIS_PENDING) {
|
||||
if (del_fltr->status == VIRTCHNL_FDIR_SUCCESS ||
|
||||
del_fltr->status ==
|
||||
VIRTCHNL_FDIR_FAILURE_RULE_NONEXIST) {
|
||||
fdir->state = IAVF_FDIR_FLTR_INACTIVE;
|
||||
} else {
|
||||
fdir->state = IAVF_FDIR_FLTR_ACTIVE;
|
||||
dev_info(&adapter->pdev->dev, "Failed to disable Flow Director filter with status: %d\n",
|
||||
del_fltr->status);
|
||||
iavf_print_fdir_fltr(adapter, fdir);
|
||||
}
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
|
@ -1193,6 +1193,13 @@ int octep_device_setup(struct octep_device *oct)
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
INIT_WORK(&oct->tx_timeout_task, octep_tx_timeout_task);
|
||||
INIT_WORK(&oct->ctrl_mbox_task, octep_ctrl_mbox_task);
|
||||
INIT_DELAYED_WORK(&oct->intr_poll_task, octep_intr_poll_task);
|
||||
oct->poll_non_ioq_intr = true;
|
||||
queue_delayed_work(octep_wq, &oct->intr_poll_task,
|
||||
msecs_to_jiffies(OCTEP_INTR_POLL_TIME_MSECS));
|
||||
|
||||
atomic_set(&oct->hb_miss_cnt, 0);
|
||||
INIT_DELAYED_WORK(&oct->hb_task, octep_hb_timeout_task);
|
||||
|
||||
@ -1258,7 +1265,8 @@ static bool get_fw_ready_status(struct pci_dev *pdev)
|
||||
|
||||
pci_read_config_byte(pdev, (pos + 8), &status);
|
||||
dev_info(&pdev->dev, "Firmware ready status = %u\n", status);
|
||||
return status;
|
||||
#define FW_STATUS_READY 1ULL
|
||||
return status == FW_STATUS_READY;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
@ -1325,21 +1333,18 @@ static int octep_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
goto err_octep_config;
|
||||
}
|
||||
|
||||
octep_ctrl_net_get_info(octep_dev, OCTEP_CTRL_NET_INVALID_VFID,
|
||||
&octep_dev->conf->fw_info);
|
||||
err = octep_ctrl_net_get_info(octep_dev, OCTEP_CTRL_NET_INVALID_VFID,
|
||||
&octep_dev->conf->fw_info);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to get firmware info\n");
|
||||
goto register_dev_err;
|
||||
}
|
||||
dev_info(&octep_dev->pdev->dev, "Heartbeat interval %u msecs Heartbeat miss count %u\n",
|
||||
octep_dev->conf->fw_info.hb_interval,
|
||||
octep_dev->conf->fw_info.hb_miss_count);
|
||||
queue_delayed_work(octep_wq, &octep_dev->hb_task,
|
||||
msecs_to_jiffies(octep_dev->conf->fw_info.hb_interval));
|
||||
|
||||
INIT_WORK(&octep_dev->tx_timeout_task, octep_tx_timeout_task);
|
||||
INIT_WORK(&octep_dev->ctrl_mbox_task, octep_ctrl_mbox_task);
|
||||
INIT_DELAYED_WORK(&octep_dev->intr_poll_task, octep_intr_poll_task);
|
||||
octep_dev->poll_non_ioq_intr = true;
|
||||
queue_delayed_work(octep_wq, &octep_dev->intr_poll_task,
|
||||
msecs_to_jiffies(OCTEP_INTR_POLL_TIME_MSECS));
|
||||
|
||||
netdev->netdev_ops = &octep_netdev_ops;
|
||||
octep_set_ethtool_ops(netdev);
|
||||
netif_carrier_off(netdev);
|
||||
|
@ -373,6 +373,11 @@ void rpm_lmac_pause_frm_config(void *rpmd, int lmac_id, bool enable)
|
||||
cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE;
|
||||
rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
|
||||
|
||||
/* Disable forward pause to driver */
|
||||
cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
|
||||
cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_FWD;
|
||||
rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
|
||||
|
||||
/* Enable channel mask for all LMACS */
|
||||
if (is_dev_rpm2(rpm))
|
||||
rpm_write(rpm, lmac_id, RPM2_CMR_CHAN_MSK_OR, 0xffff);
|
||||
@ -616,12 +621,10 @@ int rpm_lmac_pfc_config(void *rpmd, int lmac_id, u8 tx_pause, u8 rx_pause, u16 p
|
||||
|
||||
if (rx_pause) {
|
||||
cfg &= ~(RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE |
|
||||
RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE |
|
||||
RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_FWD);
|
||||
RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE);
|
||||
} else {
|
||||
cfg |= (RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE |
|
||||
RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE |
|
||||
RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_FWD);
|
||||
RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE);
|
||||
}
|
||||
|
||||
if (tx_pause) {
|
||||
|
@ -538,7 +538,7 @@ static int rvu_nix_register_reporters(struct rvu_devlink *rvu_dl)
|
||||
|
||||
rvu_dl->devlink_wq = create_workqueue("rvu_devlink_wq");
|
||||
if (!rvu_dl->devlink_wq)
|
||||
goto err;
|
||||
return -ENOMEM;
|
||||
|
||||
INIT_WORK(&rvu_reporters->intr_work, rvu_nix_intr_work);
|
||||
INIT_WORK(&rvu_reporters->gen_work, rvu_nix_gen_work);
|
||||
@ -546,9 +546,6 @@ static int rvu_nix_register_reporters(struct rvu_devlink *rvu_dl)
|
||||
INIT_WORK(&rvu_reporters->ras_work, rvu_nix_ras_work);
|
||||
|
||||
return 0;
|
||||
err:
|
||||
rvu_nix_health_reporters_destroy(rvu_dl);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
static int rvu_nix_health_reporters_create(struct rvu_devlink *rvu_dl)
|
||||
|
@ -671,6 +671,7 @@ void rvu_npc_install_promisc_entry(struct rvu *rvu, u16 pcifunc,
|
||||
int blkaddr, ucast_idx, index;
|
||||
struct nix_rx_action action = { 0 };
|
||||
u64 relaxed_mask;
|
||||
u8 flow_key_alg;
|
||||
|
||||
if (!hw->cap.nix_rx_multicast && is_cgx_vf(rvu, pcifunc))
|
||||
return;
|
||||
@ -701,6 +702,8 @@ void rvu_npc_install_promisc_entry(struct rvu *rvu, u16 pcifunc,
|
||||
action.op = NIX_RX_ACTIONOP_UCAST;
|
||||
}
|
||||
|
||||
flow_key_alg = action.flow_key_alg;
|
||||
|
||||
/* RX_ACTION set to MCAST for CGX PF's */
|
||||
if (hw->cap.nix_rx_multicast && pfvf->use_mce_list &&
|
||||
is_pf_cgxmapped(rvu, rvu_get_pf(pcifunc))) {
|
||||
@ -740,7 +743,7 @@ void rvu_npc_install_promisc_entry(struct rvu *rvu, u16 pcifunc,
|
||||
req.vf = pcifunc;
|
||||
req.index = action.index;
|
||||
req.match_id = action.match_id;
|
||||
req.flow_key_alg = action.flow_key_alg;
|
||||
req.flow_key_alg = flow_key_alg;
|
||||
|
||||
rvu_mbox_handler_npc_install_flow(rvu, &req, &rsp);
|
||||
}
|
||||
@ -854,6 +857,7 @@ void rvu_npc_install_allmulti_entry(struct rvu *rvu, u16 pcifunc, int nixlf,
|
||||
u8 mac_addr[ETH_ALEN] = { 0 };
|
||||
struct nix_rx_action action = { 0 };
|
||||
struct rvu_pfvf *pfvf;
|
||||
u8 flow_key_alg;
|
||||
u16 vf_func;
|
||||
|
||||
/* Only CGX PF/VF can add allmulticast entry */
|
||||
@ -888,6 +892,7 @@ void rvu_npc_install_allmulti_entry(struct rvu *rvu, u16 pcifunc, int nixlf,
|
||||
*(u64 *)&action = npc_get_mcam_action(rvu, mcam,
|
||||
blkaddr, ucast_idx);
|
||||
|
||||
flow_key_alg = action.flow_key_alg;
|
||||
if (action.op != NIX_RX_ACTIONOP_RSS) {
|
||||
*(u64 *)&action = 0;
|
||||
action.op = NIX_RX_ACTIONOP_UCAST;
|
||||
@ -924,7 +929,7 @@ void rvu_npc_install_allmulti_entry(struct rvu *rvu, u16 pcifunc, int nixlf,
|
||||
req.vf = pcifunc | vf_func;
|
||||
req.index = action.index;
|
||||
req.match_id = action.match_id;
|
||||
req.flow_key_alg = action.flow_key_alg;
|
||||
req.flow_key_alg = flow_key_alg;
|
||||
|
||||
rvu_mbox_handler_npc_install_flow(rvu, &req, &rsp);
|
||||
}
|
||||
@ -990,11 +995,38 @@ static void npc_update_vf_flow_entry(struct rvu *rvu, struct npc_mcam *mcam,
|
||||
mutex_unlock(&mcam->lock);
|
||||
}
|
||||
|
||||
static void npc_update_rx_action_with_alg_idx(struct rvu *rvu, struct nix_rx_action action,
|
||||
struct rvu_pfvf *pfvf, int mcam_index, int blkaddr,
|
||||
int alg_idx)
|
||||
|
||||
{
|
||||
struct npc_mcam *mcam = &rvu->hw->mcam;
|
||||
struct rvu_hwinfo *hw = rvu->hw;
|
||||
int bank, op_rss;
|
||||
|
||||
if (!is_mcam_entry_enabled(rvu, mcam, blkaddr, mcam_index))
|
||||
return;
|
||||
|
||||
op_rss = (!hw->cap.nix_rx_multicast || !pfvf->use_mce_list);
|
||||
|
||||
bank = npc_get_bank(mcam, mcam_index);
|
||||
mcam_index &= (mcam->banksize - 1);
|
||||
|
||||
/* If Rx action is MCAST update only RSS algorithm index */
|
||||
if (!op_rss) {
|
||||
*(u64 *)&action = rvu_read64(rvu, blkaddr,
|
||||
NPC_AF_MCAMEX_BANKX_ACTION(mcam_index, bank));
|
||||
|
||||
action.flow_key_alg = alg_idx;
|
||||
}
|
||||
rvu_write64(rvu, blkaddr,
|
||||
NPC_AF_MCAMEX_BANKX_ACTION(mcam_index, bank), *(u64 *)&action);
|
||||
}
|
||||
|
||||
void rvu_npc_update_flowkey_alg_idx(struct rvu *rvu, u16 pcifunc, int nixlf,
|
||||
int group, int alg_idx, int mcam_index)
|
||||
{
|
||||
struct npc_mcam *mcam = &rvu->hw->mcam;
|
||||
struct rvu_hwinfo *hw = rvu->hw;
|
||||
struct nix_rx_action action;
|
||||
int blkaddr, index, bank;
|
||||
struct rvu_pfvf *pfvf;
|
||||
@ -1050,15 +1082,16 @@ void rvu_npc_update_flowkey_alg_idx(struct rvu *rvu, u16 pcifunc, int nixlf,
|
||||
/* If PF's promiscuous entry is enabled,
|
||||
* Set RSS action for that entry as well
|
||||
*/
|
||||
if ((!hw->cap.nix_rx_multicast || !pfvf->use_mce_list) &&
|
||||
is_mcam_entry_enabled(rvu, mcam, blkaddr, index)) {
|
||||
bank = npc_get_bank(mcam, index);
|
||||
index &= (mcam->banksize - 1);
|
||||
npc_update_rx_action_with_alg_idx(rvu, action, pfvf, index, blkaddr,
|
||||
alg_idx);
|
||||
|
||||
rvu_write64(rvu, blkaddr,
|
||||
NPC_AF_MCAMEX_BANKX_ACTION(index, bank),
|
||||
*(u64 *)&action);
|
||||
}
|
||||
index = npc_get_nixlf_mcam_index(mcam, pcifunc,
|
||||
nixlf, NIXLF_ALLMULTI_ENTRY);
|
||||
/* If PF's allmulti entry is enabled,
|
||||
* Set RSS action for that entry as well
|
||||
*/
|
||||
npc_update_rx_action_with_alg_idx(rvu, action, pfvf, index, blkaddr,
|
||||
alg_idx);
|
||||
}
|
||||
|
||||
void npc_enadis_default_mce_entry(struct rvu *rvu, u16 pcifunc,
|
||||
|
@ -1650,6 +1650,21 @@ static void otx2_free_hw_resources(struct otx2_nic *pf)
|
||||
mutex_unlock(&mbox->lock);
|
||||
}
|
||||
|
||||
static bool otx2_promisc_use_mce_list(struct otx2_nic *pfvf)
|
||||
{
|
||||
int vf;
|
||||
|
||||
/* The AF driver will determine whether to allow the VF netdev or not */
|
||||
if (is_otx2_vf(pfvf->pcifunc))
|
||||
return true;
|
||||
|
||||
/* check if there are any trusted VFs associated with the PF netdev */
|
||||
for (vf = 0; vf < pci_num_vf(pfvf->pdev); vf++)
|
||||
if (pfvf->vf_configs[vf].trusted)
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
static void otx2_do_set_rx_mode(struct otx2_nic *pf)
|
||||
{
|
||||
struct net_device *netdev = pf->netdev;
|
||||
@ -1682,7 +1697,8 @@ static void otx2_do_set_rx_mode(struct otx2_nic *pf)
|
||||
if (netdev->flags & (IFF_ALLMULTI | IFF_MULTICAST))
|
||||
req->mode |= NIX_RX_MODE_ALLMULTI;
|
||||
|
||||
req->mode |= NIX_RX_MODE_USE_MCE;
|
||||
if (otx2_promisc_use_mce_list(pf))
|
||||
req->mode |= NIX_RX_MODE_USE_MCE;
|
||||
|
||||
otx2_sync_mbox_msg(&pf->mbox);
|
||||
mutex_unlock(&pf->mbox.lock);
|
||||
@ -2691,11 +2707,14 @@ static int otx2_ndo_set_vf_trust(struct net_device *netdev, int vf,
|
||||
pf->vf_configs[vf].trusted = enable;
|
||||
rc = otx2_set_vf_permissions(pf, vf, OTX2_TRUSTED_VF);
|
||||
|
||||
if (rc)
|
||||
if (rc) {
|
||||
pf->vf_configs[vf].trusted = !enable;
|
||||
else
|
||||
} else {
|
||||
netdev_info(pf->netdev, "VF %d is %strusted\n",
|
||||
vf, enable ? "" : "not ");
|
||||
otx2_set_rx_mode(netdev);
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
|
@ -826,6 +826,7 @@ enum {
|
||||
MLX5E_STATE_DESTROYING,
|
||||
MLX5E_STATE_XDP_TX_ENABLED,
|
||||
MLX5E_STATE_XDP_ACTIVE,
|
||||
MLX5E_STATE_CHANNELS_ACTIVE,
|
||||
};
|
||||
|
||||
struct mlx5e_modify_sq_param {
|
||||
|
@ -83,6 +83,9 @@ mlx5e_tc_post_act_offload(struct mlx5e_post_act *post_act,
|
||||
struct mlx5_flow_spec *spec;
|
||||
int err;
|
||||
|
||||
if (IS_ERR(post_act))
|
||||
return PTR_ERR(post_act);
|
||||
|
||||
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||
if (!spec)
|
||||
return -ENOMEM;
|
||||
@ -111,6 +114,9 @@ mlx5e_tc_post_act_add(struct mlx5e_post_act *post_act, struct mlx5_flow_attr *po
|
||||
struct mlx5e_post_act_handle *handle;
|
||||
int err;
|
||||
|
||||
if (IS_ERR(post_act))
|
||||
return ERR_CAST(post_act);
|
||||
|
||||
handle = kzalloc(sizeof(*handle), GFP_KERNEL);
|
||||
if (!handle)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
@ -121,7 +121,14 @@ static bool mlx5e_ipsec_update_esn_state(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
if (x->xso.type == XFRM_DEV_OFFLOAD_CRYPTO)
|
||||
esn_msb = xfrm_replay_seqhi(x, htonl(seq_bottom));
|
||||
|
||||
sa_entry->esn_state.esn = esn;
|
||||
if (sa_entry->esn_state.esn_msb)
|
||||
sa_entry->esn_state.esn = esn;
|
||||
else
|
||||
/* According to RFC4303, section "3.3.3. Sequence Number Generation",
|
||||
* the first packet sent using a given SA will contain a sequence
|
||||
* number of 1.
|
||||
*/
|
||||
sa_entry->esn_state.esn = max_t(u32, esn, 1);
|
||||
sa_entry->esn_state.esn_msb = esn_msb;
|
||||
|
||||
if (unlikely(overlap && seq_bottom < MLX5E_IPSEC_ESN_SCOPE_MID)) {
|
||||
@ -335,6 +342,27 @@ void mlx5e_ipsec_build_accel_xfrm_attrs(struct mlx5e_ipsec_sa_entry *sa_entry,
|
||||
attrs->replay_esn.esn = sa_entry->esn_state.esn;
|
||||
attrs->replay_esn.esn_msb = sa_entry->esn_state.esn_msb;
|
||||
attrs->replay_esn.overlap = sa_entry->esn_state.overlap;
|
||||
switch (x->replay_esn->replay_window) {
|
||||
case 32:
|
||||
attrs->replay_esn.replay_window =
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_32BIT;
|
||||
break;
|
||||
case 64:
|
||||
attrs->replay_esn.replay_window =
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_64BIT;
|
||||
break;
|
||||
case 128:
|
||||
attrs->replay_esn.replay_window =
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_128BIT;
|
||||
break;
|
||||
case 256:
|
||||
attrs->replay_esn.replay_window =
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_256BIT;
|
||||
break;
|
||||
default:
|
||||
WARN_ON(true);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
attrs->dir = x->xso.dir;
|
||||
@ -907,9 +935,11 @@ void mlx5e_ipsec_cleanup(struct mlx5e_priv *priv)
|
||||
return;
|
||||
|
||||
mlx5e_accel_ipsec_fs_cleanup(ipsec);
|
||||
if (mlx5_ipsec_device_caps(priv->mdev) & MLX5_IPSEC_CAP_TUNNEL)
|
||||
if (ipsec->netevent_nb.notifier_call) {
|
||||
unregister_netevent_notifier(&ipsec->netevent_nb);
|
||||
if (mlx5_ipsec_device_caps(priv->mdev) & MLX5_IPSEC_CAP_PACKET_OFFLOAD)
|
||||
ipsec->netevent_nb.notifier_call = NULL;
|
||||
}
|
||||
if (ipsec->aso)
|
||||
mlx5e_ipsec_aso_cleanup(ipsec);
|
||||
destroy_workqueue(ipsec->wq);
|
||||
kfree(ipsec);
|
||||
@ -1018,6 +1048,12 @@ static int mlx5e_xfrm_validate_policy(struct mlx5_core_dev *mdev,
|
||||
}
|
||||
}
|
||||
|
||||
if (x->xdo.type == XFRM_DEV_OFFLOAD_PACKET &&
|
||||
!(mlx5_ipsec_device_caps(mdev) & MLX5_IPSEC_CAP_PACKET_OFFLOAD)) {
|
||||
NL_SET_ERR_MSG_MOD(extack, "Packet offload is not supported");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -1113,14 +1149,6 @@ static const struct xfrmdev_ops mlx5e_ipsec_xfrmdev_ops = {
|
||||
.xdo_dev_state_free = mlx5e_xfrm_free_state,
|
||||
.xdo_dev_offload_ok = mlx5e_ipsec_offload_ok,
|
||||
.xdo_dev_state_advance_esn = mlx5e_xfrm_advance_esn_state,
|
||||
};
|
||||
|
||||
static const struct xfrmdev_ops mlx5e_ipsec_packet_xfrmdev_ops = {
|
||||
.xdo_dev_state_add = mlx5e_xfrm_add_state,
|
||||
.xdo_dev_state_delete = mlx5e_xfrm_del_state,
|
||||
.xdo_dev_state_free = mlx5e_xfrm_free_state,
|
||||
.xdo_dev_offload_ok = mlx5e_ipsec_offload_ok,
|
||||
.xdo_dev_state_advance_esn = mlx5e_xfrm_advance_esn_state,
|
||||
|
||||
.xdo_dev_state_update_curlft = mlx5e_xfrm_update_curlft,
|
||||
.xdo_dev_policy_add = mlx5e_xfrm_add_policy,
|
||||
@ -1138,11 +1166,7 @@ void mlx5e_ipsec_build_netdev(struct mlx5e_priv *priv)
|
||||
|
||||
mlx5_core_info(mdev, "mlx5e: IPSec ESP acceleration enabled\n");
|
||||
|
||||
if (mlx5_ipsec_device_caps(mdev) & MLX5_IPSEC_CAP_PACKET_OFFLOAD)
|
||||
netdev->xfrmdev_ops = &mlx5e_ipsec_packet_xfrmdev_ops;
|
||||
else
|
||||
netdev->xfrmdev_ops = &mlx5e_ipsec_xfrmdev_ops;
|
||||
|
||||
netdev->xfrmdev_ops = &mlx5e_ipsec_xfrmdev_ops;
|
||||
netdev->features |= NETIF_F_HW_ESP;
|
||||
netdev->hw_enc_features |= NETIF_F_HW_ESP;
|
||||
|
||||
|
@ -189,11 +189,19 @@ struct mlx5e_ipsec_ft {
|
||||
u32 refcnt;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_drop {
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_fc *fc;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_rule {
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_modify_hdr *modify_hdr;
|
||||
struct mlx5_pkt_reformat *pkt_reformat;
|
||||
struct mlx5_fc *fc;
|
||||
struct mlx5e_ipsec_drop replay;
|
||||
struct mlx5e_ipsec_drop auth;
|
||||
struct mlx5e_ipsec_drop trailer;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_miss {
|
||||
@ -201,19 +209,6 @@ struct mlx5e_ipsec_miss {
|
||||
struct mlx5_flow_handle *rule;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_rx {
|
||||
struct mlx5e_ipsec_ft ft;
|
||||
struct mlx5e_ipsec_miss pol;
|
||||
struct mlx5e_ipsec_miss sa;
|
||||
struct mlx5e_ipsec_rule status;
|
||||
struct mlx5e_ipsec_miss status_drop;
|
||||
struct mlx5_fc *status_drop_cnt;
|
||||
struct mlx5e_ipsec_fc *fc;
|
||||
struct mlx5_fs_chains *chains;
|
||||
u8 allow_tunnel_mode : 1;
|
||||
struct xarray ipsec_obj_id_map;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_tx_create_attr {
|
||||
int prio;
|
||||
int pol_level;
|
||||
@ -248,6 +243,7 @@ struct mlx5e_ipsec {
|
||||
struct mlx5_ipsec_fs *roce;
|
||||
u8 is_uplink_rep: 1;
|
||||
struct mlx5e_ipsec_mpv_work mpv_work;
|
||||
struct xarray ipsec_obj_id_map;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_esn_state {
|
||||
|
@ -32,6 +32,22 @@ struct mlx5e_ipsec_tx {
|
||||
u8 allow_tunnel_mode : 1;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_status_checks {
|
||||
struct mlx5_flow_group *drop_all_group;
|
||||
struct mlx5e_ipsec_drop all;
|
||||
};
|
||||
|
||||
struct mlx5e_ipsec_rx {
|
||||
struct mlx5e_ipsec_ft ft;
|
||||
struct mlx5e_ipsec_miss pol;
|
||||
struct mlx5e_ipsec_miss sa;
|
||||
struct mlx5e_ipsec_rule status;
|
||||
struct mlx5e_ipsec_status_checks status_drops;
|
||||
struct mlx5e_ipsec_fc *fc;
|
||||
struct mlx5_fs_chains *chains;
|
||||
u8 allow_tunnel_mode : 1;
|
||||
};
|
||||
|
||||
/* IPsec RX flow steering */
|
||||
static enum mlx5_traffic_types family2tt(u32 family)
|
||||
{
|
||||
@ -128,14 +144,37 @@ static struct mlx5_flow_table *ipsec_ft_create(struct mlx5_flow_namespace *ns,
|
||||
return mlx5_create_auto_grouped_flow_table(ns, &ft_attr);
|
||||
}
|
||||
|
||||
static int ipsec_status_rule(struct mlx5_core_dev *mdev,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest)
|
||||
static void ipsec_rx_status_drop_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
u8 action[MLX5_UN_SZ_BYTES(set_add_copy_action_in_auto)] = {};
|
||||
mlx5_del_flow_rules(rx->status_drops.all.rule);
|
||||
mlx5_fc_destroy(ipsec->mdev, rx->status_drops.all.fc);
|
||||
mlx5_destroy_flow_group(rx->status_drops.drop_all_group);
|
||||
}
|
||||
|
||||
static void ipsec_rx_status_pass_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
mlx5_del_flow_rules(rx->status.rule);
|
||||
|
||||
if (rx != ipsec->rx_esw)
|
||||
return;
|
||||
|
||||
#ifdef CONFIG_MLX5_ESWITCH
|
||||
mlx5_chains_put_table(esw_chains(ipsec->mdev->priv.eswitch), 0, 1, 0);
|
||||
#endif
|
||||
}
|
||||
|
||||
static int rx_add_rule_drop_auth_trailer(struct mlx5e_ipsec_sa_entry *sa_entry,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
struct mlx5e_ipsec *ipsec = sa_entry->ipsec;
|
||||
struct mlx5_flow_table *ft = rx->ft.status;
|
||||
struct mlx5_core_dev *mdev = ipsec->mdev;
|
||||
struct mlx5_flow_destination dest = {};
|
||||
struct mlx5_flow_act flow_act = {};
|
||||
struct mlx5_modify_hdr *modify_hdr;
|
||||
struct mlx5_flow_handle *fte;
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_fc *flow_counter;
|
||||
struct mlx5_flow_spec *spec;
|
||||
int err;
|
||||
|
||||
@ -143,48 +182,273 @@ static int ipsec_status_rule(struct mlx5_core_dev *mdev,
|
||||
if (!spec)
|
||||
return -ENOMEM;
|
||||
|
||||
/* Action to copy 7 bit ipsec_syndrome to regB[24:30] */
|
||||
MLX5_SET(copy_action_in, action, action_type, MLX5_ACTION_TYPE_COPY);
|
||||
MLX5_SET(copy_action_in, action, src_field, MLX5_ACTION_IN_FIELD_IPSEC_SYNDROME);
|
||||
MLX5_SET(copy_action_in, action, src_offset, 0);
|
||||
MLX5_SET(copy_action_in, action, length, 7);
|
||||
MLX5_SET(copy_action_in, action, dst_field, MLX5_ACTION_IN_FIELD_METADATA_REG_B);
|
||||
MLX5_SET(copy_action_in, action, dst_offset, 24);
|
||||
|
||||
modify_hdr = mlx5_modify_header_alloc(mdev, MLX5_FLOW_NAMESPACE_KERNEL,
|
||||
1, action);
|
||||
|
||||
if (IS_ERR(modify_hdr)) {
|
||||
err = PTR_ERR(modify_hdr);
|
||||
flow_counter = mlx5_fc_create(mdev, true);
|
||||
if (IS_ERR(flow_counter)) {
|
||||
err = PTR_ERR(flow_counter);
|
||||
mlx5_core_err(mdev,
|
||||
"fail to alloc ipsec copy modify_header_id err=%d\n", err);
|
||||
goto out_spec;
|
||||
"Failed to add ipsec rx status drop rule counter, err=%d\n", err);
|
||||
goto err_cnt;
|
||||
}
|
||||
sa_entry->ipsec_rule.auth.fc = flow_counter;
|
||||
|
||||
/* create fte */
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_MOD_HDR |
|
||||
MLX5_FLOW_CONTEXT_ACTION_FWD_DEST |
|
||||
MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
flow_act.modify_hdr = modify_hdr;
|
||||
fte = mlx5_add_flow_rules(rx->ft.status, spec, &flow_act, dest, 2);
|
||||
if (IS_ERR(fte)) {
|
||||
err = PTR_ERR(fte);
|
||||
mlx5_core_err(mdev, "fail to add ipsec rx err copy rule err=%d\n", err);
|
||||
goto out;
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_DROP | MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
flow_act.flags = FLOW_ACT_NO_APPEND;
|
||||
dest.type = MLX5_FLOW_DESTINATION_TYPE_COUNTER;
|
||||
dest.counter_id = mlx5_fc_id(flow_counter);
|
||||
if (rx == ipsec->rx_esw)
|
||||
spec->flow_context.flow_source = MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria, misc_parameters_2.ipsec_syndrome);
|
||||
MLX5_SET(fte_match_param, spec->match_value, misc_parameters_2.ipsec_syndrome, 1);
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria, misc_parameters_2.metadata_reg_c_2);
|
||||
MLX5_SET(fte_match_param, spec->match_value,
|
||||
misc_parameters_2.metadata_reg_c_2,
|
||||
sa_entry->ipsec_obj_id | BIT(31));
|
||||
spec->match_criteria_enable = MLX5_MATCH_MISC_PARAMETERS_2;
|
||||
rule = mlx5_add_flow_rules(ft, spec, &flow_act, &dest, 1);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule, err=%d\n", err);
|
||||
goto err_rule;
|
||||
}
|
||||
sa_entry->ipsec_rule.auth.rule = rule;
|
||||
|
||||
flow_counter = mlx5_fc_create(mdev, true);
|
||||
if (IS_ERR(flow_counter)) {
|
||||
err = PTR_ERR(flow_counter);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule counter, err=%d\n", err);
|
||||
goto err_cnt_2;
|
||||
}
|
||||
sa_entry->ipsec_rule.trailer.fc = flow_counter;
|
||||
|
||||
dest.counter_id = mlx5_fc_id(flow_counter);
|
||||
MLX5_SET(fte_match_param, spec->match_value, misc_parameters_2.ipsec_syndrome, 2);
|
||||
rule = mlx5_add_flow_rules(ft, spec, &flow_act, &dest, 1);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule, err=%d\n", err);
|
||||
goto err_rule_2;
|
||||
}
|
||||
sa_entry->ipsec_rule.trailer.rule = rule;
|
||||
|
||||
kvfree(spec);
|
||||
rx->status.rule = fte;
|
||||
rx->status.modify_hdr = modify_hdr;
|
||||
return 0;
|
||||
|
||||
out:
|
||||
mlx5_modify_header_dealloc(mdev, modify_hdr);
|
||||
out_spec:
|
||||
err_rule_2:
|
||||
mlx5_fc_destroy(mdev, sa_entry->ipsec_rule.trailer.fc);
|
||||
err_cnt_2:
|
||||
mlx5_del_flow_rules(sa_entry->ipsec_rule.auth.rule);
|
||||
err_rule:
|
||||
mlx5_fc_destroy(mdev, sa_entry->ipsec_rule.auth.fc);
|
||||
err_cnt:
|
||||
kvfree(spec);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int rx_add_rule_drop_replay(struct mlx5e_ipsec_sa_entry *sa_entry, struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
struct mlx5e_ipsec *ipsec = sa_entry->ipsec;
|
||||
struct mlx5_flow_table *ft = rx->ft.status;
|
||||
struct mlx5_core_dev *mdev = ipsec->mdev;
|
||||
struct mlx5_flow_destination dest = {};
|
||||
struct mlx5_flow_act flow_act = {};
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_fc *flow_counter;
|
||||
struct mlx5_flow_spec *spec;
|
||||
int err;
|
||||
|
||||
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||
if (!spec)
|
||||
return -ENOMEM;
|
||||
|
||||
flow_counter = mlx5_fc_create(mdev, true);
|
||||
if (IS_ERR(flow_counter)) {
|
||||
err = PTR_ERR(flow_counter);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule counter, err=%d\n", err);
|
||||
goto err_cnt;
|
||||
}
|
||||
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_DROP | MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
flow_act.flags = FLOW_ACT_NO_APPEND;
|
||||
dest.type = MLX5_FLOW_DESTINATION_TYPE_COUNTER;
|
||||
dest.counter_id = mlx5_fc_id(flow_counter);
|
||||
if (rx == ipsec->rx_esw)
|
||||
spec->flow_context.flow_source = MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria, misc_parameters_2.metadata_reg_c_4);
|
||||
MLX5_SET(fte_match_param, spec->match_value, misc_parameters_2.metadata_reg_c_4, 1);
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria, misc_parameters_2.metadata_reg_c_2);
|
||||
MLX5_SET(fte_match_param, spec->match_value, misc_parameters_2.metadata_reg_c_2,
|
||||
sa_entry->ipsec_obj_id | BIT(31));
|
||||
spec->match_criteria_enable = MLX5_MATCH_MISC_PARAMETERS_2;
|
||||
rule = mlx5_add_flow_rules(ft, spec, &flow_act, &dest, 1);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule, err=%d\n", err);
|
||||
goto err_rule;
|
||||
}
|
||||
|
||||
sa_entry->ipsec_rule.replay.rule = rule;
|
||||
sa_entry->ipsec_rule.replay.fc = flow_counter;
|
||||
|
||||
kvfree(spec);
|
||||
return 0;
|
||||
|
||||
err_rule:
|
||||
mlx5_fc_destroy(mdev, flow_counter);
|
||||
err_cnt:
|
||||
kvfree(spec);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ipsec_rx_status_drop_all_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
|
||||
struct mlx5_flow_table *ft = rx->ft.status;
|
||||
struct mlx5_core_dev *mdev = ipsec->mdev;
|
||||
struct mlx5_flow_destination dest = {};
|
||||
struct mlx5_flow_act flow_act = {};
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_fc *flow_counter;
|
||||
struct mlx5_flow_spec *spec;
|
||||
struct mlx5_flow_group *g;
|
||||
u32 *flow_group_in;
|
||||
int err = 0;
|
||||
|
||||
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||
if (!flow_group_in || !spec) {
|
||||
err = -ENOMEM;
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
MLX5_SET(create_flow_group_in, flow_group_in, start_flow_index, ft->max_fte - 1);
|
||||
MLX5_SET(create_flow_group_in, flow_group_in, end_flow_index, ft->max_fte - 1);
|
||||
g = mlx5_create_flow_group(ft, flow_group_in);
|
||||
if (IS_ERR(g)) {
|
||||
err = PTR_ERR(g);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop flow group, err=%d\n", err);
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
flow_counter = mlx5_fc_create(mdev, false);
|
||||
if (IS_ERR(flow_counter)) {
|
||||
err = PTR_ERR(flow_counter);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule counter, err=%d\n", err);
|
||||
goto err_cnt;
|
||||
}
|
||||
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_DROP | MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
dest.type = MLX5_FLOW_DESTINATION_TYPE_COUNTER;
|
||||
dest.counter_id = mlx5_fc_id(flow_counter);
|
||||
if (rx == ipsec->rx_esw)
|
||||
spec->flow_context.flow_source = MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||
rule = mlx5_add_flow_rules(ft, spec, &flow_act, &dest, 1);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule, err=%d\n", err);
|
||||
goto err_rule;
|
||||
}
|
||||
|
||||
rx->status_drops.drop_all_group = g;
|
||||
rx->status_drops.all.rule = rule;
|
||||
rx->status_drops.all.fc = flow_counter;
|
||||
|
||||
kvfree(flow_group_in);
|
||||
kvfree(spec);
|
||||
return 0;
|
||||
|
||||
err_rule:
|
||||
mlx5_fc_destroy(mdev, flow_counter);
|
||||
err_cnt:
|
||||
mlx5_destroy_flow_group(g);
|
||||
err_out:
|
||||
kvfree(flow_group_in);
|
||||
kvfree(spec);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ipsec_rx_status_pass_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest)
|
||||
{
|
||||
struct mlx5_flow_act flow_act = {};
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_flow_spec *spec;
|
||||
int err;
|
||||
|
||||
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||
if (!spec)
|
||||
return -ENOMEM;
|
||||
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria,
|
||||
misc_parameters_2.ipsec_syndrome);
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria,
|
||||
misc_parameters_2.metadata_reg_c_4);
|
||||
MLX5_SET(fte_match_param, spec->match_value,
|
||||
misc_parameters_2.ipsec_syndrome, 0);
|
||||
MLX5_SET(fte_match_param, spec->match_value,
|
||||
misc_parameters_2.metadata_reg_c_4, 0);
|
||||
if (rx == ipsec->rx_esw)
|
||||
spec->flow_context.flow_source = MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||
spec->match_criteria_enable = MLX5_MATCH_MISC_PARAMETERS_2;
|
||||
flow_act.flags = FLOW_ACT_NO_APPEND;
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST |
|
||||
MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
rule = mlx5_add_flow_rules(rx->ft.status, spec, &flow_act, dest, 2);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_warn(ipsec->mdev,
|
||||
"Failed to add ipsec rx status pass rule, err=%d\n", err);
|
||||
goto err_rule;
|
||||
}
|
||||
|
||||
rx->status.rule = rule;
|
||||
kvfree(spec);
|
||||
return 0;
|
||||
|
||||
err_rule:
|
||||
kvfree(spec);
|
||||
return err;
|
||||
}
|
||||
|
||||
static void mlx5_ipsec_rx_status_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
ipsec_rx_status_pass_destroy(ipsec, rx);
|
||||
ipsec_rx_status_drop_destroy(ipsec, rx);
|
||||
}
|
||||
|
||||
static int mlx5_ipsec_rx_status_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest)
|
||||
{
|
||||
int err;
|
||||
|
||||
err = ipsec_rx_status_drop_all_create(ipsec, rx);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = ipsec_rx_status_pass_create(ipsec, rx, dest);
|
||||
if (err)
|
||||
goto err_pass_create;
|
||||
|
||||
return 0;
|
||||
|
||||
err_pass_create:
|
||||
ipsec_rx_status_drop_destroy(ipsec, rx);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ipsec_miss_create(struct mlx5_core_dev *mdev,
|
||||
struct mlx5_flow_table *ft,
|
||||
struct mlx5e_ipsec_miss *miss,
|
||||
@ -333,12 +597,7 @@ static void rx_destroy(struct mlx5_core_dev *mdev, struct mlx5e_ipsec *ipsec,
|
||||
mlx5_destroy_flow_table(rx->ft.sa);
|
||||
if (rx->allow_tunnel_mode)
|
||||
mlx5_eswitch_unblock_encap(mdev);
|
||||
if (rx == ipsec->rx_esw) {
|
||||
mlx5_esw_ipsec_rx_status_destroy(ipsec, rx);
|
||||
} else {
|
||||
mlx5_del_flow_rules(rx->status.rule);
|
||||
mlx5_modify_header_dealloc(mdev, rx->status.modify_hdr);
|
||||
}
|
||||
mlx5_ipsec_rx_status_destroy(ipsec, rx);
|
||||
mlx5_destroy_flow_table(rx->ft.status);
|
||||
|
||||
mlx5_ipsec_fs_roce_rx_destroy(ipsec->roce, family, mdev);
|
||||
@ -419,7 +678,7 @@ static int rx_create(struct mlx5_core_dev *mdev, struct mlx5e_ipsec *ipsec,
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
ft = ipsec_ft_create(attr.ns, attr.status_level, attr.prio, 1, 0);
|
||||
ft = ipsec_ft_create(attr.ns, attr.status_level, attr.prio, 3, 0);
|
||||
if (IS_ERR(ft)) {
|
||||
err = PTR_ERR(ft);
|
||||
goto err_fs_ft_status;
|
||||
@ -428,10 +687,7 @@ static int rx_create(struct mlx5_core_dev *mdev, struct mlx5e_ipsec *ipsec,
|
||||
|
||||
dest[1].type = MLX5_FLOW_DESTINATION_TYPE_COUNTER;
|
||||
dest[1].counter_id = mlx5_fc_id(rx->fc->cnt);
|
||||
if (rx == ipsec->rx_esw)
|
||||
err = mlx5_esw_ipsec_rx_status_create(ipsec, rx, dest);
|
||||
else
|
||||
err = ipsec_status_rule(mdev, rx, dest);
|
||||
err = mlx5_ipsec_rx_status_create(ipsec, rx, dest);
|
||||
if (err)
|
||||
goto err_add;
|
||||
|
||||
@ -956,13 +1212,22 @@ static void setup_fte_esp(struct mlx5_flow_spec *spec)
|
||||
MLX5_SET(fte_match_param, spec->match_value, outer_headers.ip_protocol, IPPROTO_ESP);
|
||||
}
|
||||
|
||||
static void setup_fte_spi(struct mlx5_flow_spec *spec, u32 spi)
|
||||
static void setup_fte_spi(struct mlx5_flow_spec *spec, u32 spi, bool encap)
|
||||
{
|
||||
/* SPI number */
|
||||
spec->match_criteria_enable |= MLX5_MATCH_MISC_PARAMETERS;
|
||||
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria, misc_parameters.outer_esp_spi);
|
||||
MLX5_SET(fte_match_param, spec->match_value, misc_parameters.outer_esp_spi, spi);
|
||||
if (encap) {
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria,
|
||||
misc_parameters.inner_esp_spi);
|
||||
MLX5_SET(fte_match_param, spec->match_value,
|
||||
misc_parameters.inner_esp_spi, spi);
|
||||
} else {
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria,
|
||||
misc_parameters.outer_esp_spi);
|
||||
MLX5_SET(fte_match_param, spec->match_value,
|
||||
misc_parameters.outer_esp_spi, spi);
|
||||
}
|
||||
}
|
||||
|
||||
static void setup_fte_no_frags(struct mlx5_flow_spec *spec)
|
||||
@ -1052,29 +1317,48 @@ static int setup_modify_header(struct mlx5e_ipsec *ipsec, int type, u32 val, u8
|
||||
struct mlx5_flow_act *flow_act)
|
||||
{
|
||||
enum mlx5_flow_namespace_type ns_type = ipsec_fs_get_ns(ipsec, type, dir);
|
||||
u8 action[MLX5_UN_SZ_BYTES(set_add_copy_action_in_auto)] = {};
|
||||
u8 action[3][MLX5_UN_SZ_BYTES(set_add_copy_action_in_auto)] = {};
|
||||
struct mlx5_core_dev *mdev = ipsec->mdev;
|
||||
struct mlx5_modify_hdr *modify_hdr;
|
||||
u8 num_of_actions = 1;
|
||||
|
||||
MLX5_SET(set_action_in, action, action_type, MLX5_ACTION_TYPE_SET);
|
||||
MLX5_SET(set_action_in, action[0], action_type, MLX5_ACTION_TYPE_SET);
|
||||
switch (dir) {
|
||||
case XFRM_DEV_OFFLOAD_IN:
|
||||
MLX5_SET(set_action_in, action, field,
|
||||
MLX5_SET(set_action_in, action[0], field,
|
||||
MLX5_ACTION_IN_FIELD_METADATA_REG_B);
|
||||
|
||||
num_of_actions++;
|
||||
MLX5_SET(set_action_in, action[1], action_type, MLX5_ACTION_TYPE_SET);
|
||||
MLX5_SET(set_action_in, action[1], field, MLX5_ACTION_IN_FIELD_METADATA_REG_C_2);
|
||||
MLX5_SET(set_action_in, action[1], data, val);
|
||||
MLX5_SET(set_action_in, action[1], offset, 0);
|
||||
MLX5_SET(set_action_in, action[1], length, 32);
|
||||
|
||||
if (type == XFRM_DEV_OFFLOAD_CRYPTO) {
|
||||
num_of_actions++;
|
||||
MLX5_SET(set_action_in, action[2], action_type,
|
||||
MLX5_ACTION_TYPE_SET);
|
||||
MLX5_SET(set_action_in, action[2], field,
|
||||
MLX5_ACTION_IN_FIELD_METADATA_REG_C_4);
|
||||
MLX5_SET(set_action_in, action[2], data, 0);
|
||||
MLX5_SET(set_action_in, action[2], offset, 0);
|
||||
MLX5_SET(set_action_in, action[2], length, 32);
|
||||
}
|
||||
break;
|
||||
case XFRM_DEV_OFFLOAD_OUT:
|
||||
MLX5_SET(set_action_in, action, field,
|
||||
MLX5_SET(set_action_in, action[0], field,
|
||||
MLX5_ACTION_IN_FIELD_METADATA_REG_C_4);
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
MLX5_SET(set_action_in, action, data, val);
|
||||
MLX5_SET(set_action_in, action, offset, 0);
|
||||
MLX5_SET(set_action_in, action, length, 32);
|
||||
MLX5_SET(set_action_in, action[0], data, val);
|
||||
MLX5_SET(set_action_in, action[0], offset, 0);
|
||||
MLX5_SET(set_action_in, action[0], length, 32);
|
||||
|
||||
modify_hdr = mlx5_modify_header_alloc(mdev, ns_type, 1, action);
|
||||
modify_hdr = mlx5_modify_header_alloc(mdev, ns_type, num_of_actions, action);
|
||||
if (IS_ERR(modify_hdr)) {
|
||||
mlx5_core_err(mdev, "Failed to allocate modify_header %ld\n",
|
||||
PTR_ERR(modify_hdr));
|
||||
@ -1321,8 +1605,9 @@ static int rx_add_rule(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
else
|
||||
setup_fte_addr6(spec, attrs->saddr.a6, attrs->daddr.a6);
|
||||
|
||||
setup_fte_spi(spec, attrs->spi);
|
||||
setup_fte_esp(spec);
|
||||
setup_fte_spi(spec, attrs->spi, attrs->encap);
|
||||
if (!attrs->encap)
|
||||
setup_fte_esp(spec);
|
||||
setup_fte_no_frags(spec);
|
||||
setup_fte_upper_proto_match(spec, &attrs->upspec);
|
||||
|
||||
@ -1372,6 +1657,15 @@ static int rx_add_rule(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
mlx5_core_err(mdev, "fail to add RX ipsec rule err=%d\n", err);
|
||||
goto err_add_flow;
|
||||
}
|
||||
if (attrs->type == XFRM_DEV_OFFLOAD_PACKET)
|
||||
err = rx_add_rule_drop_replay(sa_entry, rx);
|
||||
if (err)
|
||||
goto err_add_replay;
|
||||
|
||||
err = rx_add_rule_drop_auth_trailer(sa_entry, rx);
|
||||
if (err)
|
||||
goto err_drop_reason;
|
||||
|
||||
kvfree(spec);
|
||||
|
||||
sa_entry->ipsec_rule.rule = rule;
|
||||
@ -1380,6 +1674,13 @@ static int rx_add_rule(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
sa_entry->ipsec_rule.pkt_reformat = flow_act.pkt_reformat;
|
||||
return 0;
|
||||
|
||||
err_drop_reason:
|
||||
if (sa_entry->ipsec_rule.replay.rule) {
|
||||
mlx5_del_flow_rules(sa_entry->ipsec_rule.replay.rule);
|
||||
mlx5_fc_destroy(mdev, sa_entry->ipsec_rule.replay.fc);
|
||||
}
|
||||
err_add_replay:
|
||||
mlx5_del_flow_rules(rule);
|
||||
err_add_flow:
|
||||
mlx5_fc_destroy(mdev, counter);
|
||||
err_add_cnt:
|
||||
@ -1428,7 +1729,7 @@ static int tx_add_rule(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
|
||||
switch (attrs->type) {
|
||||
case XFRM_DEV_OFFLOAD_CRYPTO:
|
||||
setup_fte_spi(spec, attrs->spi);
|
||||
setup_fte_spi(spec, attrs->spi, false);
|
||||
setup_fte_esp(spec);
|
||||
setup_fte_reg_a(spec);
|
||||
break;
|
||||
@ -1809,8 +2110,11 @@ static int mlx5e_ipsec_block_tc_offload(struct mlx5_core_dev *mdev)
|
||||
struct mlx5_eswitch *esw = mdev->priv.eswitch;
|
||||
int err = 0;
|
||||
|
||||
if (esw)
|
||||
down_write(&esw->mode_lock);
|
||||
if (esw) {
|
||||
err = mlx5_esw_lock(esw);
|
||||
if (err)
|
||||
return err;
|
||||
}
|
||||
|
||||
if (mdev->num_block_ipsec) {
|
||||
err = -EBUSY;
|
||||
@ -1821,7 +2125,7 @@ static int mlx5e_ipsec_block_tc_offload(struct mlx5_core_dev *mdev)
|
||||
|
||||
unlock:
|
||||
if (esw)
|
||||
up_write(&esw->mode_lock);
|
||||
mlx5_esw_unlock(esw);
|
||||
|
||||
return err;
|
||||
}
|
||||
@ -1887,6 +2191,17 @@ void mlx5e_accel_ipsec_fs_del_rule(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
|
||||
if (ipsec_rule->modify_hdr)
|
||||
mlx5_modify_header_dealloc(mdev, ipsec_rule->modify_hdr);
|
||||
|
||||
mlx5_del_flow_rules(ipsec_rule->trailer.rule);
|
||||
mlx5_fc_destroy(mdev, ipsec_rule->trailer.fc);
|
||||
|
||||
mlx5_del_flow_rules(ipsec_rule->auth.rule);
|
||||
mlx5_fc_destroy(mdev, ipsec_rule->auth.fc);
|
||||
|
||||
if (ipsec_rule->replay.rule) {
|
||||
mlx5_del_flow_rules(ipsec_rule->replay.rule);
|
||||
mlx5_fc_destroy(mdev, ipsec_rule->replay.fc);
|
||||
}
|
||||
mlx5_esw_ipsec_rx_id_mapping_remove(sa_entry);
|
||||
rx_ft_put(sa_entry->ipsec, sa_entry->attrs.family, sa_entry->attrs.type);
|
||||
}
|
||||
@ -1957,7 +2272,7 @@ void mlx5e_accel_ipsec_fs_cleanup(struct mlx5e_ipsec *ipsec)
|
||||
kfree(ipsec->rx_ipv6);
|
||||
|
||||
if (ipsec->is_uplink_rep) {
|
||||
xa_destroy(&ipsec->rx_esw->ipsec_obj_id_map);
|
||||
xa_destroy(&ipsec->ipsec_obj_id_map);
|
||||
|
||||
mutex_destroy(&ipsec->tx_esw->ft.mutex);
|
||||
WARN_ON(ipsec->tx_esw->ft.refcnt);
|
||||
@ -2020,7 +2335,7 @@ int mlx5e_accel_ipsec_fs_init(struct mlx5e_ipsec *ipsec,
|
||||
mutex_init(&ipsec->tx_esw->ft.mutex);
|
||||
mutex_init(&ipsec->rx_esw->ft.mutex);
|
||||
ipsec->tx_esw->ns = ns_esw;
|
||||
xa_init_flags(&ipsec->rx_esw->ipsec_obj_id_map, XA_FLAGS_ALLOC1);
|
||||
xa_init_flags(&ipsec->ipsec_obj_id_map, XA_FLAGS_ALLOC1);
|
||||
} else if (mlx5_ipsec_device_caps(mdev) & MLX5_IPSEC_CAP_ROCE) {
|
||||
ipsec->roce = mlx5_ipsec_fs_roce_init(mdev, devcom);
|
||||
} else {
|
||||
|
@ -6,6 +6,8 @@
|
||||
#include "ipsec.h"
|
||||
#include "lib/crypto.h"
|
||||
#include "lib/ipsec_fs_roce.h"
|
||||
#include "fs_core.h"
|
||||
#include "eswitch.h"
|
||||
|
||||
enum {
|
||||
MLX5_IPSEC_ASO_REMOVE_FLOW_PKT_CNT_OFFSET,
|
||||
@ -38,7 +40,10 @@ u32 mlx5_ipsec_device_caps(struct mlx5_core_dev *mdev)
|
||||
MLX5_CAP_ETH(mdev, insert_trailer) && MLX5_CAP_ETH(mdev, swp))
|
||||
caps |= MLX5_IPSEC_CAP_CRYPTO;
|
||||
|
||||
if (MLX5_CAP_IPSEC(mdev, ipsec_full_offload)) {
|
||||
if (MLX5_CAP_IPSEC(mdev, ipsec_full_offload) &&
|
||||
(mdev->priv.steering->mode == MLX5_FLOW_STEERING_MODE_DMFS ||
|
||||
(mdev->priv.steering->mode == MLX5_FLOW_STEERING_MODE_SMFS &&
|
||||
is_mdev_legacy_mode(mdev)))) {
|
||||
if (MLX5_CAP_FLOWTABLE_NIC_TX(mdev,
|
||||
reformat_add_esp_trasport) &&
|
||||
MLX5_CAP_FLOWTABLE_NIC_RX(mdev,
|
||||
@ -95,7 +100,7 @@ static void mlx5e_ipsec_packet_setup(void *obj, u32 pdn,
|
||||
|
||||
if (attrs->dir == XFRM_DEV_OFFLOAD_IN) {
|
||||
MLX5_SET(ipsec_aso, aso_ctx, window_sz,
|
||||
attrs->replay_esn.replay_window / 64);
|
||||
attrs->replay_esn.replay_window);
|
||||
MLX5_SET(ipsec_aso, aso_ctx, mode,
|
||||
MLX5_IPSEC_ASO_REPLAY_PROTECTION);
|
||||
}
|
||||
@ -559,6 +564,7 @@ void mlx5e_ipsec_aso_cleanup(struct mlx5e_ipsec *ipsec)
|
||||
dma_unmap_single(pdev, aso->dma_addr, sizeof(aso->ctx),
|
||||
DMA_BIDIRECTIONAL);
|
||||
kfree(aso);
|
||||
ipsec->aso = NULL;
|
||||
}
|
||||
|
||||
static void mlx5e_ipsec_aso_copy(struct mlx5_wqe_aso_ctrl_seg *ctrl,
|
||||
|
@ -2731,6 +2731,7 @@ void mlx5e_close_channels(struct mlx5e_channels *chs)
|
||||
{
|
||||
int i;
|
||||
|
||||
ASSERT_RTNL();
|
||||
if (chs->ptp) {
|
||||
mlx5e_ptp_close(chs->ptp);
|
||||
chs->ptp = NULL;
|
||||
@ -3012,17 +3013,29 @@ void mlx5e_activate_priv_channels(struct mlx5e_priv *priv)
|
||||
if (mlx5e_is_vport_rep(priv))
|
||||
mlx5e_rep_activate_channels(priv);
|
||||
|
||||
set_bit(MLX5E_STATE_CHANNELS_ACTIVE, &priv->state);
|
||||
|
||||
mlx5e_wait_channels_min_rx_wqes(&priv->channels);
|
||||
|
||||
if (priv->rx_res)
|
||||
mlx5e_rx_res_channels_activate(priv->rx_res, &priv->channels);
|
||||
}
|
||||
|
||||
static void mlx5e_cancel_tx_timeout_work(struct mlx5e_priv *priv)
|
||||
{
|
||||
WARN_ON_ONCE(test_bit(MLX5E_STATE_CHANNELS_ACTIVE, &priv->state));
|
||||
if (current_work() != &priv->tx_timeout_work)
|
||||
cancel_work_sync(&priv->tx_timeout_work);
|
||||
}
|
||||
|
||||
void mlx5e_deactivate_priv_channels(struct mlx5e_priv *priv)
|
||||
{
|
||||
if (priv->rx_res)
|
||||
mlx5e_rx_res_channels_deactivate(priv->rx_res);
|
||||
|
||||
clear_bit(MLX5E_STATE_CHANNELS_ACTIVE, &priv->state);
|
||||
mlx5e_cancel_tx_timeout_work(priv);
|
||||
|
||||
if (mlx5e_is_vport_rep(priv))
|
||||
mlx5e_rep_deactivate_channels(priv);
|
||||
|
||||
@ -4801,8 +4814,17 @@ static void mlx5e_tx_timeout_work(struct work_struct *work)
|
||||
struct net_device *netdev = priv->netdev;
|
||||
int i;
|
||||
|
||||
rtnl_lock();
|
||||
mutex_lock(&priv->state_lock);
|
||||
/* Take rtnl_lock to ensure no change in netdev->real_num_tx_queues
|
||||
* through this flow. However, channel closing flows have to wait for
|
||||
* this work to finish while holding rtnl lock too. So either get the
|
||||
* lock or find that channels are being closed for other reason and
|
||||
* this work is not relevant anymore.
|
||||
*/
|
||||
while (!rtnl_trylock()) {
|
||||
if (!test_bit(MLX5E_STATE_CHANNELS_ACTIVE, &priv->state))
|
||||
return;
|
||||
msleep(20);
|
||||
}
|
||||
|
||||
if (!test_bit(MLX5E_STATE_OPENED, &priv->state))
|
||||
goto unlock;
|
||||
@ -4821,7 +4843,6 @@ static void mlx5e_tx_timeout_work(struct work_struct *work)
|
||||
}
|
||||
|
||||
unlock:
|
||||
mutex_unlock(&priv->state_lock);
|
||||
rtnl_unlock();
|
||||
}
|
||||
|
||||
|
@ -1497,7 +1497,7 @@ mlx5e_vport_vf_rep_load(struct mlx5_core_dev *dev, struct mlx5_eswitch_rep *rep)
|
||||
|
||||
dl_port = mlx5_esw_offloads_devlink_port(dev->priv.eswitch,
|
||||
rpriv->rep->vport);
|
||||
if (dl_port) {
|
||||
if (!IS_ERR(dl_port)) {
|
||||
SET_NETDEV_DEVLINK_PORT(netdev, dl_port);
|
||||
mlx5e_rep_vnic_reporter_create(priv, dl_port);
|
||||
}
|
||||
|
@ -444,6 +444,9 @@ mlx5e_tc_add_flow_meter(struct mlx5e_priv *priv,
|
||||
struct mlx5e_flow_meter_handle *meter;
|
||||
enum mlx5e_post_meter_type type;
|
||||
|
||||
if (IS_ERR(post_act))
|
||||
return PTR_ERR(post_act);
|
||||
|
||||
meter = mlx5e_tc_meter_replace(priv->mdev, &attr->meter_attr.params);
|
||||
if (IS_ERR(meter)) {
|
||||
mlx5_core_err(priv->mdev, "Failed to get flow meter\n");
|
||||
@ -3738,6 +3741,20 @@ out_free:
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
set_branch_dest_ft(struct mlx5e_priv *priv, struct mlx5_flow_attr *attr)
|
||||
{
|
||||
struct mlx5e_post_act *post_act = get_post_action(priv);
|
||||
|
||||
if (IS_ERR(post_act))
|
||||
return PTR_ERR(post_act);
|
||||
|
||||
attr->action |= MLX5_FLOW_CONTEXT_ACTION_FWD_DEST;
|
||||
attr->dest_ft = mlx5e_tc_post_act_get_ft(post_act);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
alloc_branch_attr(struct mlx5e_tc_flow *flow,
|
||||
struct mlx5e_tc_act_branch_ctrl *cond,
|
||||
@ -3761,8 +3778,8 @@ alloc_branch_attr(struct mlx5e_tc_flow *flow,
|
||||
break;
|
||||
case FLOW_ACTION_ACCEPT:
|
||||
case FLOW_ACTION_PIPE:
|
||||
attr->action |= MLX5_FLOW_CONTEXT_ACTION_FWD_DEST;
|
||||
attr->dest_ft = mlx5e_tc_post_act_get_ft(get_post_action(flow->priv));
|
||||
if (set_branch_dest_ft(flow->priv, attr))
|
||||
goto out_err;
|
||||
break;
|
||||
case FLOW_ACTION_JUMP:
|
||||
if (*jump_count) {
|
||||
@ -3771,8 +3788,8 @@ alloc_branch_attr(struct mlx5e_tc_flow *flow,
|
||||
goto out_err;
|
||||
}
|
||||
*jump_count = cond->extval;
|
||||
attr->action |= MLX5_FLOW_CONTEXT_ACTION_FWD_DEST;
|
||||
attr->dest_ft = mlx5e_tc_post_act_get_ft(get_post_action(flow->priv));
|
||||
if (set_branch_dest_ft(flow->priv, attr))
|
||||
goto out_err;
|
||||
break;
|
||||
default:
|
||||
err = -EOPNOTSUPP;
|
||||
|
@ -21,158 +21,6 @@ enum {
|
||||
MLX5_ESW_IPSEC_TX_ESP_FT_CNT_LEVEL,
|
||||
};
|
||||
|
||||
static void esw_ipsec_rx_status_drop_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
mlx5_del_flow_rules(rx->status_drop.rule);
|
||||
mlx5_destroy_flow_group(rx->status_drop.group);
|
||||
mlx5_fc_destroy(ipsec->mdev, rx->status_drop_cnt);
|
||||
}
|
||||
|
||||
static void esw_ipsec_rx_status_pass_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
mlx5_del_flow_rules(rx->status.rule);
|
||||
mlx5_chains_put_table(esw_chains(ipsec->mdev->priv.eswitch), 0, 1, 0);
|
||||
}
|
||||
|
||||
static int esw_ipsec_rx_status_drop_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
|
||||
struct mlx5_flow_table *ft = rx->ft.status;
|
||||
struct mlx5_core_dev *mdev = ipsec->mdev;
|
||||
struct mlx5_flow_destination dest = {};
|
||||
struct mlx5_flow_act flow_act = {};
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_fc *flow_counter;
|
||||
struct mlx5_flow_spec *spec;
|
||||
struct mlx5_flow_group *g;
|
||||
u32 *flow_group_in;
|
||||
int err = 0;
|
||||
|
||||
flow_group_in = kvzalloc(inlen, GFP_KERNEL);
|
||||
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||
if (!flow_group_in || !spec) {
|
||||
err = -ENOMEM;
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
MLX5_SET(create_flow_group_in, flow_group_in, start_flow_index, ft->max_fte - 1);
|
||||
MLX5_SET(create_flow_group_in, flow_group_in, end_flow_index, ft->max_fte - 1);
|
||||
g = mlx5_create_flow_group(ft, flow_group_in);
|
||||
if (IS_ERR(g)) {
|
||||
err = PTR_ERR(g);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop flow group, err=%d\n", err);
|
||||
goto err_out;
|
||||
}
|
||||
|
||||
flow_counter = mlx5_fc_create(mdev, false);
|
||||
if (IS_ERR(flow_counter)) {
|
||||
err = PTR_ERR(flow_counter);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule counter, err=%d\n", err);
|
||||
goto err_cnt;
|
||||
}
|
||||
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_DROP | MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
dest.type = MLX5_FLOW_DESTINATION_TYPE_COUNTER;
|
||||
dest.counter_id = mlx5_fc_id(flow_counter);
|
||||
spec->flow_context.flow_source = MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||
rule = mlx5_add_flow_rules(ft, spec, &flow_act, &dest, 1);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_err(mdev,
|
||||
"Failed to add ipsec rx status drop rule, err=%d\n", err);
|
||||
goto err_rule;
|
||||
}
|
||||
|
||||
rx->status_drop.group = g;
|
||||
rx->status_drop.rule = rule;
|
||||
rx->status_drop_cnt = flow_counter;
|
||||
|
||||
kvfree(flow_group_in);
|
||||
kvfree(spec);
|
||||
return 0;
|
||||
|
||||
err_rule:
|
||||
mlx5_fc_destroy(mdev, flow_counter);
|
||||
err_cnt:
|
||||
mlx5_destroy_flow_group(g);
|
||||
err_out:
|
||||
kvfree(flow_group_in);
|
||||
kvfree(spec);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int esw_ipsec_rx_status_pass_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest)
|
||||
{
|
||||
struct mlx5_flow_act flow_act = {};
|
||||
struct mlx5_flow_handle *rule;
|
||||
struct mlx5_flow_spec *spec;
|
||||
int err;
|
||||
|
||||
spec = kvzalloc(sizeof(*spec), GFP_KERNEL);
|
||||
if (!spec)
|
||||
return -ENOMEM;
|
||||
|
||||
MLX5_SET_TO_ONES(fte_match_param, spec->match_criteria,
|
||||
misc_parameters_2.ipsec_syndrome);
|
||||
MLX5_SET(fte_match_param, spec->match_value,
|
||||
misc_parameters_2.ipsec_syndrome, 0);
|
||||
spec->flow_context.flow_source = MLX5_FLOW_CONTEXT_FLOW_SOURCE_UPLINK;
|
||||
spec->match_criteria_enable = MLX5_MATCH_MISC_PARAMETERS_2;
|
||||
flow_act.flags = FLOW_ACT_NO_APPEND;
|
||||
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST |
|
||||
MLX5_FLOW_CONTEXT_ACTION_COUNT;
|
||||
rule = mlx5_add_flow_rules(rx->ft.status, spec, &flow_act, dest, 2);
|
||||
if (IS_ERR(rule)) {
|
||||
err = PTR_ERR(rule);
|
||||
mlx5_core_warn(ipsec->mdev,
|
||||
"Failed to add ipsec rx status pass rule, err=%d\n", err);
|
||||
goto err_rule;
|
||||
}
|
||||
|
||||
rx->status.rule = rule;
|
||||
kvfree(spec);
|
||||
return 0;
|
||||
|
||||
err_rule:
|
||||
kvfree(spec);
|
||||
return err;
|
||||
}
|
||||
|
||||
void mlx5_esw_ipsec_rx_status_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx)
|
||||
{
|
||||
esw_ipsec_rx_status_pass_destroy(ipsec, rx);
|
||||
esw_ipsec_rx_status_drop_destroy(ipsec, rx);
|
||||
}
|
||||
|
||||
int mlx5_esw_ipsec_rx_status_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest)
|
||||
{
|
||||
int err;
|
||||
|
||||
err = esw_ipsec_rx_status_drop_create(ipsec, rx);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = esw_ipsec_rx_status_pass_create(ipsec, rx, dest);
|
||||
if (err)
|
||||
goto err_pass_create;
|
||||
|
||||
return 0;
|
||||
|
||||
err_pass_create:
|
||||
esw_ipsec_rx_status_drop_destroy(ipsec, rx);
|
||||
return err;
|
||||
}
|
||||
|
||||
void mlx5_esw_ipsec_rx_create_attr_set(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx_create_attr *attr)
|
||||
{
|
||||
@ -202,7 +50,7 @@ int mlx5_esw_ipsec_rx_setup_modify_header(struct mlx5e_ipsec_sa_entry *sa_entry,
|
||||
u32 mapped_id;
|
||||
int err;
|
||||
|
||||
err = xa_alloc_bh(&ipsec->rx_esw->ipsec_obj_id_map, &mapped_id,
|
||||
err = xa_alloc_bh(&ipsec->ipsec_obj_id_map, &mapped_id,
|
||||
xa_mk_value(sa_entry->ipsec_obj_id),
|
||||
XA_LIMIT(1, ESW_IPSEC_RX_MAPPED_ID_MASK), 0);
|
||||
if (err)
|
||||
@ -233,7 +81,7 @@ int mlx5_esw_ipsec_rx_setup_modify_header(struct mlx5e_ipsec_sa_entry *sa_entry,
|
||||
return 0;
|
||||
|
||||
err_header_alloc:
|
||||
xa_erase_bh(&ipsec->rx_esw->ipsec_obj_id_map, mapped_id);
|
||||
xa_erase_bh(&ipsec->ipsec_obj_id_map, mapped_id);
|
||||
return err;
|
||||
}
|
||||
|
||||
@ -242,7 +90,7 @@ void mlx5_esw_ipsec_rx_id_mapping_remove(struct mlx5e_ipsec_sa_entry *sa_entry)
|
||||
struct mlx5e_ipsec *ipsec = sa_entry->ipsec;
|
||||
|
||||
if (sa_entry->rx_mapped_id)
|
||||
xa_erase_bh(&ipsec->rx_esw->ipsec_obj_id_map,
|
||||
xa_erase_bh(&ipsec->ipsec_obj_id_map,
|
||||
sa_entry->rx_mapped_id);
|
||||
}
|
||||
|
||||
@ -252,7 +100,7 @@ int mlx5_esw_ipsec_rx_ipsec_obj_id_search(struct mlx5e_priv *priv, u32 id,
|
||||
struct mlx5e_ipsec *ipsec = priv->ipsec;
|
||||
void *val;
|
||||
|
||||
val = xa_load(&ipsec->rx_esw->ipsec_obj_id_map, id);
|
||||
val = xa_load(&ipsec->ipsec_obj_id_map, id);
|
||||
if (!val)
|
||||
return -ENOENT;
|
||||
|
||||
@ -304,7 +152,7 @@ void mlx5_esw_ipsec_restore_dest_uplink(struct mlx5_core_dev *mdev)
|
||||
|
||||
xa_for_each(&esw->offloads.vport_reps, i, rep) {
|
||||
rpriv = rep->rep_data[REP_ETH].priv;
|
||||
if (!rpriv || !rpriv->netdev)
|
||||
if (!rpriv || !rpriv->netdev || !atomic_read(&rpriv->tc_ht.nelems))
|
||||
continue;
|
||||
|
||||
rhashtable_walk_enter(&rpriv->tc_ht, &iter);
|
||||
|
@ -8,11 +8,6 @@ struct mlx5e_ipsec;
|
||||
struct mlx5e_ipsec_sa_entry;
|
||||
|
||||
#ifdef CONFIG_MLX5_ESWITCH
|
||||
void mlx5_esw_ipsec_rx_status_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx);
|
||||
int mlx5_esw_ipsec_rx_status_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest);
|
||||
void mlx5_esw_ipsec_rx_create_attr_set(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx_create_attr *attr);
|
||||
int mlx5_esw_ipsec_rx_status_pass_dest_get(struct mlx5e_ipsec *ipsec,
|
||||
@ -26,16 +21,6 @@ void mlx5_esw_ipsec_tx_create_attr_set(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_tx_create_attr *attr);
|
||||
void mlx5_esw_ipsec_restore_dest_uplink(struct mlx5_core_dev *mdev);
|
||||
#else
|
||||
static inline void mlx5_esw_ipsec_rx_status_destroy(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx) {}
|
||||
|
||||
static inline int mlx5_esw_ipsec_rx_status_create(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx *rx,
|
||||
struct mlx5_flow_destination *dest)
|
||||
{
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
static inline void mlx5_esw_ipsec_rx_create_attr_set(struct mlx5e_ipsec *ipsec,
|
||||
struct mlx5e_ipsec_rx_create_attr *attr) {}
|
||||
|
||||
|
@ -1463,7 +1463,7 @@ int mlx5_eswitch_enable_locked(struct mlx5_eswitch *esw, int num_vfs)
|
||||
{
|
||||
int err;
|
||||
|
||||
lockdep_assert_held(&esw->mode_lock);
|
||||
devl_assert_locked(priv_to_devlink(esw->dev));
|
||||
|
||||
if (!MLX5_CAP_ESW_FLOWTABLE_FDB(esw->dev, ft_support)) {
|
||||
esw_warn(esw->dev, "FDB is not supported, aborting ...\n");
|
||||
@ -1531,7 +1531,6 @@ int mlx5_eswitch_enable(struct mlx5_eswitch *esw, int num_vfs)
|
||||
if (toggle_lag)
|
||||
mlx5_lag_disable_change(esw->dev);
|
||||
|
||||
down_write(&esw->mode_lock);
|
||||
if (!mlx5_esw_is_fdb_created(esw)) {
|
||||
ret = mlx5_eswitch_enable_locked(esw, num_vfs);
|
||||
} else {
|
||||
@ -1554,8 +1553,6 @@ int mlx5_eswitch_enable(struct mlx5_eswitch *esw, int num_vfs)
|
||||
}
|
||||
}
|
||||
|
||||
up_write(&esw->mode_lock);
|
||||
|
||||
if (toggle_lag)
|
||||
mlx5_lag_enable_change(esw->dev);
|
||||
|
||||
@ -1569,12 +1566,11 @@ void mlx5_eswitch_disable_sriov(struct mlx5_eswitch *esw, bool clear_vf)
|
||||
return;
|
||||
|
||||
devl_assert_locked(priv_to_devlink(esw->dev));
|
||||
down_write(&esw->mode_lock);
|
||||
/* If driver is unloaded, this function is called twice by remove_one()
|
||||
* and mlx5_unload(). Prevent the second call.
|
||||
*/
|
||||
if (!esw->esw_funcs.num_vfs && !esw->esw_funcs.num_ec_vfs && !clear_vf)
|
||||
goto unlock;
|
||||
return;
|
||||
|
||||
esw_info(esw->dev, "Unload vfs: mode(%s), nvfs(%d), necvfs(%d), active vports(%d)\n",
|
||||
esw->mode == MLX5_ESWITCH_LEGACY ? "LEGACY" : "OFFLOADS",
|
||||
@ -1603,9 +1599,6 @@ void mlx5_eswitch_disable_sriov(struct mlx5_eswitch *esw, bool clear_vf)
|
||||
esw->esw_funcs.num_vfs = 0;
|
||||
else
|
||||
esw->esw_funcs.num_ec_vfs = 0;
|
||||
|
||||
unlock:
|
||||
up_write(&esw->mode_lock);
|
||||
}
|
||||
|
||||
/* Free resources for corresponding eswitch mode. It is called by devlink
|
||||
@ -1647,10 +1640,8 @@ void mlx5_eswitch_disable(struct mlx5_eswitch *esw)
|
||||
|
||||
devl_assert_locked(priv_to_devlink(esw->dev));
|
||||
mlx5_lag_disable_change(esw->dev);
|
||||
down_write(&esw->mode_lock);
|
||||
mlx5_eswitch_disable_locked(esw);
|
||||
esw->mode = MLX5_ESWITCH_LEGACY;
|
||||
up_write(&esw->mode_lock);
|
||||
mlx5_lag_enable_change(esw->dev);
|
||||
}
|
||||
|
||||
@ -2254,8 +2245,13 @@ bool mlx5_esw_hold(struct mlx5_core_dev *mdev)
|
||||
if (!mlx5_esw_allowed(esw))
|
||||
return true;
|
||||
|
||||
if (down_read_trylock(&esw->mode_lock) != 0)
|
||||
if (down_read_trylock(&esw->mode_lock) != 0) {
|
||||
if (esw->eswitch_operation_in_progress) {
|
||||
up_read(&esw->mode_lock);
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
@ -2312,7 +2308,8 @@ int mlx5_esw_try_lock(struct mlx5_eswitch *esw)
|
||||
if (down_write_trylock(&esw->mode_lock) == 0)
|
||||
return -EINVAL;
|
||||
|
||||
if (atomic64_read(&esw->user_count) > 0) {
|
||||
if (esw->eswitch_operation_in_progress ||
|
||||
atomic64_read(&esw->user_count) > 0) {
|
||||
up_write(&esw->mode_lock);
|
||||
return -EBUSY;
|
||||
}
|
||||
@ -2320,6 +2317,18 @@ int mlx5_esw_try_lock(struct mlx5_eswitch *esw)
|
||||
return esw->mode;
|
||||
}
|
||||
|
||||
int mlx5_esw_lock(struct mlx5_eswitch *esw)
|
||||
{
|
||||
down_write(&esw->mode_lock);
|
||||
|
||||
if (esw->eswitch_operation_in_progress) {
|
||||
up_write(&esw->mode_lock);
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* mlx5_esw_unlock() - Release write lock on esw mode lock
|
||||
* @esw: eswitch device.
|
||||
|
@ -383,6 +383,7 @@ struct mlx5_eswitch {
|
||||
struct xarray paired;
|
||||
struct mlx5_devcom_comp_dev *devcom;
|
||||
u16 enabled_ipsec_vf_count;
|
||||
bool eswitch_operation_in_progress;
|
||||
};
|
||||
|
||||
void esw_offloads_disable(struct mlx5_eswitch *esw);
|
||||
@ -827,6 +828,7 @@ void mlx5_esw_release(struct mlx5_core_dev *dev);
|
||||
void mlx5_esw_get(struct mlx5_core_dev *dev);
|
||||
void mlx5_esw_put(struct mlx5_core_dev *dev);
|
||||
int mlx5_esw_try_lock(struct mlx5_eswitch *esw);
|
||||
int mlx5_esw_lock(struct mlx5_eswitch *esw);
|
||||
void mlx5_esw_unlock(struct mlx5_eswitch *esw);
|
||||
|
||||
void esw_vport_change_handle_locked(struct mlx5_vport *vport);
|
||||
|
@ -3653,14 +3653,18 @@ static int esw_inline_mode_to_devlink(u8 mlx5_mode, u8 *mode)
|
||||
|
||||
static bool esw_offloads_devlink_ns_eq_netdev_ns(struct devlink *devlink)
|
||||
{
|
||||
struct mlx5_core_dev *dev = devlink_priv(devlink);
|
||||
struct net *devl_net, *netdev_net;
|
||||
struct mlx5_eswitch *esw;
|
||||
bool ret = false;
|
||||
|
||||
esw = mlx5_devlink_eswitch_nocheck_get(devlink);
|
||||
netdev_net = dev_net(esw->dev->mlx5e_res.uplink_netdev);
|
||||
devl_net = devlink_net(devlink);
|
||||
|
||||
return net_eq(devl_net, netdev_net);
|
||||
mutex_lock(&dev->mlx5e_res.uplink_netdev_lock);
|
||||
if (dev->mlx5e_res.uplink_netdev) {
|
||||
netdev_net = dev_net(dev->mlx5e_res.uplink_netdev);
|
||||
devl_net = devlink_net(devlink);
|
||||
ret = net_eq(devl_net, netdev_net);
|
||||
}
|
||||
mutex_unlock(&dev->mlx5e_res.uplink_netdev_lock);
|
||||
return ret;
|
||||
}
|
||||
|
||||
int mlx5_eswitch_block_mode(struct mlx5_core_dev *dev)
|
||||
@ -3733,13 +3737,16 @@ int mlx5_devlink_eswitch_mode_set(struct devlink *devlink, u16 mode,
|
||||
goto unlock;
|
||||
}
|
||||
|
||||
esw->eswitch_operation_in_progress = true;
|
||||
up_write(&esw->mode_lock);
|
||||
|
||||
mlx5_eswitch_disable_locked(esw);
|
||||
if (mode == DEVLINK_ESWITCH_MODE_SWITCHDEV) {
|
||||
if (mlx5_devlink_trap_get_num_active(esw->dev)) {
|
||||
NL_SET_ERR_MSG_MOD(extack,
|
||||
"Can't change mode while devlink traps are active");
|
||||
err = -EOPNOTSUPP;
|
||||
goto unlock;
|
||||
goto skip;
|
||||
}
|
||||
err = esw_offloads_start(esw, extack);
|
||||
} else if (mode == DEVLINK_ESWITCH_MODE_LEGACY) {
|
||||
@ -3749,6 +3756,9 @@ int mlx5_devlink_eswitch_mode_set(struct devlink *devlink, u16 mode,
|
||||
err = -EINVAL;
|
||||
}
|
||||
|
||||
skip:
|
||||
down_write(&esw->mode_lock);
|
||||
esw->eswitch_operation_in_progress = false;
|
||||
unlock:
|
||||
mlx5_esw_unlock(esw);
|
||||
enable_lag:
|
||||
@ -3759,16 +3769,12 @@ enable_lag:
|
||||
int mlx5_devlink_eswitch_mode_get(struct devlink *devlink, u16 *mode)
|
||||
{
|
||||
struct mlx5_eswitch *esw;
|
||||
int err;
|
||||
|
||||
esw = mlx5_devlink_eswitch_get(devlink);
|
||||
if (IS_ERR(esw))
|
||||
return PTR_ERR(esw);
|
||||
|
||||
down_read(&esw->mode_lock);
|
||||
err = esw_mode_to_devlink(esw->mode, mode);
|
||||
up_read(&esw->mode_lock);
|
||||
return err;
|
||||
return esw_mode_to_devlink(esw->mode, mode);
|
||||
}
|
||||
|
||||
static int mlx5_esw_vports_inline_set(struct mlx5_eswitch *esw, u8 mlx5_mode,
|
||||
@ -3862,11 +3868,15 @@ int mlx5_devlink_eswitch_inline_mode_set(struct devlink *devlink, u8 mode,
|
||||
if (err)
|
||||
goto out;
|
||||
|
||||
err = mlx5_esw_vports_inline_set(esw, mlx5_mode, extack);
|
||||
if (err)
|
||||
goto out;
|
||||
esw->eswitch_operation_in_progress = true;
|
||||
up_write(&esw->mode_lock);
|
||||
|
||||
esw->offloads.inline_mode = mlx5_mode;
|
||||
err = mlx5_esw_vports_inline_set(esw, mlx5_mode, extack);
|
||||
if (!err)
|
||||
esw->offloads.inline_mode = mlx5_mode;
|
||||
|
||||
down_write(&esw->mode_lock);
|
||||
esw->eswitch_operation_in_progress = false;
|
||||
up_write(&esw->mode_lock);
|
||||
return 0;
|
||||
|
||||
@ -3878,16 +3888,12 @@ out:
|
||||
int mlx5_devlink_eswitch_inline_mode_get(struct devlink *devlink, u8 *mode)
|
||||
{
|
||||
struct mlx5_eswitch *esw;
|
||||
int err;
|
||||
|
||||
esw = mlx5_devlink_eswitch_get(devlink);
|
||||
if (IS_ERR(esw))
|
||||
return PTR_ERR(esw);
|
||||
|
||||
down_read(&esw->mode_lock);
|
||||
err = esw_inline_mode_to_devlink(esw->offloads.inline_mode, mode);
|
||||
up_read(&esw->mode_lock);
|
||||
return err;
|
||||
return esw_inline_mode_to_devlink(esw->offloads.inline_mode, mode);
|
||||
}
|
||||
|
||||
bool mlx5_eswitch_block_encap(struct mlx5_core_dev *dev)
|
||||
@ -3969,6 +3975,9 @@ int mlx5_devlink_eswitch_encap_mode_set(struct devlink *devlink,
|
||||
goto unlock;
|
||||
}
|
||||
|
||||
esw->eswitch_operation_in_progress = true;
|
||||
up_write(&esw->mode_lock);
|
||||
|
||||
esw_destroy_offloads_fdb_tables(esw);
|
||||
|
||||
esw->offloads.encap = encap;
|
||||
@ -3982,6 +3991,9 @@ int mlx5_devlink_eswitch_encap_mode_set(struct devlink *devlink,
|
||||
(void)esw_create_offloads_fdb_tables(esw);
|
||||
}
|
||||
|
||||
down_write(&esw->mode_lock);
|
||||
esw->eswitch_operation_in_progress = false;
|
||||
|
||||
unlock:
|
||||
up_write(&esw->mode_lock);
|
||||
return err;
|
||||
@ -3996,9 +4008,7 @@ int mlx5_devlink_eswitch_encap_mode_get(struct devlink *devlink,
|
||||
if (IS_ERR(esw))
|
||||
return PTR_ERR(esw);
|
||||
|
||||
down_read(&esw->mode_lock);
|
||||
*encap = esw->offloads.encap;
|
||||
up_read(&esw->mode_lock);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -325,6 +325,29 @@ static void mlx5_fw_live_patch_event(struct work_struct *work)
|
||||
mlx5_core_err(dev, "Failed to reload FW tracer\n");
|
||||
}
|
||||
|
||||
#if IS_ENABLED(CONFIG_HOTPLUG_PCI_PCIE)
|
||||
static int mlx5_check_hotplug_interrupt(struct mlx5_core_dev *dev)
|
||||
{
|
||||
struct pci_dev *bridge = dev->pdev->bus->self;
|
||||
u16 reg16;
|
||||
int err;
|
||||
|
||||
if (!bridge)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
err = pcie_capability_read_word(bridge, PCI_EXP_SLTCTL, ®16);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
if ((reg16 & PCI_EXP_SLTCTL_HPIE) && (reg16 & PCI_EXP_SLTCTL_DLLSCE)) {
|
||||
mlx5_core_warn(dev, "FW reset is not supported as HotPlug is enabled\n");
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
static int mlx5_check_dev_ids(struct mlx5_core_dev *dev, u16 dev_id)
|
||||
{
|
||||
struct pci_bus *bridge_bus = dev->pdev->bus;
|
||||
@ -357,6 +380,12 @@ static bool mlx5_is_reset_now_capable(struct mlx5_core_dev *dev)
|
||||
return false;
|
||||
}
|
||||
|
||||
#if IS_ENABLED(CONFIG_HOTPLUG_PCI_PCIE)
|
||||
err = mlx5_check_hotplug_interrupt(dev);
|
||||
if (err)
|
||||
return false;
|
||||
#endif
|
||||
|
||||
err = pci_read_config_word(dev->pdev, PCI_DEVICE_ID, &dev_id);
|
||||
if (err)
|
||||
return false;
|
||||
|
@ -933,6 +933,7 @@ static void qed_ilt_shadow_free(struct qed_hwfn *p_hwfn)
|
||||
p_dma->virt_addr = NULL;
|
||||
}
|
||||
kfree(p_mngr->ilt_shadow);
|
||||
p_mngr->ilt_shadow = NULL;
|
||||
}
|
||||
|
||||
static int qed_ilt_blk_alloc(struct qed_hwfn *p_hwfn,
|
||||
|
@ -30,6 +30,8 @@
|
||||
|
||||
#define QCASPI_MAX_REGS 0x20
|
||||
|
||||
#define QCASPI_RX_MAX_FRAMES 4
|
||||
|
||||
static const u16 qcaspi_spi_regs[] = {
|
||||
SPI_REG_BFR_SIZE,
|
||||
SPI_REG_WRBUF_SPC_AVA,
|
||||
@ -252,9 +254,9 @@ qcaspi_get_ringparam(struct net_device *dev, struct ethtool_ringparam *ring,
|
||||
{
|
||||
struct qcaspi *qca = netdev_priv(dev);
|
||||
|
||||
ring->rx_max_pending = 4;
|
||||
ring->rx_max_pending = QCASPI_RX_MAX_FRAMES;
|
||||
ring->tx_max_pending = TX_RING_MAX_LEN;
|
||||
ring->rx_pending = 4;
|
||||
ring->rx_pending = QCASPI_RX_MAX_FRAMES;
|
||||
ring->tx_pending = qca->txr.count;
|
||||
}
|
||||
|
||||
@ -263,22 +265,21 @@ qcaspi_set_ringparam(struct net_device *dev, struct ethtool_ringparam *ring,
|
||||
struct kernel_ethtool_ringparam *kernel_ring,
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
const struct net_device_ops *ops = dev->netdev_ops;
|
||||
struct qcaspi *qca = netdev_priv(dev);
|
||||
|
||||
if ((ring->rx_pending) ||
|
||||
if (ring->rx_pending != QCASPI_RX_MAX_FRAMES ||
|
||||
(ring->rx_mini_pending) ||
|
||||
(ring->rx_jumbo_pending))
|
||||
return -EINVAL;
|
||||
|
||||
if (netif_running(dev))
|
||||
ops->ndo_stop(dev);
|
||||
if (qca->spi_thread)
|
||||
kthread_park(qca->spi_thread);
|
||||
|
||||
qca->txr.count = max_t(u32, ring->tx_pending, TX_RING_MIN_LEN);
|
||||
qca->txr.count = min_t(u16, qca->txr.count, TX_RING_MAX_LEN);
|
||||
|
||||
if (netif_running(dev))
|
||||
ops->ndo_open(dev);
|
||||
if (qca->spi_thread)
|
||||
kthread_unpark(qca->spi_thread);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -580,6 +580,18 @@ qcaspi_spi_thread(void *data)
|
||||
netdev_info(qca->net_dev, "SPI thread created\n");
|
||||
while (!kthread_should_stop()) {
|
||||
set_current_state(TASK_INTERRUPTIBLE);
|
||||
if (kthread_should_park()) {
|
||||
netif_tx_disable(qca->net_dev);
|
||||
netif_carrier_off(qca->net_dev);
|
||||
qcaspi_flush_tx_ring(qca);
|
||||
kthread_parkme();
|
||||
if (qca->sync == QCASPI_SYNC_READY) {
|
||||
netif_carrier_on(qca->net_dev);
|
||||
netif_wake_queue(qca->net_dev);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
if ((qca->intr_req == qca->intr_svc) &&
|
||||
!qca->txr.skb[qca->txr.head])
|
||||
schedule();
|
||||
@ -608,11 +620,17 @@ qcaspi_spi_thread(void *data)
|
||||
if (intr_cause & SPI_INT_CPU_ON) {
|
||||
qcaspi_qca7k_sync(qca, QCASPI_EVENT_CPUON);
|
||||
|
||||
/* Frame decoding in progress */
|
||||
if (qca->frm_handle.state != qca->frm_handle.init)
|
||||
qca->net_dev->stats.rx_dropped++;
|
||||
|
||||
qcafrm_fsm_init_spi(&qca->frm_handle);
|
||||
qca->stats.device_reset++;
|
||||
|
||||
/* not synced. */
|
||||
if (qca->sync != QCASPI_SYNC_READY)
|
||||
continue;
|
||||
|
||||
qca->stats.device_reset++;
|
||||
netif_wake_queue(qca->net_dev);
|
||||
netif_carrier_on(qca->net_dev);
|
||||
}
|
||||
|
@ -59,26 +59,19 @@ static int loongson_dwmac_probe(struct pci_dev *pdev, const struct pci_device_id
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
if (!of_device_is_compatible(np, "loongson, pci-gmac")) {
|
||||
pr_info("dwmac_loongson_pci: Incompatible OF node\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
plat = devm_kzalloc(&pdev->dev, sizeof(*plat), GFP_KERNEL);
|
||||
if (!plat)
|
||||
return -ENOMEM;
|
||||
|
||||
plat->mdio_bus_data = devm_kzalloc(&pdev->dev,
|
||||
sizeof(*plat->mdio_bus_data),
|
||||
GFP_KERNEL);
|
||||
if (!plat->mdio_bus_data)
|
||||
return -ENOMEM;
|
||||
|
||||
plat->mdio_node = of_get_child_by_name(np, "mdio");
|
||||
if (plat->mdio_node) {
|
||||
dev_info(&pdev->dev, "Found MDIO subnode\n");
|
||||
|
||||
plat->mdio_bus_data = devm_kzalloc(&pdev->dev,
|
||||
sizeof(*plat->mdio_bus_data),
|
||||
GFP_KERNEL);
|
||||
if (!plat->mdio_bus_data) {
|
||||
ret = -ENOMEM;
|
||||
goto err_put_node;
|
||||
}
|
||||
plat->mdio_bus_data->needs_reset = true;
|
||||
}
|
||||
|
||||
|
@ -34,6 +34,7 @@
|
||||
#define RGMII_CONFIG_LOOPBACK_EN BIT(2)
|
||||
#define RGMII_CONFIG_PROG_SWAP BIT(1)
|
||||
#define RGMII_CONFIG_DDR_MODE BIT(0)
|
||||
#define RGMII_CONFIG_SGMII_CLK_DVDR GENMASK(18, 10)
|
||||
|
||||
/* SDCC_HC_REG_DLL_CONFIG fields */
|
||||
#define SDCC_DLL_CONFIG_DLL_RST BIT(30)
|
||||
@ -78,6 +79,8 @@
|
||||
#define ETHQOS_MAC_CTRL_SPEED_MODE BIT(14)
|
||||
#define ETHQOS_MAC_CTRL_PORT_SEL BIT(15)
|
||||
|
||||
#define SGMII_10M_RX_CLK_DVDR 0x31
|
||||
|
||||
struct ethqos_emac_por {
|
||||
unsigned int offset;
|
||||
unsigned int value;
|
||||
@ -598,6 +601,9 @@ static int ethqos_configure_rgmii(struct qcom_ethqos *ethqos)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* On interface toggle MAC registers gets reset.
|
||||
* Configure MAC block for SGMII on ethernet phy link up
|
||||
*/
|
||||
static int ethqos_configure_sgmii(struct qcom_ethqos *ethqos)
|
||||
{
|
||||
int val;
|
||||
@ -617,6 +623,10 @@ static int ethqos_configure_sgmii(struct qcom_ethqos *ethqos)
|
||||
case SPEED_10:
|
||||
val |= ETHQOS_MAC_CTRL_PORT_SEL;
|
||||
val &= ~ETHQOS_MAC_CTRL_SPEED_MODE;
|
||||
rgmii_updatel(ethqos, RGMII_CONFIG_SGMII_CLK_DVDR,
|
||||
FIELD_PREP(RGMII_CONFIG_SGMII_CLK_DVDR,
|
||||
SGMII_10M_RX_CLK_DVDR),
|
||||
RGMII_IO_MACRO_CONFIG);
|
||||
break;
|
||||
}
|
||||
|
||||
|
@ -591,7 +591,11 @@ int stmmac_mdio_register(struct net_device *ndev)
|
||||
new_bus->parent = priv->device;
|
||||
|
||||
err = of_mdiobus_register(new_bus, mdio_node);
|
||||
if (err != 0) {
|
||||
if (err == -ENODEV) {
|
||||
err = 0;
|
||||
dev_info(dev, "MDIO bus is disabled\n");
|
||||
goto bus_register_fail;
|
||||
} else if (err) {
|
||||
dev_err_probe(dev, err, "Cannot register the MDIO bus\n");
|
||||
goto bus_register_fail;
|
||||
}
|
||||
|
@ -281,8 +281,10 @@ static int __team_options_register(struct team *team,
|
||||
return 0;
|
||||
|
||||
inst_rollback:
|
||||
for (i--; i >= 0; i--)
|
||||
for (i--; i >= 0; i--) {
|
||||
__team_option_inst_del_option(team, dst_opts[i]);
|
||||
list_del(&dst_opts[i]->list);
|
||||
}
|
||||
|
||||
i = option_count;
|
||||
alloc_rollback:
|
||||
|
@ -621,7 +621,7 @@ struct mlx5_ifc_fte_match_set_misc_bits {
|
||||
|
||||
u8 reserved_at_140[0x8];
|
||||
u8 bth_dst_qp[0x18];
|
||||
u8 reserved_at_160[0x20];
|
||||
u8 inner_esp_spi[0x20];
|
||||
u8 outer_esp_spi[0x20];
|
||||
u8 reserved_at_1a0[0x60];
|
||||
};
|
||||
@ -12001,6 +12001,13 @@ enum {
|
||||
MLX5_IPSEC_ASO_INC_SN = 0x2,
|
||||
};
|
||||
|
||||
enum {
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_32BIT = 0x0,
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_64BIT = 0x1,
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_128BIT = 0x2,
|
||||
MLX5_IPSEC_ASO_REPLAY_WIN_256BIT = 0x3,
|
||||
};
|
||||
|
||||
struct mlx5_ifc_ipsec_aso_bits {
|
||||
u8 valid[0x1];
|
||||
u8 reserved_at_201[0x1];
|
||||
|
@ -31,17 +31,22 @@ struct prefix_info {
|
||||
__u8 length;
|
||||
__u8 prefix_len;
|
||||
|
||||
union __packed {
|
||||
__u8 flags;
|
||||
struct __packed {
|
||||
#if defined(__BIG_ENDIAN_BITFIELD)
|
||||
__u8 onlink : 1,
|
||||
__u8 onlink : 1,
|
||||
autoconf : 1,
|
||||
reserved : 6;
|
||||
#elif defined(__LITTLE_ENDIAN_BITFIELD)
|
||||
__u8 reserved : 6,
|
||||
__u8 reserved : 6,
|
||||
autoconf : 1,
|
||||
onlink : 1;
|
||||
#else
|
||||
#error "Please fix <asm/byteorder.h>"
|
||||
#endif
|
||||
};
|
||||
};
|
||||
__be32 valid;
|
||||
__be32 prefered;
|
||||
__be32 reserved2;
|
||||
@ -49,6 +54,9 @@ struct prefix_info {
|
||||
struct in6_addr prefix;
|
||||
};
|
||||
|
||||
/* rfc4861 4.6.2: IPv6 PIO is 32 bytes in size */
|
||||
static_assert(sizeof(struct prefix_info) == 32);
|
||||
|
||||
#include <linux/ipv6.h>
|
||||
#include <linux/netdevice.h>
|
||||
#include <net/if_inet6.h>
|
||||
|
@ -22,10 +22,6 @@
|
||||
#define IF_RS_SENT 0x10
|
||||
#define IF_READY 0x80000000
|
||||
|
||||
/* prefix flags */
|
||||
#define IF_PREFIX_ONLINK 0x01
|
||||
#define IF_PREFIX_AUTOCONF 0x02
|
||||
|
||||
enum {
|
||||
INET6_IFADDR_STATE_PREDAD,
|
||||
INET6_IFADDR_STATE_DAD,
|
||||
|
@ -62,6 +62,8 @@ struct nf_flowtable_type {
|
||||
enum flow_offload_tuple_dir dir,
|
||||
struct nf_flow_rule *flow_rule);
|
||||
void (*free)(struct nf_flowtable *ft);
|
||||
void (*get)(struct nf_flowtable *ft);
|
||||
void (*put)(struct nf_flowtable *ft);
|
||||
nf_hookfn *hook;
|
||||
struct module *owner;
|
||||
};
|
||||
@ -240,6 +242,11 @@ nf_flow_table_offload_add_cb(struct nf_flowtable *flow_table,
|
||||
}
|
||||
|
||||
list_add_tail(&block_cb->list, &block->cb_list);
|
||||
up_write(&flow_table->flow_block_lock);
|
||||
|
||||
if (flow_table->type->get)
|
||||
flow_table->type->get(flow_table);
|
||||
return 0;
|
||||
|
||||
unlock:
|
||||
up_write(&flow_table->flow_block_lock);
|
||||
@ -262,6 +269,9 @@ nf_flow_table_offload_del_cb(struct nf_flowtable *flow_table,
|
||||
WARN_ON(true);
|
||||
}
|
||||
up_write(&flow_table->flow_block_lock);
|
||||
|
||||
if (flow_table->type->put)
|
||||
flow_table->type->put(flow_table);
|
||||
}
|
||||
|
||||
void flow_offload_route_init(struct flow_offload *flow,
|
||||
|
@ -1775,15 +1775,14 @@ static int atalk_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)
|
||||
break;
|
||||
}
|
||||
case TIOCINQ: {
|
||||
/*
|
||||
* These two are safe on a single CPU system as only
|
||||
* user tasks fiddle here
|
||||
*/
|
||||
struct sk_buff *skb = skb_peek(&sk->sk_receive_queue);
|
||||
struct sk_buff *skb;
|
||||
long amount = 0;
|
||||
|
||||
spin_lock_irq(&sk->sk_receive_queue.lock);
|
||||
skb = skb_peek(&sk->sk_receive_queue);
|
||||
if (skb)
|
||||
amount = skb->len - sizeof(struct ddpehdr);
|
||||
spin_unlock_irq(&sk->sk_receive_queue.lock);
|
||||
rc = put_user(amount, (int __user *)argp);
|
||||
break;
|
||||
}
|
||||
|
@ -73,14 +73,17 @@ static int do_vcc_ioctl(struct socket *sock, unsigned int cmd,
|
||||
case SIOCINQ:
|
||||
{
|
||||
struct sk_buff *skb;
|
||||
int amount;
|
||||
|
||||
if (sock->state != SS_CONNECTED) {
|
||||
error = -EINVAL;
|
||||
goto done;
|
||||
}
|
||||
spin_lock_irq(&sk->sk_receive_queue.lock);
|
||||
skb = skb_peek(&sk->sk_receive_queue);
|
||||
error = put_user(skb ? skb->len : 0,
|
||||
(int __user *)argp) ? -EFAULT : 0;
|
||||
amount = skb ? skb->len : 0;
|
||||
spin_unlock_irq(&sk->sk_receive_queue.lock);
|
||||
error = put_user(amount, (int __user *)argp) ? -EFAULT : 0;
|
||||
goto done;
|
||||
}
|
||||
case ATM_SETSC:
|
||||
|
@ -253,9 +253,11 @@ static int neigh_forced_gc(struct neigh_table *tbl)
|
||||
{
|
||||
int max_clean = atomic_read(&tbl->gc_entries) -
|
||||
READ_ONCE(tbl->gc_thresh2);
|
||||
u64 tmax = ktime_get_ns() + NSEC_PER_MSEC;
|
||||
unsigned long tref = jiffies - 5 * HZ;
|
||||
struct neighbour *n, *tmp;
|
||||
int shrunk = 0;
|
||||
int loop = 0;
|
||||
|
||||
NEIGH_CACHE_STAT_INC(tbl, forced_gc_runs);
|
||||
|
||||
@ -278,11 +280,16 @@ static int neigh_forced_gc(struct neigh_table *tbl)
|
||||
shrunk++;
|
||||
if (shrunk >= max_clean)
|
||||
break;
|
||||
if (++loop == 16) {
|
||||
if (ktime_get_ns() > tmax)
|
||||
goto unlock;
|
||||
loop = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
WRITE_ONCE(tbl->last_flush, jiffies);
|
||||
|
||||
unlock:
|
||||
write_unlock_bh(&tbl->lock);
|
||||
|
||||
return shrunk;
|
||||
|
@ -4522,8 +4522,9 @@ struct sk_buff *skb_segment(struct sk_buff *head_skb,
|
||||
/* GSO partial only requires that we trim off any excess that
|
||||
* doesn't fit into an MSS sized block, so take care of that
|
||||
* now.
|
||||
* Cap len to not accidentally hit GSO_BY_FRAGS.
|
||||
*/
|
||||
partial_segs = len / mss;
|
||||
partial_segs = min(len, GSO_BY_FRAGS - 1) / mss;
|
||||
if (partial_segs > 1)
|
||||
mss *= partial_segs;
|
||||
else
|
||||
|
@ -4368,6 +4368,23 @@ EXPORT_SYMBOL(tcp_do_parse_auth_options);
|
||||
* up to bandwidth of 18Gigabit/sec. 8) ]
|
||||
*/
|
||||
|
||||
/* Estimates max number of increments of remote peer TSval in
|
||||
* a replay window (based on our current RTO estimation).
|
||||
*/
|
||||
static u32 tcp_tsval_replay(const struct sock *sk)
|
||||
{
|
||||
/* If we use usec TS resolution,
|
||||
* then expect the remote peer to use the same resolution.
|
||||
*/
|
||||
if (tcp_sk(sk)->tcp_usec_ts)
|
||||
return inet_csk(sk)->icsk_rto * (USEC_PER_SEC / HZ);
|
||||
|
||||
/* RFC 7323 recommends a TSval clock between 1ms and 1sec.
|
||||
* We know that some OS (including old linux) can use 1200 Hz.
|
||||
*/
|
||||
return inet_csk(sk)->icsk_rto * 1200 / HZ;
|
||||
}
|
||||
|
||||
static int tcp_disordered_ack(const struct sock *sk, const struct sk_buff *skb)
|
||||
{
|
||||
const struct tcp_sock *tp = tcp_sk(sk);
|
||||
@ -4375,7 +4392,7 @@ static int tcp_disordered_ack(const struct sock *sk, const struct sk_buff *skb)
|
||||
u32 seq = TCP_SKB_CB(skb)->seq;
|
||||
u32 ack = TCP_SKB_CB(skb)->ack_seq;
|
||||
|
||||
return (/* 1. Pure ACK with correct sequence number. */
|
||||
return /* 1. Pure ACK with correct sequence number. */
|
||||
(th->ack && seq == TCP_SKB_CB(skb)->end_seq && seq == tp->rcv_nxt) &&
|
||||
|
||||
/* 2. ... and duplicate ACK. */
|
||||
@ -4385,7 +4402,8 @@ static int tcp_disordered_ack(const struct sock *sk, const struct sk_buff *skb)
|
||||
!tcp_may_update_window(tp, ack, seq, ntohs(th->window) << tp->rx_opt.snd_wscale) &&
|
||||
|
||||
/* 4. ... and sits in replay window. */
|
||||
(s32)(tp->rx_opt.ts_recent - tp->rx_opt.rcv_tsval) <= (inet_csk(sk)->icsk_rto * 1024) / HZ);
|
||||
(s32)(tp->rx_opt.ts_recent - tp->rx_opt.rcv_tsval) <=
|
||||
tcp_tsval_replay(sk);
|
||||
}
|
||||
|
||||
static inline bool tcp_paws_discard(const struct sock *sk,
|
||||
|
@ -3293,7 +3293,13 @@ int __tcp_retransmit_skb(struct sock *sk, struct sk_buff *skb, int segs)
|
||||
if (skb_still_in_host_queue(sk, skb))
|
||||
return -EBUSY;
|
||||
|
||||
start:
|
||||
if (before(TCP_SKB_CB(skb)->seq, tp->snd_una)) {
|
||||
if (unlikely(TCP_SKB_CB(skb)->tcp_flags & TCPHDR_SYN)) {
|
||||
TCP_SKB_CB(skb)->tcp_flags &= ~TCPHDR_SYN;
|
||||
TCP_SKB_CB(skb)->seq++;
|
||||
goto start;
|
||||
}
|
||||
if (unlikely(before(TCP_SKB_CB(skb)->end_seq, tp->snd_una))) {
|
||||
WARN_ON_ONCE(1);
|
||||
return -EINVAL;
|
||||
|
@ -6149,11 +6149,7 @@ static int inet6_fill_prefix(struct sk_buff *skb, struct inet6_dev *idev,
|
||||
pmsg->prefix_len = pinfo->prefix_len;
|
||||
pmsg->prefix_type = pinfo->type;
|
||||
pmsg->prefix_pad3 = 0;
|
||||
pmsg->prefix_flags = 0;
|
||||
if (pinfo->onlink)
|
||||
pmsg->prefix_flags |= IF_PREFIX_ONLINK;
|
||||
if (pinfo->autoconf)
|
||||
pmsg->prefix_flags |= IF_PREFIX_AUTOCONF;
|
||||
pmsg->prefix_flags = pinfo->flags;
|
||||
|
||||
if (nla_put(skb, PREFIX_ADDRESS, sizeof(pinfo->prefix), &pinfo->prefix))
|
||||
goto nla_put_failure;
|
||||
|
@ -1315,9 +1315,11 @@ static int rose_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)
|
||||
case TIOCINQ: {
|
||||
struct sk_buff *skb;
|
||||
long amount = 0L;
|
||||
/* These two are safe on a single CPU system as only user tasks fiddle here */
|
||||
|
||||
spin_lock_irq(&sk->sk_receive_queue.lock);
|
||||
if ((skb = skb_peek(&sk->sk_receive_queue)) != NULL)
|
||||
amount = skb->len;
|
||||
spin_unlock_irq(&sk->sk_receive_queue.lock);
|
||||
return put_user(amount, (unsigned int __user *) argp);
|
||||
}
|
||||
|
||||
|
@ -286,9 +286,31 @@ static bool tcf_ct_flow_is_outdated(const struct flow_offload *flow)
|
||||
!test_bit(NF_FLOW_HW_ESTABLISHED, &flow->flags);
|
||||
}
|
||||
|
||||
static void tcf_ct_flow_table_get_ref(struct tcf_ct_flow_table *ct_ft);
|
||||
|
||||
static void tcf_ct_nf_get(struct nf_flowtable *ft)
|
||||
{
|
||||
struct tcf_ct_flow_table *ct_ft =
|
||||
container_of(ft, struct tcf_ct_flow_table, nf_ft);
|
||||
|
||||
tcf_ct_flow_table_get_ref(ct_ft);
|
||||
}
|
||||
|
||||
static void tcf_ct_flow_table_put(struct tcf_ct_flow_table *ct_ft);
|
||||
|
||||
static void tcf_ct_nf_put(struct nf_flowtable *ft)
|
||||
{
|
||||
struct tcf_ct_flow_table *ct_ft =
|
||||
container_of(ft, struct tcf_ct_flow_table, nf_ft);
|
||||
|
||||
tcf_ct_flow_table_put(ct_ft);
|
||||
}
|
||||
|
||||
static struct nf_flowtable_type flowtable_ct = {
|
||||
.gc = tcf_ct_flow_is_outdated,
|
||||
.action = tcf_ct_flow_table_fill_actions,
|
||||
.get = tcf_ct_nf_get,
|
||||
.put = tcf_ct_nf_put,
|
||||
.owner = THIS_MODULE,
|
||||
};
|
||||
|
||||
@ -337,9 +359,13 @@ err_alloc:
|
||||
return err;
|
||||
}
|
||||
|
||||
static void tcf_ct_flow_table_get_ref(struct tcf_ct_flow_table *ct_ft)
|
||||
{
|
||||
refcount_inc(&ct_ft->ref);
|
||||
}
|
||||
|
||||
static void tcf_ct_flow_table_cleanup_work(struct work_struct *work)
|
||||
{
|
||||
struct flow_block_cb *block_cb, *tmp_cb;
|
||||
struct tcf_ct_flow_table *ct_ft;
|
||||
struct flow_block *block;
|
||||
|
||||
@ -347,13 +373,9 @@ static void tcf_ct_flow_table_cleanup_work(struct work_struct *work)
|
||||
rwork);
|
||||
nf_flow_table_free(&ct_ft->nf_ft);
|
||||
|
||||
/* Remove any remaining callbacks before cleanup */
|
||||
block = &ct_ft->nf_ft.flow_block;
|
||||
down_write(&ct_ft->nf_ft.flow_block_lock);
|
||||
list_for_each_entry_safe(block_cb, tmp_cb, &block->cb_list, list) {
|
||||
list_del(&block_cb->list);
|
||||
flow_block_cb_free(block_cb);
|
||||
}
|
||||
WARN_ON(!list_empty(&block->cb_list));
|
||||
up_write(&ct_ft->nf_ft.flow_block_lock);
|
||||
kfree(ct_ft);
|
||||
|
||||
|
@ -843,7 +843,7 @@ static s64 virtio_transport_has_space(struct vsock_sock *vsk)
|
||||
struct virtio_vsock_sock *vvs = vsk->trans;
|
||||
s64 bytes;
|
||||
|
||||
bytes = vvs->peer_buf_alloc - (vvs->tx_cnt - vvs->peer_fwd_cnt);
|
||||
bytes = (s64)vvs->peer_buf_alloc - (vvs->tx_cnt - vvs->peer_fwd_cnt);
|
||||
if (bytes < 0)
|
||||
bytes = 0;
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user