mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-14 09:44:35 +08:00
a7e117109a
The vc_ops structure is used to allow different handlers for virtchnl commands when the driver is in representor mode. The current implementation uses a copy of the ops table in each VF, and modifies this copy dynamically. The usual practice in kernel code is to store the ops table in a constant structure and point to different versions. This has a number of advantages: 1. Reduced memory usage. Each VF merely points to the correct table, so they're able to re-use the same constant lookup table in memory. 2. Consistency. It becomes more difficult to accidentally update or edit only one op call. Instead, the code switches to the correct able by a single pointer write. In general this is atomic, either the pointer is updated or its not. 3. Code Layout. The VF structure can store a pointer to the table without needing to have the full structure definition defined prior to the VF structure definition. This will aid in future refactoring of code by allowing the VF pointer to be kept in ice_vf_lib.h while the virtchnl ops table can be maintained in ice_virtchnl.h There is one major downside in the case of the vc_ops structure. Most of the operations in the table are the same between the two current implementations. This can appear to lead to duplication since each implementation must now fill in the complete table. It could make spotting the differences in the representor mode more challenging. Unfortunately, methods to make this less error prone either add complexity overhead (macros using CPP token concatenation) or don't work on all compilers we support (constant initializer from another constant structure). The cost of maintaining two structures does not out weigh the benefits of the constant table model. While we're making these changes, go ahead and rename the structure and implementations with "virtchnl" instead of "vc_vf_". This will more closely align with the planned file renaming, and avoid similar names when we later introduce a "vf ops" table for separating Scalable IOV and Single Root IOV implementations. Leave the accessor/assignment functions in order to avoid issues with compiling with options disabled. The interface makes it easier to handle when CONFIG_PCI_IOV is disabled in the kernel. Signed-off-by: Jacob Keller <jacob.e.keller@intel.com> Tested-by: Sandeep Penigalapati <sandeep.penigalapati@intel.com> Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com>
462 lines
11 KiB
C
462 lines
11 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* Copyright (C) 2019-2021, Intel Corporation. */
|
|
|
|
#include "ice.h"
|
|
#include "ice_eswitch.h"
|
|
#include "ice_devlink.h"
|
|
#include "ice_sriov.h"
|
|
#include "ice_tc_lib.h"
|
|
|
|
/**
|
|
* ice_repr_get_sw_port_id - get port ID associated with representor
|
|
* @repr: pointer to port representor
|
|
*/
|
|
static int ice_repr_get_sw_port_id(struct ice_repr *repr)
|
|
{
|
|
return repr->vf->pf->hw.port_info->lport;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_get_phys_port_name - get phys port name
|
|
* @netdev: pointer to port representor netdev
|
|
* @buf: write here port name
|
|
* @len: max length of buf
|
|
*/
|
|
static int
|
|
ice_repr_get_phys_port_name(struct net_device *netdev, char *buf, size_t len)
|
|
{
|
|
struct ice_netdev_priv *np = netdev_priv(netdev);
|
|
struct ice_repr *repr = np->repr;
|
|
int res;
|
|
|
|
/* Devlink port is registered and devlink core is taking care of name formatting. */
|
|
if (repr->vf->devlink_port.devlink)
|
|
return -EOPNOTSUPP;
|
|
|
|
res = snprintf(buf, len, "pf%dvfr%d", ice_repr_get_sw_port_id(repr),
|
|
repr->vf->vf_id);
|
|
if (res <= 0)
|
|
return -EOPNOTSUPP;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_get_stats64 - get VF stats for VFPR use
|
|
* @netdev: pointer to port representor netdev
|
|
* @stats: pointer to struct where stats can be stored
|
|
*/
|
|
static void
|
|
ice_repr_get_stats64(struct net_device *netdev, struct rtnl_link_stats64 *stats)
|
|
{
|
|
struct ice_netdev_priv *np = netdev_priv(netdev);
|
|
struct ice_eth_stats *eth_stats;
|
|
struct ice_vsi *vsi;
|
|
|
|
if (ice_is_vf_disabled(np->repr->vf))
|
|
return;
|
|
vsi = np->repr->src_vsi;
|
|
|
|
ice_update_vsi_stats(vsi);
|
|
eth_stats = &vsi->eth_stats;
|
|
|
|
stats->tx_packets = eth_stats->tx_unicast + eth_stats->tx_broadcast +
|
|
eth_stats->tx_multicast;
|
|
stats->rx_packets = eth_stats->rx_unicast + eth_stats->rx_broadcast +
|
|
eth_stats->rx_multicast;
|
|
stats->tx_bytes = eth_stats->tx_bytes;
|
|
stats->rx_bytes = eth_stats->rx_bytes;
|
|
stats->multicast = eth_stats->rx_multicast;
|
|
stats->tx_errors = eth_stats->tx_errors;
|
|
stats->tx_dropped = eth_stats->tx_discards;
|
|
stats->rx_dropped = eth_stats->rx_discards;
|
|
}
|
|
|
|
/**
|
|
* ice_netdev_to_repr - Get port representor for given netdevice
|
|
* @netdev: pointer to port representor netdev
|
|
*/
|
|
struct ice_repr *ice_netdev_to_repr(struct net_device *netdev)
|
|
{
|
|
struct ice_netdev_priv *np = netdev_priv(netdev);
|
|
|
|
return np->repr;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_open - Enable port representor's network interface
|
|
* @netdev: network interface device structure
|
|
*
|
|
* The open entry point is called when a port representor's network
|
|
* interface is made active by the system (IFF_UP). Corresponding
|
|
* VF is notified about link status change.
|
|
*
|
|
* Returns 0 on success
|
|
*/
|
|
static int ice_repr_open(struct net_device *netdev)
|
|
{
|
|
struct ice_repr *repr = ice_netdev_to_repr(netdev);
|
|
struct ice_vf *vf;
|
|
|
|
vf = repr->vf;
|
|
vf->link_forced = true;
|
|
vf->link_up = true;
|
|
ice_vc_notify_vf_link_state(vf);
|
|
|
|
netif_carrier_on(netdev);
|
|
netif_tx_start_all_queues(netdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_stop - Disable port representor's network interface
|
|
* @netdev: network interface device structure
|
|
*
|
|
* The stop entry point is called when a port representor's network
|
|
* interface is de-activated by the system. Corresponding
|
|
* VF is notified about link status change.
|
|
*
|
|
* Returns 0 on success
|
|
*/
|
|
static int ice_repr_stop(struct net_device *netdev)
|
|
{
|
|
struct ice_repr *repr = ice_netdev_to_repr(netdev);
|
|
struct ice_vf *vf;
|
|
|
|
vf = repr->vf;
|
|
vf->link_forced = true;
|
|
vf->link_up = false;
|
|
ice_vc_notify_vf_link_state(vf);
|
|
|
|
netif_carrier_off(netdev);
|
|
netif_tx_stop_all_queues(netdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct devlink_port *
|
|
ice_repr_get_devlink_port(struct net_device *netdev)
|
|
{
|
|
struct ice_repr *repr = ice_netdev_to_repr(netdev);
|
|
|
|
return &repr->vf->devlink_port;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_sp_stats64 - get slow path stats for port representor
|
|
* @dev: network interface device structure
|
|
* @stats: netlink stats structure
|
|
*
|
|
* RX/TX stats are being swapped here to be consistent with VF stats. In slow
|
|
* path, port representor receives data when the corresponding VF is sending it
|
|
* (and vice versa), TX and RX bytes/packets are effectively swapped on port
|
|
* representor.
|
|
*/
|
|
static int
|
|
ice_repr_sp_stats64(const struct net_device *dev,
|
|
struct rtnl_link_stats64 *stats)
|
|
{
|
|
struct ice_netdev_priv *np = netdev_priv(dev);
|
|
int vf_id = np->repr->vf->vf_id;
|
|
struct ice_tx_ring *tx_ring;
|
|
struct ice_rx_ring *rx_ring;
|
|
u64 pkts, bytes;
|
|
|
|
tx_ring = np->vsi->tx_rings[vf_id];
|
|
ice_fetch_u64_stats_per_ring(&tx_ring->syncp, tx_ring->stats,
|
|
&pkts, &bytes);
|
|
stats->rx_packets = pkts;
|
|
stats->rx_bytes = bytes;
|
|
|
|
rx_ring = np->vsi->rx_rings[vf_id];
|
|
ice_fetch_u64_stats_per_ring(&rx_ring->syncp, rx_ring->stats,
|
|
&pkts, &bytes);
|
|
stats->tx_packets = pkts;
|
|
stats->tx_bytes = bytes;
|
|
stats->tx_dropped = rx_ring->rx_stats.alloc_page_failed +
|
|
rx_ring->rx_stats.alloc_buf_failed;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static bool
|
|
ice_repr_ndo_has_offload_stats(const struct net_device *dev, int attr_id)
|
|
{
|
|
return attr_id == IFLA_OFFLOAD_XSTATS_CPU_HIT;
|
|
}
|
|
|
|
static int
|
|
ice_repr_ndo_get_offload_stats(int attr_id, const struct net_device *dev,
|
|
void *sp)
|
|
{
|
|
if (attr_id == IFLA_OFFLOAD_XSTATS_CPU_HIT)
|
|
return ice_repr_sp_stats64(dev, (struct rtnl_link_stats64 *)sp);
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int
|
|
ice_repr_setup_tc_cls_flower(struct ice_repr *repr,
|
|
struct flow_cls_offload *flower)
|
|
{
|
|
switch (flower->command) {
|
|
case FLOW_CLS_REPLACE:
|
|
return ice_add_cls_flower(repr->netdev, repr->src_vsi, flower);
|
|
case FLOW_CLS_DESTROY:
|
|
return ice_del_cls_flower(repr->src_vsi, flower);
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
static int
|
|
ice_repr_setup_tc_block_cb(enum tc_setup_type type, void *type_data,
|
|
void *cb_priv)
|
|
{
|
|
struct flow_cls_offload *flower = (struct flow_cls_offload *)type_data;
|
|
struct ice_netdev_priv *np = (struct ice_netdev_priv *)cb_priv;
|
|
|
|
switch (type) {
|
|
case TC_SETUP_CLSFLOWER:
|
|
return ice_repr_setup_tc_cls_flower(np->repr, flower);
|
|
default:
|
|
return -EOPNOTSUPP;
|
|
}
|
|
}
|
|
|
|
static LIST_HEAD(ice_repr_block_cb_list);
|
|
|
|
static int
|
|
ice_repr_setup_tc(struct net_device *netdev, enum tc_setup_type type,
|
|
void *type_data)
|
|
{
|
|
struct ice_netdev_priv *np = netdev_priv(netdev);
|
|
|
|
switch (type) {
|
|
case TC_SETUP_BLOCK:
|
|
return flow_block_cb_setup_simple((struct flow_block_offload *)
|
|
type_data,
|
|
&ice_repr_block_cb_list,
|
|
ice_repr_setup_tc_block_cb,
|
|
np, np, true);
|
|
default:
|
|
return -EOPNOTSUPP;
|
|
}
|
|
}
|
|
|
|
static const struct net_device_ops ice_repr_netdev_ops = {
|
|
.ndo_get_phys_port_name = ice_repr_get_phys_port_name,
|
|
.ndo_get_stats64 = ice_repr_get_stats64,
|
|
.ndo_open = ice_repr_open,
|
|
.ndo_stop = ice_repr_stop,
|
|
.ndo_start_xmit = ice_eswitch_port_start_xmit,
|
|
.ndo_get_devlink_port = ice_repr_get_devlink_port,
|
|
.ndo_setup_tc = ice_repr_setup_tc,
|
|
.ndo_has_offload_stats = ice_repr_ndo_has_offload_stats,
|
|
.ndo_get_offload_stats = ice_repr_ndo_get_offload_stats,
|
|
};
|
|
|
|
/**
|
|
* ice_is_port_repr_netdev - Check if a given netdevice is a port representor netdev
|
|
* @netdev: pointer to netdev
|
|
*/
|
|
bool ice_is_port_repr_netdev(struct net_device *netdev)
|
|
{
|
|
return netdev && (netdev->netdev_ops == &ice_repr_netdev_ops);
|
|
}
|
|
|
|
/**
|
|
* ice_repr_reg_netdev - register port representor netdev
|
|
* @netdev: pointer to port representor netdev
|
|
*/
|
|
static int
|
|
ice_repr_reg_netdev(struct net_device *netdev)
|
|
{
|
|
eth_hw_addr_random(netdev);
|
|
netdev->netdev_ops = &ice_repr_netdev_ops;
|
|
ice_set_ethtool_repr_ops(netdev);
|
|
|
|
netdev->hw_features |= NETIF_F_HW_TC;
|
|
|
|
netif_carrier_off(netdev);
|
|
netif_tx_stop_all_queues(netdev);
|
|
|
|
return register_netdev(netdev);
|
|
}
|
|
|
|
/**
|
|
* ice_repr_add - add representor for VF
|
|
* @vf: pointer to VF structure
|
|
*/
|
|
static int ice_repr_add(struct ice_vf *vf)
|
|
{
|
|
struct ice_q_vector *q_vector;
|
|
struct ice_netdev_priv *np;
|
|
struct ice_repr *repr;
|
|
int err;
|
|
|
|
repr = kzalloc(sizeof(*repr), GFP_KERNEL);
|
|
if (!repr)
|
|
return -ENOMEM;
|
|
|
|
#ifdef CONFIG_ICE_SWITCHDEV
|
|
repr->mac_rule = kzalloc(sizeof(*repr->mac_rule), GFP_KERNEL);
|
|
if (!repr->mac_rule) {
|
|
err = -ENOMEM;
|
|
goto err_alloc_rule;
|
|
}
|
|
#endif
|
|
|
|
repr->netdev = alloc_etherdev(sizeof(struct ice_netdev_priv));
|
|
if (!repr->netdev) {
|
|
err = -ENOMEM;
|
|
goto err_alloc;
|
|
}
|
|
|
|
repr->src_vsi = ice_get_vf_vsi(vf);
|
|
repr->vf = vf;
|
|
vf->repr = repr;
|
|
np = netdev_priv(repr->netdev);
|
|
np->repr = repr;
|
|
|
|
q_vector = kzalloc(sizeof(*q_vector), GFP_KERNEL);
|
|
if (!q_vector) {
|
|
err = -ENOMEM;
|
|
goto err_alloc_q_vector;
|
|
}
|
|
repr->q_vector = q_vector;
|
|
|
|
err = ice_devlink_create_vf_port(vf);
|
|
if (err)
|
|
goto err_devlink;
|
|
|
|
repr->netdev->min_mtu = ETH_MIN_MTU;
|
|
repr->netdev->max_mtu = ICE_MAX_MTU;
|
|
|
|
err = ice_repr_reg_netdev(repr->netdev);
|
|
if (err)
|
|
goto err_netdev;
|
|
|
|
devlink_port_type_eth_set(&vf->devlink_port, repr->netdev);
|
|
|
|
ice_virtchnl_set_repr_ops(vf);
|
|
|
|
return 0;
|
|
|
|
err_netdev:
|
|
ice_devlink_destroy_vf_port(vf);
|
|
err_devlink:
|
|
kfree(repr->q_vector);
|
|
vf->repr->q_vector = NULL;
|
|
err_alloc_q_vector:
|
|
free_netdev(repr->netdev);
|
|
repr->netdev = NULL;
|
|
err_alloc:
|
|
#ifdef CONFIG_ICE_SWITCHDEV
|
|
kfree(repr->mac_rule);
|
|
repr->mac_rule = NULL;
|
|
err_alloc_rule:
|
|
#endif
|
|
kfree(repr);
|
|
vf->repr = NULL;
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_rem - remove representor from VF
|
|
* @vf: pointer to VF structure
|
|
*/
|
|
static void ice_repr_rem(struct ice_vf *vf)
|
|
{
|
|
if (!vf->repr)
|
|
return;
|
|
|
|
ice_devlink_destroy_vf_port(vf);
|
|
kfree(vf->repr->q_vector);
|
|
vf->repr->q_vector = NULL;
|
|
unregister_netdev(vf->repr->netdev);
|
|
free_netdev(vf->repr->netdev);
|
|
vf->repr->netdev = NULL;
|
|
#ifdef CONFIG_ICE_SWITCHDEV
|
|
kfree(vf->repr->mac_rule);
|
|
vf->repr->mac_rule = NULL;
|
|
#endif
|
|
kfree(vf->repr);
|
|
vf->repr = NULL;
|
|
|
|
ice_virtchnl_set_dflt_ops(vf);
|
|
}
|
|
|
|
/**
|
|
* ice_repr_rem_from_all_vfs - remove port representor for all VFs
|
|
* @pf: pointer to PF structure
|
|
*/
|
|
void ice_repr_rem_from_all_vfs(struct ice_pf *pf)
|
|
{
|
|
struct ice_vf *vf;
|
|
unsigned int bkt;
|
|
|
|
lockdep_assert_held(&pf->vfs.table_lock);
|
|
|
|
ice_for_each_vf(pf, bkt, vf)
|
|
ice_repr_rem(vf);
|
|
}
|
|
|
|
/**
|
|
* ice_repr_add_for_all_vfs - add port representor for all VFs
|
|
* @pf: pointer to PF structure
|
|
*/
|
|
int ice_repr_add_for_all_vfs(struct ice_pf *pf)
|
|
{
|
|
struct ice_vf *vf;
|
|
unsigned int bkt;
|
|
int err;
|
|
|
|
lockdep_assert_held(&pf->vfs.table_lock);
|
|
|
|
ice_for_each_vf(pf, bkt, vf) {
|
|
err = ice_repr_add(vf);
|
|
if (err)
|
|
goto err;
|
|
}
|
|
|
|
return 0;
|
|
|
|
err:
|
|
ice_repr_rem_from_all_vfs(pf);
|
|
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* ice_repr_start_tx_queues - start Tx queues of port representor
|
|
* @repr: pointer to repr structure
|
|
*/
|
|
void ice_repr_start_tx_queues(struct ice_repr *repr)
|
|
{
|
|
netif_carrier_on(repr->netdev);
|
|
netif_tx_start_all_queues(repr->netdev);
|
|
}
|
|
|
|
/**
|
|
* ice_repr_stop_tx_queues - stop Tx queues of port representor
|
|
* @repr: pointer to repr structure
|
|
*/
|
|
void ice_repr_stop_tx_queues(struct ice_repr *repr)
|
|
{
|
|
netif_carrier_off(repr->netdev);
|
|
netif_tx_stop_all_queues(repr->netdev);
|
|
}
|
|
|
|
/**
|
|
* ice_repr_set_traffic_vsi - set traffic VSI for port representor
|
|
* @repr: repr on with VSI will be set
|
|
* @vsi: pointer to VSI that will be used by port representor to pass traffic
|
|
*/
|
|
void ice_repr_set_traffic_vsi(struct ice_repr *repr, struct ice_vsi *vsi)
|
|
{
|
|
struct ice_netdev_priv *np = netdev_priv(repr->netdev);
|
|
|
|
np->vsi = vsi;
|
|
}
|