2018-03-20 21:58:05 +07:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0 */
|
|
|
|
/* Copyright (c) 2018, Intel Corporation. */
|
|
|
|
|
|
|
|
#ifndef _ICE_H_
|
|
|
|
#define _ICE_H_
|
|
|
|
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/netdevice.h>
|
|
|
|
#include <linux/compiler.h>
|
2018-03-20 21:58:09 +07:00
|
|
|
#include <linux/etherdevice.h>
|
2018-03-20 21:58:13 +07:00
|
|
|
#include <linux/skbuff.h>
|
2018-03-20 21:58:11 +07:00
|
|
|
#include <linux/cpumask.h>
|
2018-03-20 21:58:16 +07:00
|
|
|
#include <linux/rtnetlink.h>
|
2018-03-20 21:58:11 +07:00
|
|
|
#include <linux/if_vlan.h>
|
2018-03-20 21:58:13 +07:00
|
|
|
#include <linux/dma-mapping.h>
|
2018-03-20 21:58:05 +07:00
|
|
|
#include <linux/pci.h>
|
2018-03-20 21:58:10 +07:00
|
|
|
#include <linux/workqueue.h>
|
2018-03-20 21:58:05 +07:00
|
|
|
#include <linux/aer.h>
|
2018-03-20 21:58:10 +07:00
|
|
|
#include <linux/interrupt.h>
|
2018-03-20 21:58:16 +07:00
|
|
|
#include <linux/ethtool.h>
|
2018-03-20 21:58:10 +07:00
|
|
|
#include <linux/timer.h>
|
2018-03-20 21:58:06 +07:00
|
|
|
#include <linux/delay.h>
|
2018-03-20 21:58:05 +07:00
|
|
|
#include <linux/bitmap.h>
|
2018-03-20 21:58:11 +07:00
|
|
|
#include <linux/log2.h>
|
2018-03-20 21:58:15 +07:00
|
|
|
#include <linux/ip.h>
|
|
|
|
#include <linux/ipv6.h>
|
2018-03-20 21:58:10 +07:00
|
|
|
#include <linux/if_bridge.h>
|
2018-09-20 07:42:55 +07:00
|
|
|
#include <linux/avf/virtchnl.h>
|
2018-03-20 21:58:15 +07:00
|
|
|
#include <net/ipv6.h>
|
2018-03-20 21:58:05 +07:00
|
|
|
#include "ice_devids.h"
|
|
|
|
#include "ice_type.h"
|
2018-03-20 21:58:10 +07:00
|
|
|
#include "ice_txrx.h"
|
ice: Get switch config, scheduler config and device capabilities
This patch adds to the initialization flow by getting switch
configuration, scheduler configuration and device capabilities.
Switch configuration:
On boot, an L2 switch element is created in the firmware per physical
function. Each physical function is also mapped to a port, to which its
switch element is connected. In other words, this switch can be visualized
as an embedded vSwitch that can connect a physical function's virtual
station interfaces (VSIs) to the egress/ingress port. Egress/ingress
filters will be eventually created and applied on this switch element.
As part of the initialization flow, the driver gets configuration data
from this switch element and stores it.
Scheduler configuration:
The Tx scheduler is a subsystem responsible for setting and enforcing QoS.
As part of the initialization flow, the driver queries and stores the
default scheduler configuration for the given physical function.
Device capabilities:
As part of initialization, the driver has to determine what the device is
capable of (ex. max queues, VSIs, etc). This information is obtained from
the firmware and stored by the driver.
CC: Shannon Nelson <shannon.nelson@oracle.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Acked-by: Shannon Nelson <shannon.nelson@oracle.com>
Tested-by: Tony Brelinski <tonyx.brelinski@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-03-20 21:58:08 +07:00
|
|
|
#include "ice_switch.h"
|
2018-03-20 21:58:07 +07:00
|
|
|
#include "ice_common.h"
|
ice: Get switch config, scheduler config and device capabilities
This patch adds to the initialization flow by getting switch
configuration, scheduler configuration and device capabilities.
Switch configuration:
On boot, an L2 switch element is created in the firmware per physical
function. Each physical function is also mapped to a port, to which its
switch element is connected. In other words, this switch can be visualized
as an embedded vSwitch that can connect a physical function's virtual
station interfaces (VSIs) to the egress/ingress port. Egress/ingress
filters will be eventually created and applied on this switch element.
As part of the initialization flow, the driver gets configuration data
from this switch element and stores it.
Scheduler configuration:
The Tx scheduler is a subsystem responsible for setting and enforcing QoS.
As part of the initialization flow, the driver queries and stores the
default scheduler configuration for the given physical function.
Device capabilities:
As part of initialization, the driver has to determine what the device is
capable of (ex. max queues, VSIs, etc). This information is obtained from
the firmware and stored by the driver.
CC: Shannon Nelson <shannon.nelson@oracle.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Acked-by: Shannon Nelson <shannon.nelson@oracle.com>
Tested-by: Tony Brelinski <tonyx.brelinski@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-03-20 21:58:08 +07:00
|
|
|
#include "ice_sched.h"
|
2018-09-20 07:42:55 +07:00
|
|
|
#include "ice_virtchnl_pf.h"
|
2018-09-20 07:42:57 +07:00
|
|
|
#include "ice_sriov.h"
|
2018-03-20 21:58:05 +07:00
|
|
|
|
2018-03-20 21:58:16 +07:00
|
|
|
extern const char ice_drv_ver[];
|
2018-03-20 21:58:05 +07:00
|
|
|
#define ICE_BAR0 0
|
2018-03-20 21:58:11 +07:00
|
|
|
#define ICE_DFLT_NUM_DESC 128
|
|
|
|
#define ICE_REQ_DESC_MULTIPLE 32
|
2018-09-20 07:23:11 +07:00
|
|
|
#define ICE_MIN_NUM_DESC ICE_REQ_DESC_MULTIPLE
|
|
|
|
#define ICE_MAX_NUM_DESC 8160
|
2018-03-20 21:58:17 +07:00
|
|
|
#define ICE_DFLT_TRAFFIC_CLASS BIT(0)
|
2018-03-20 21:58:10 +07:00
|
|
|
#define ICE_INT_NAME_STR_LEN (IFNAMSIZ + 16)
|
2018-03-20 21:58:16 +07:00
|
|
|
#define ICE_ETHTOOL_FWVER_LEN 32
|
2018-03-20 21:58:07 +07:00
|
|
|
#define ICE_AQ_LEN 64
|
2018-09-20 07:42:54 +07:00
|
|
|
#define ICE_MBXQ_LEN 64
|
2018-03-20 21:58:10 +07:00
|
|
|
#define ICE_MIN_MSIX 2
|
2018-03-20 21:58:11 +07:00
|
|
|
#define ICE_NO_VSI 0xffff
|
2018-03-20 21:58:10 +07:00
|
|
|
#define ICE_MAX_TXQS 2048
|
|
|
|
#define ICE_MAX_RXQS 2048
|
2018-03-20 21:58:11 +07:00
|
|
|
#define ICE_VSI_MAP_CONTIG 0
|
|
|
|
#define ICE_VSI_MAP_SCATTER 1
|
|
|
|
#define ICE_MAX_SCATTER_TXQS 16
|
|
|
|
#define ICE_MAX_SCATTER_RXQS 16
|
2018-03-20 21:58:13 +07:00
|
|
|
#define ICE_Q_WAIT_RETRY_LIMIT 10
|
|
|
|
#define ICE_Q_WAIT_MAX_RETRY (5 * ICE_Q_WAIT_RETRY_LIMIT)
|
2018-03-20 21:58:15 +07:00
|
|
|
#define ICE_MAX_LG_RSS_QS 256
|
|
|
|
#define ICE_MAX_SMALL_RSS_QS 8
|
2018-03-20 21:58:10 +07:00
|
|
|
#define ICE_RES_VALID_BIT 0x8000
|
|
|
|
#define ICE_RES_MISC_VEC_ID (ICE_RES_VALID_BIT - 1)
|
2018-03-20 21:58:11 +07:00
|
|
|
#define ICE_INVAL_Q_INDEX 0xffff
|
2018-08-09 20:29:50 +07:00
|
|
|
#define ICE_INVAL_VFID 256
|
2018-09-20 07:42:54 +07:00
|
|
|
#define ICE_MAX_VF_COUNT 256
|
2018-09-20 07:42:55 +07:00
|
|
|
#define ICE_MAX_QS_PER_VF 256
|
|
|
|
#define ICE_MIN_QS_PER_VF 1
|
|
|
|
#define ICE_DFLT_QS_PER_VF 4
|
2018-09-20 07:42:59 +07:00
|
|
|
#define ICE_MAX_BASE_QS_PER_VF 16
|
2018-09-20 07:42:55 +07:00
|
|
|
#define ICE_MAX_INTR_PER_VF 65
|
|
|
|
#define ICE_MIN_INTR_PER_VF (ICE_MIN_QS_PER_VF + 1)
|
|
|
|
#define ICE_DFLT_INTR_PER_VF (ICE_DFLT_QS_PER_VF + 1)
|
2018-03-20 21:58:05 +07:00
|
|
|
|
2018-10-27 00:40:51 +07:00
|
|
|
#define ICE_MAX_RESET_WAIT 20
|
|
|
|
|
2018-03-20 21:58:16 +07:00
|
|
|
#define ICE_VSIQF_HKEY_ARRAY_SIZE ((VSIQF_HKEY_MAX_INDEX + 1) * 4)
|
|
|
|
|
2018-03-20 21:58:05 +07:00
|
|
|
#define ICE_DFLT_NETIF_M (NETIF_MSG_DRV | NETIF_MSG_PROBE | NETIF_MSG_LINK)
|
|
|
|
|
2018-03-20 21:58:11 +07:00
|
|
|
#define ICE_MAX_MTU (ICE_AQ_SET_MAC_FRAME_SIZE_MAX - \
|
|
|
|
ETH_HLEN + ETH_FCS_LEN + VLAN_HLEN)
|
|
|
|
|
|
|
|
#define ICE_UP_TABLE_TRANSLATE(val, i) \
|
|
|
|
(((val) << ICE_AQ_VSI_UP_TABLE_UP##i##_S) & \
|
|
|
|
ICE_AQ_VSI_UP_TABLE_UP##i##_M)
|
|
|
|
|
2018-03-20 21:58:14 +07:00
|
|
|
#define ICE_TX_DESC(R, i) (&(((struct ice_tx_desc *)((R)->desc))[i]))
|
2018-03-20 21:58:13 +07:00
|
|
|
#define ICE_RX_DESC(R, i) (&(((union ice_32b_rx_flex_desc *)((R)->desc))[i]))
|
2018-03-20 21:58:15 +07:00
|
|
|
#define ICE_TX_CTX_DESC(R, i) (&(((struct ice_tx_ctx_desc *)((R)->desc))[i]))
|
2018-03-20 21:58:13 +07:00
|
|
|
|
2018-03-20 21:58:18 +07:00
|
|
|
/* Macro for each VSI in a PF */
|
|
|
|
#define ice_for_each_vsi(pf, i) \
|
|
|
|
for ((i) = 0; (i) < (pf)->num_alloc_vsi; (i)++)
|
|
|
|
|
|
|
|
/* Macros for each tx/rx ring in a VSI */
|
2018-03-20 21:58:13 +07:00
|
|
|
#define ice_for_each_txq(vsi, i) \
|
|
|
|
for ((i) = 0; (i) < (vsi)->num_txq; (i)++)
|
|
|
|
|
|
|
|
#define ice_for_each_rxq(vsi, i) \
|
|
|
|
for ((i) = 0; (i) < (vsi)->num_rxq; (i)++)
|
|
|
|
|
ice: Report stats for allocated queues via ethtool stats
It is not safe to have the string table for statistics change order or
size over the lifetime of a given netdevice. This is because of the
nature of the 3-step process for obtaining stats. First, user space
performs a request for the size of the strings table. Second it performs
a separate request for the strings themselves, after allocating space
for the table. Third, it requests the stats themselves, also allocating
space for the table.
If the size decreased, there is potential to see garbage data or stats
values. In the worst case, we could potentially see stats values become
mis-aligned with their strings, so that it looks like a statistic is
being reported differently than it actually is.
Even worse, if the size increased, there is potential that the strings
table or stats table was not allocated large enough and the stats code
could access and write to memory it should not, potentially resulting in
undefined behavior and system crashes.
It isn't even safe if the size always changes under the RTNL lock. This
is because the calls take place over multiple user space commands, so it
is not possible to hold the RTNL lock for the entire duration of
obtaining strings and stats. Further, not all consumers of the ethtool
API are the user space ethtool program, and it is possible that one
assumes the strings will not change (valid under the current contract),
and thus only requests the stats values when requesting stats in a loop.
Finally, it's not possible in the general case to detect when the size
changes, because it is quite possible that one value which could impact
the stat size increased, while another decreased. This would result in
the same total number of stats, but reordering them so that stats no
longer line up with the strings they belong to. Since only size changes
aren't enough, we would need some sort of hash or token to determine
when the strings no longer match. This would require extending the
ethtool stats commands, but there is no more space in the relevant
structures.
The real solution to resolve this would be to add a completely new API
for stats, probably over netlink.
In the ice driver, the only thing impacting the stats that is not
constant is the number of queues. Instead of reporting stats for each
used queue, report stats for each allocated queue. We do not change the
number of queues allocated for a given netdevice, as we pass this into
the alloc_etherdev_mq() function to set the num_tx_queues and
num_rx_queues.
This resolves the potential bugs at the slight cost of displaying many
queue statistics which will not be activated.
Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Tested-by: Tony Brelinski <tonyx.brelinski@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-08-09 20:28:54 +07:00
|
|
|
/* Macros for each allocated tx/rx ring whether used or not in a VSI */
|
|
|
|
#define ice_for_each_alloc_txq(vsi, i) \
|
|
|
|
for ((i) = 0; (i) < (vsi)->alloc_txq; (i)++)
|
|
|
|
|
|
|
|
#define ice_for_each_alloc_rxq(vsi, i) \
|
|
|
|
for ((i) = 0; (i) < (vsi)->alloc_rxq; (i)++)
|
|
|
|
|
2018-03-20 21:58:11 +07:00
|
|
|
struct ice_tc_info {
|
|
|
|
u16 qoffset;
|
2018-10-27 01:44:35 +07:00
|
|
|
u16 qcount_tx;
|
|
|
|
u16 qcount_rx;
|
|
|
|
u8 netdev_tc;
|
2018-03-20 21:58:11 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ice_tc_cfg {
|
|
|
|
u8 numtc; /* Total number of enabled TCs */
|
|
|
|
u8 ena_tc; /* TX map */
|
|
|
|
struct ice_tc_info tc_info[ICE_MAX_TRAFFIC_CLASS];
|
|
|
|
};
|
|
|
|
|
2018-03-20 21:58:10 +07:00
|
|
|
struct ice_res_tracker {
|
|
|
|
u16 num_entries;
|
|
|
|
u16 search_hint;
|
|
|
|
u16 list[1];
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ice_sw {
|
|
|
|
struct ice_pf *pf;
|
|
|
|
u16 sw_id; /* switch ID for this switch */
|
|
|
|
u16 bridge_mode; /* VEB/VEPA/Port Virtualizer */
|
|
|
|
};
|
|
|
|
|
2018-03-20 21:58:05 +07:00
|
|
|
enum ice_state {
|
|
|
|
__ICE_DOWN,
|
2018-03-20 21:58:18 +07:00
|
|
|
__ICE_NEEDS_RESTART,
|
2018-08-09 20:29:50 +07:00
|
|
|
__ICE_PREPARED_FOR_RESET, /* set by driver when prepared */
|
2018-09-20 07:23:11 +07:00
|
|
|
__ICE_RESET_OICR_RECV, /* set by driver after rcv reset OICR */
|
2018-03-20 21:58:10 +07:00
|
|
|
__ICE_PFR_REQ, /* set by driver and peers */
|
2018-03-20 21:58:18 +07:00
|
|
|
__ICE_CORER_REQ, /* set by driver and peers */
|
|
|
|
__ICE_GLOBR_REQ, /* set by driver and peers */
|
|
|
|
__ICE_CORER_RECV, /* set by OICR handler */
|
|
|
|
__ICE_GLOBR_RECV, /* set by OICR handler */
|
|
|
|
__ICE_EMPR_RECV, /* set by OICR handler */
|
|
|
|
__ICE_SUSPENDED, /* set on module remove path */
|
|
|
|
__ICE_RESET_FAILED, /* set by reset/rebuild */
|
2018-09-20 07:42:55 +07:00
|
|
|
/* When checking for the PF to be in a nominal operating state, the
|
|
|
|
* bits that are grouped at the beginning of the list need to be
|
|
|
|
* checked. Bits occurring before __ICE_STATE_NOMINAL_CHECK_BITS will
|
|
|
|
* be checked. If you need to add a bit into consideration for nominal
|
|
|
|
* operating state, it must be added before
|
|
|
|
* __ICE_STATE_NOMINAL_CHECK_BITS. Do not move this entry's position
|
|
|
|
* without appropriate consideration.
|
|
|
|
*/
|
|
|
|
__ICE_STATE_NOMINAL_CHECK_BITS,
|
2018-03-20 21:58:10 +07:00
|
|
|
__ICE_ADMINQ_EVENT_PENDING,
|
2018-09-20 07:42:54 +07:00
|
|
|
__ICE_MAILBOXQ_EVENT_PENDING,
|
2018-08-09 20:29:53 +07:00
|
|
|
__ICE_MDD_EVENT_PENDING,
|
2018-09-20 07:42:57 +07:00
|
|
|
__ICE_VFLR_EVENT_PENDING,
|
2018-03-20 21:58:19 +07:00
|
|
|
__ICE_FLTR_OVERFLOW_PROMISC,
|
2018-09-20 07:42:55 +07:00
|
|
|
__ICE_VF_DIS,
|
2018-03-20 21:58:16 +07:00
|
|
|
__ICE_CFG_BUSY,
|
2018-03-20 21:58:10 +07:00
|
|
|
__ICE_SERVICE_SCHED,
|
2018-08-09 20:29:57 +07:00
|
|
|
__ICE_SERVICE_DIS,
|
2018-03-20 21:58:05 +07:00
|
|
|
__ICE_STATE_NBITS /* must be last */
|
|
|
|
};
|
|
|
|
|
2018-03-20 21:58:19 +07:00
|
|
|
enum ice_vsi_flags {
|
|
|
|
ICE_VSI_FLAG_UMAC_FLTR_CHANGED,
|
|
|
|
ICE_VSI_FLAG_MMAC_FLTR_CHANGED,
|
|
|
|
ICE_VSI_FLAG_VLAN_FLTR_CHANGED,
|
|
|
|
ICE_VSI_FLAG_PROMISC_CHANGED,
|
|
|
|
ICE_VSI_FLAG_NBITS /* must be last */
|
|
|
|
};
|
|
|
|
|
2018-03-20 21:58:10 +07:00
|
|
|
/* struct that defines a VSI, associated with a dev */
|
|
|
|
struct ice_vsi {
|
|
|
|
struct net_device *netdev;
|
2018-03-20 21:58:11 +07:00
|
|
|
struct ice_sw *vsw; /* switch this VSI is on */
|
|
|
|
struct ice_pf *back; /* back pointer to PF */
|
2018-03-20 21:58:10 +07:00
|
|
|
struct ice_port_info *port_info; /* back pointer to port_info */
|
2018-03-20 21:58:11 +07:00
|
|
|
struct ice_ring **rx_rings; /* rx ring array */
|
|
|
|
struct ice_ring **tx_rings; /* tx ring array */
|
|
|
|
struct ice_q_vector **q_vectors; /* q_vector array */
|
2018-03-20 21:58:13 +07:00
|
|
|
|
|
|
|
irqreturn_t (*irq_handler)(int irq, void *data);
|
|
|
|
|
2018-03-20 21:58:16 +07:00
|
|
|
u64 tx_linearize;
|
2018-03-20 21:58:11 +07:00
|
|
|
DECLARE_BITMAP(state, __ICE_STATE_NBITS);
|
2018-03-20 21:58:19 +07:00
|
|
|
DECLARE_BITMAP(flags, ICE_VSI_FLAG_NBITS);
|
|
|
|
unsigned int current_netdev_flags;
|
2018-03-20 21:58:16 +07:00
|
|
|
u32 tx_restart;
|
|
|
|
u32 tx_busy;
|
|
|
|
u32 rx_buf_failed;
|
|
|
|
u32 rx_page_failed;
|
2018-03-20 21:58:11 +07:00
|
|
|
int num_q_vectors;
|
ice: Split irq_tracker into sw_irq_tracker and hw_irq_tracker
For the PF driver, when mapping interrupts to queues, we need to request
IRQs from the kernel and we also have to allocate interrupts from
the device.
Similarly, when the VF driver (iavf.ko) initializes, it requests the kernel
IRQs that it needs but it can't directly allocate interrupts in the device.
Instead, it sends a mailbox message to the ice driver, which then allocates
interrupts in the device on the VF driver's behalf.
Currently both these cases end up having to reserve entries in
pf->irq_tracker but irq_tracker itself is sized based on how many vectors
the PF driver needs. Under the right circumstances, the VF driver can fail
to get entries in irq_tracker, which will result in the VF driver failing
probe.
To fix this, sw_irq_tracker and hw_irq_tracker are introduced. The
sw_irq_tracker tracks only the PF's IRQ request and doesn't play any
role in VF init. hw_irq_tracker represents the device's interrupt space.
When interrupts have to be allocated in the device for either PF or VF,
hw_irq_tracker will be looked up to see if the device has run out of
interrupts.
Signed-off-by: Preethi Banala <preethi.banala@intel.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-09-20 07:23:16 +07:00
|
|
|
int sw_base_vector; /* Irq base for OS reserved vectors */
|
|
|
|
int hw_base_vector; /* HW (absolute) index of a vector */
|
2018-03-20 21:58:11 +07:00
|
|
|
enum ice_vsi_type type;
|
2018-03-20 21:58:10 +07:00
|
|
|
u16 vsi_num; /* HW (absolute) index of this VSI */
|
2018-03-20 21:58:11 +07:00
|
|
|
u16 idx; /* software index in pf->vsi[] */
|
|
|
|
|
|
|
|
/* Interrupt thresholds */
|
|
|
|
u16 work_lmt;
|
|
|
|
|
2018-09-20 07:42:56 +07:00
|
|
|
s16 vf_id; /* VF ID for SR-IOV VSIs */
|
|
|
|
|
2018-03-20 21:58:15 +07:00
|
|
|
/* RSS config */
|
|
|
|
u16 rss_table_size; /* HW RSS table size */
|
|
|
|
u16 rss_size; /* Allocated RSS queues */
|
|
|
|
u8 *rss_hkey_user; /* User configured hash keys */
|
|
|
|
u8 *rss_lut_user; /* User configured lookup table entries */
|
|
|
|
u8 rss_lut_type; /* used to configure Get/Set RSS LUT AQ call */
|
|
|
|
|
2018-03-20 21:58:13 +07:00
|
|
|
u16 max_frame;
|
|
|
|
u16 rx_buf_len;
|
|
|
|
|
2018-03-20 21:58:11 +07:00
|
|
|
struct ice_aqc_vsi_props info; /* VSI properties */
|
|
|
|
|
2018-03-20 21:58:16 +07:00
|
|
|
/* VSI stats */
|
|
|
|
struct rtnl_link_stats64 net_stats;
|
|
|
|
struct ice_eth_stats eth_stats;
|
|
|
|
struct ice_eth_stats eth_stats_prev;
|
|
|
|
|
2018-03-20 21:58:19 +07:00
|
|
|
struct list_head tmp_sync_list; /* MAC filters to be synced */
|
|
|
|
struct list_head tmp_unsync_list; /* MAC filters to be unsynced */
|
|
|
|
|
2018-08-09 20:29:02 +07:00
|
|
|
u8 irqs_ready;
|
|
|
|
u8 current_isup; /* Sync 'link up' logging */
|
|
|
|
u8 stat_offsets_loaded;
|
2018-03-20 21:58:13 +07:00
|
|
|
|
2018-03-20 21:58:11 +07:00
|
|
|
/* queue information */
|
|
|
|
u8 tx_mapping_mode; /* ICE_MAP_MODE_[CONTIG|SCATTER] */
|
|
|
|
u8 rx_mapping_mode; /* ICE_MAP_MODE_[CONTIG|SCATTER] */
|
|
|
|
u16 txq_map[ICE_MAX_TXQS]; /* index in pf->avail_txqs */
|
|
|
|
u16 rxq_map[ICE_MAX_RXQS]; /* index in pf->avail_rxqs */
|
|
|
|
u16 alloc_txq; /* Allocated Tx queues */
|
|
|
|
u16 num_txq; /* Used Tx queues */
|
|
|
|
u16 alloc_rxq; /* Allocated Rx queues */
|
|
|
|
u16 num_rxq; /* Used Rx queues */
|
|
|
|
u16 num_desc;
|
|
|
|
struct ice_tc_cfg tc_cfg;
|
|
|
|
} ____cacheline_internodealigned_in_smp;
|
|
|
|
|
|
|
|
/* struct that defines an interrupt vector */
|
|
|
|
struct ice_q_vector {
|
|
|
|
struct ice_vsi *vsi;
|
|
|
|
cpumask_t affinity_mask;
|
|
|
|
struct napi_struct napi;
|
|
|
|
struct ice_ring_container rx;
|
|
|
|
struct ice_ring_container tx;
|
2018-03-20 21:58:13 +07:00
|
|
|
struct irq_affinity_notify affinity_notify;
|
2018-03-20 21:58:11 +07:00
|
|
|
u16 v_idx; /* index in the vsi->q_vector array. */
|
|
|
|
u8 num_ring_tx; /* total number of tx rings in vector */
|
|
|
|
u8 num_ring_rx; /* total number of rx rings in vector */
|
2018-03-20 21:58:13 +07:00
|
|
|
char name[ICE_INT_NAME_STR_LEN];
|
2018-09-20 07:23:19 +07:00
|
|
|
/* in usecs, need to use ice_intrl_to_usecs_reg() before writing this
|
|
|
|
* value to the device
|
|
|
|
*/
|
|
|
|
u8 intrl;
|
2018-03-20 21:58:10 +07:00
|
|
|
} ____cacheline_internodealigned_in_smp;
|
|
|
|
|
|
|
|
enum ice_pf_flags {
|
|
|
|
ICE_FLAG_MSIX_ENA,
|
|
|
|
ICE_FLAG_FLTR_SYNC,
|
|
|
|
ICE_FLAG_RSS_ENA,
|
2018-09-20 07:42:55 +07:00
|
|
|
ICE_FLAG_SRIOV_ENA,
|
2018-09-20 07:42:54 +07:00
|
|
|
ICE_FLAG_SRIOV_CAPABLE,
|
2018-03-20 21:58:10 +07:00
|
|
|
ICE_PF_FLAGS_NBITS /* must be last */
|
|
|
|
};
|
|
|
|
|
2018-03-20 21:58:05 +07:00
|
|
|
struct ice_pf {
|
|
|
|
struct pci_dev *pdev;
|
ice: Split irq_tracker into sw_irq_tracker and hw_irq_tracker
For the PF driver, when mapping interrupts to queues, we need to request
IRQs from the kernel and we also have to allocate interrupts from
the device.
Similarly, when the VF driver (iavf.ko) initializes, it requests the kernel
IRQs that it needs but it can't directly allocate interrupts in the device.
Instead, it sends a mailbox message to the ice driver, which then allocates
interrupts in the device on the VF driver's behalf.
Currently both these cases end up having to reserve entries in
pf->irq_tracker but irq_tracker itself is sized based on how many vectors
the PF driver needs. Under the right circumstances, the VF driver can fail
to get entries in irq_tracker, which will result in the VF driver failing
probe.
To fix this, sw_irq_tracker and hw_irq_tracker are introduced. The
sw_irq_tracker tracks only the PF's IRQ request and doesn't play any
role in VF init. hw_irq_tracker represents the device's interrupt space.
When interrupts have to be allocated in the device for either PF or VF,
hw_irq_tracker will be looked up to see if the device has run out of
interrupts.
Signed-off-by: Preethi Banala <preethi.banala@intel.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-09-20 07:23:16 +07:00
|
|
|
|
|
|
|
/* OS reserved IRQ details */
|
2018-03-20 21:58:10 +07:00
|
|
|
struct msix_entry *msix_entries;
|
ice: Split irq_tracker into sw_irq_tracker and hw_irq_tracker
For the PF driver, when mapping interrupts to queues, we need to request
IRQs from the kernel and we also have to allocate interrupts from
the device.
Similarly, when the VF driver (iavf.ko) initializes, it requests the kernel
IRQs that it needs but it can't directly allocate interrupts in the device.
Instead, it sends a mailbox message to the ice driver, which then allocates
interrupts in the device on the VF driver's behalf.
Currently both these cases end up having to reserve entries in
pf->irq_tracker but irq_tracker itself is sized based on how many vectors
the PF driver needs. Under the right circumstances, the VF driver can fail
to get entries in irq_tracker, which will result in the VF driver failing
probe.
To fix this, sw_irq_tracker and hw_irq_tracker are introduced. The
sw_irq_tracker tracks only the PF's IRQ request and doesn't play any
role in VF init. hw_irq_tracker represents the device's interrupt space.
When interrupts have to be allocated in the device for either PF or VF,
hw_irq_tracker will be looked up to see if the device has run out of
interrupts.
Signed-off-by: Preethi Banala <preethi.banala@intel.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-09-20 07:23:16 +07:00
|
|
|
struct ice_res_tracker *sw_irq_tracker;
|
|
|
|
|
|
|
|
/* HW reserved Interrupts for this PF */
|
|
|
|
struct ice_res_tracker *hw_irq_tracker;
|
|
|
|
|
2018-03-20 21:58:10 +07:00
|
|
|
struct ice_vsi **vsi; /* VSIs created by the driver */
|
|
|
|
struct ice_sw *first_sw; /* first switch created by firmware */
|
2018-09-20 07:42:55 +07:00
|
|
|
/* Virtchnl/SR-IOV config info */
|
|
|
|
struct ice_vf *vf;
|
|
|
|
int num_alloc_vfs; /* actual number of VFs allocated */
|
2018-09-20 07:42:54 +07:00
|
|
|
u16 num_vfs_supported; /* num VFs supported for this PF */
|
2018-09-20 07:42:55 +07:00
|
|
|
u16 num_vf_qps; /* num queue pairs per VF */
|
|
|
|
u16 num_vf_msix; /* num vectors per VF */
|
2018-03-20 21:58:05 +07:00
|
|
|
DECLARE_BITMAP(state, __ICE_STATE_NBITS);
|
2018-03-20 21:58:10 +07:00
|
|
|
DECLARE_BITMAP(avail_txqs, ICE_MAX_TXQS);
|
|
|
|
DECLARE_BITMAP(avail_rxqs, ICE_MAX_RXQS);
|
|
|
|
DECLARE_BITMAP(flags, ICE_PF_FLAGS_NBITS);
|
|
|
|
unsigned long serv_tmr_period;
|
|
|
|
unsigned long serv_tmr_prev;
|
|
|
|
struct timer_list serv_tmr;
|
|
|
|
struct work_struct serv_task;
|
|
|
|
struct mutex avail_q_mutex; /* protects access to avail_[rx|tx]qs */
|
|
|
|
struct mutex sw_mutex; /* lock for protecting VSI alloc flow */
|
2018-03-20 21:58:05 +07:00
|
|
|
u32 msg_enable;
|
2018-03-20 21:58:15 +07:00
|
|
|
u32 hw_csum_rx_error;
|
ice: Split irq_tracker into sw_irq_tracker and hw_irq_tracker
For the PF driver, when mapping interrupts to queues, we need to request
IRQs from the kernel and we also have to allocate interrupts from
the device.
Similarly, when the VF driver (iavf.ko) initializes, it requests the kernel
IRQs that it needs but it can't directly allocate interrupts in the device.
Instead, it sends a mailbox message to the ice driver, which then allocates
interrupts in the device on the VF driver's behalf.
Currently both these cases end up having to reserve entries in
pf->irq_tracker but irq_tracker itself is sized based on how many vectors
the PF driver needs. Under the right circumstances, the VF driver can fail
to get entries in irq_tracker, which will result in the VF driver failing
probe.
To fix this, sw_irq_tracker and hw_irq_tracker are introduced. The
sw_irq_tracker tracks only the PF's IRQ request and doesn't play any
role in VF init. hw_irq_tracker represents the device's interrupt space.
When interrupts have to be allocated in the device for either PF or VF,
hw_irq_tracker will be looked up to see if the device has run out of
interrupts.
Signed-off-by: Preethi Banala <preethi.banala@intel.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-09-20 07:23:16 +07:00
|
|
|
u32 sw_oicr_idx; /* Other interrupt cause SW vector index */
|
|
|
|
u32 num_avail_sw_msix; /* remaining MSIX SW vectors left unclaimed */
|
|
|
|
u32 hw_oicr_idx; /* Other interrupt cause vector HW index */
|
|
|
|
u32 num_avail_hw_msix; /* remaining HW MSIX vectors left unclaimed */
|
2018-03-20 21:58:10 +07:00
|
|
|
u32 num_lan_msix; /* Total MSIX vectors for base driver */
|
|
|
|
u16 num_lan_tx; /* num lan tx queues setup */
|
|
|
|
u16 num_lan_rx; /* num lan rx queues setup */
|
|
|
|
u16 q_left_tx; /* remaining num tx queues left unclaimed */
|
|
|
|
u16 q_left_rx; /* remaining num rx queues left unclaimed */
|
|
|
|
u16 next_vsi; /* Next free slot in pf->vsi[] - 0-based! */
|
|
|
|
u16 num_alloc_vsi;
|
2018-03-20 21:58:18 +07:00
|
|
|
u16 corer_count; /* Core reset count */
|
|
|
|
u16 globr_count; /* Global reset count */
|
|
|
|
u16 empr_count; /* EMP reset count */
|
|
|
|
u16 pfr_count; /* PF reset count */
|
|
|
|
|
2018-03-20 21:58:16 +07:00
|
|
|
struct ice_hw_port_stats stats;
|
|
|
|
struct ice_hw_port_stats stats_prev;
|
2018-03-20 21:58:05 +07:00
|
|
|
struct ice_hw hw;
|
2018-08-09 20:29:02 +07:00
|
|
|
u8 stat_prev_loaded; /* has previous stats been loaded */
|
2018-08-09 20:29:53 +07:00
|
|
|
u32 tx_timeout_count;
|
|
|
|
unsigned long tx_timeout_last_recovery;
|
|
|
|
u32 tx_timeout_recovery_level;
|
2018-03-20 21:58:10 +07:00
|
|
|
char int_name[ICE_INT_NAME_STR_LEN];
|
2018-03-20 21:58:05 +07:00
|
|
|
};
|
2018-03-20 21:58:10 +07:00
|
|
|
|
2018-03-20 21:58:11 +07:00
|
|
|
struct ice_netdev_priv {
|
|
|
|
struct ice_vsi *vsi;
|
|
|
|
};
|
|
|
|
|
2018-03-20 21:58:10 +07:00
|
|
|
/**
|
|
|
|
* ice_irq_dynamic_ena - Enable default interrupt generation settings
|
|
|
|
* @hw: pointer to hw struct
|
2018-03-20 21:58:13 +07:00
|
|
|
* @vsi: pointer to vsi struct, can be NULL
|
|
|
|
* @q_vector: pointer to q_vector, can be NULL
|
2018-03-20 21:58:10 +07:00
|
|
|
*/
|
2018-03-20 21:58:13 +07:00
|
|
|
static inline void ice_irq_dynamic_ena(struct ice_hw *hw, struct ice_vsi *vsi,
|
|
|
|
struct ice_q_vector *q_vector)
|
2018-03-20 21:58:10 +07:00
|
|
|
{
|
ice: Split irq_tracker into sw_irq_tracker and hw_irq_tracker
For the PF driver, when mapping interrupts to queues, we need to request
IRQs from the kernel and we also have to allocate interrupts from
the device.
Similarly, when the VF driver (iavf.ko) initializes, it requests the kernel
IRQs that it needs but it can't directly allocate interrupts in the device.
Instead, it sends a mailbox message to the ice driver, which then allocates
interrupts in the device on the VF driver's behalf.
Currently both these cases end up having to reserve entries in
pf->irq_tracker but irq_tracker itself is sized based on how many vectors
the PF driver needs. Under the right circumstances, the VF driver can fail
to get entries in irq_tracker, which will result in the VF driver failing
probe.
To fix this, sw_irq_tracker and hw_irq_tracker are introduced. The
sw_irq_tracker tracks only the PF's IRQ request and doesn't play any
role in VF init. hw_irq_tracker represents the device's interrupt space.
When interrupts have to be allocated in the device for either PF or VF,
hw_irq_tracker will be looked up to see if the device has run out of
interrupts.
Signed-off-by: Preethi Banala <preethi.banala@intel.com>
Signed-off-by: Anirudh Venkataramanan <anirudh.venkataramanan@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2018-09-20 07:23:16 +07:00
|
|
|
u32 vector = (vsi && q_vector) ? vsi->hw_base_vector + q_vector->v_idx :
|
|
|
|
((struct ice_pf *)hw->back)->hw_oicr_idx;
|
2018-03-20 21:58:10 +07:00
|
|
|
int itr = ICE_ITR_NONE;
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
/* clear the PBA here, as this function is meant to clean out all
|
|
|
|
* previous interrupts and enable the interrupt
|
|
|
|
*/
|
|
|
|
val = GLINT_DYN_CTL_INTENA_M | GLINT_DYN_CTL_CLEARPBA_M |
|
|
|
|
(itr << GLINT_DYN_CTL_ITR_INDX_S);
|
2018-03-20 21:58:13 +07:00
|
|
|
if (vsi)
|
|
|
|
if (test_bit(__ICE_DOWN, vsi->state))
|
|
|
|
return;
|
2018-03-20 21:58:10 +07:00
|
|
|
wr32(hw, GLINT_DYN_CTL(vector), val);
|
|
|
|
}
|
2018-03-20 21:58:13 +07:00
|
|
|
|
2018-03-20 21:58:17 +07:00
|
|
|
static inline void ice_vsi_set_tc_cfg(struct ice_vsi *vsi)
|
|
|
|
{
|
|
|
|
vsi->tc_cfg.ena_tc = ICE_DFLT_TRAFFIC_CLASS;
|
|
|
|
vsi->tc_cfg.numtc = 1;
|
|
|
|
}
|
|
|
|
|
2018-03-20 21:58:16 +07:00
|
|
|
void ice_set_ethtool_ops(struct net_device *netdev);
|
|
|
|
int ice_up(struct ice_vsi *vsi);
|
|
|
|
int ice_down(struct ice_vsi *vsi);
|
2018-03-20 21:58:15 +07:00
|
|
|
int ice_set_rss(struct ice_vsi *vsi, u8 *seed, u8 *lut, u16 lut_size);
|
|
|
|
int ice_get_rss(struct ice_vsi *vsi, u8 *seed, u8 *lut, u16 lut_size);
|
|
|
|
void ice_fill_rss_lut(u8 *lut, u16 rss_table_size, u16 rss_size);
|
2018-03-20 21:58:16 +07:00
|
|
|
void ice_print_link_msg(struct ice_vsi *vsi, bool isup);
|
2018-10-27 00:40:57 +07:00
|
|
|
void ice_napi_del(struct ice_vsi *vsi);
|
2018-03-20 21:58:15 +07:00
|
|
|
|
2018-03-20 21:58:05 +07:00
|
|
|
#endif /* _ICE_H_ */
|