| // SPDX-License-Identifier: GPL-2.0 |
| /* Copyright (c) 2024, Intel Corporation. */ |
| #include "ice.h" |
| #include "ice_lib.h" |
| #include "ice_txrx.h" |
| #include "ice_fltr.h" |
| #include "ice_sf_eth.h" |
| #include "devlink/devlink_port.h" |
| #include "devlink/devlink.h" |
| |
| static const struct net_device_ops ice_sf_netdev_ops = { |
| .ndo_open = ice_open, |
| .ndo_stop = ice_stop, |
| .ndo_start_xmit = ice_start_xmit, |
| .ndo_vlan_rx_add_vid = ice_vlan_rx_add_vid, |
| .ndo_vlan_rx_kill_vid = ice_vlan_rx_kill_vid, |
| .ndo_change_mtu = ice_change_mtu, |
| .ndo_get_stats64 = ice_get_stats64, |
| .ndo_tx_timeout = ice_tx_timeout, |
| .ndo_bpf = ice_xdp, |
| .ndo_xdp_xmit = ice_xdp_xmit, |
| .ndo_xsk_wakeup = ice_xsk_wakeup, |
| }; |
| |
| /** |
| * ice_sf_cfg_netdev - Allocate, configure and register a netdev |
| * @dyn_port: subfunction associated with configured netdev |
| * @devlink_port: subfunction devlink port to be linked with netdev |
| * |
| * Return: 0 on success, negative value on failure |
| */ |
| static int ice_sf_cfg_netdev(struct ice_dynamic_port *dyn_port, |
| struct devlink_port *devlink_port) |
| { |
| struct ice_vsi *vsi = dyn_port->vsi; |
| struct ice_netdev_priv *np; |
| struct net_device *netdev; |
| int err; |
| |
| netdev = alloc_etherdev_mqs(sizeof(*np), vsi->alloc_txq, |
| vsi->alloc_rxq); |
| if (!netdev) |
| return -ENOMEM; |
| |
| SET_NETDEV_DEV(netdev, &vsi->back->pdev->dev); |
| set_bit(ICE_VSI_NETDEV_ALLOCD, vsi->state); |
| vsi->netdev = netdev; |
| np = netdev_priv(netdev); |
| np->vsi = vsi; |
| |
| ice_set_netdev_features(netdev); |
| |
| netdev->xdp_features = NETDEV_XDP_ACT_BASIC | NETDEV_XDP_ACT_REDIRECT | |
| NETDEV_XDP_ACT_XSK_ZEROCOPY | |
| NETDEV_XDP_ACT_RX_SG; |
| netdev->xdp_zc_max_segs = ICE_MAX_BUF_TXD; |
| |
| eth_hw_addr_set(netdev, dyn_port->hw_addr); |
| ether_addr_copy(netdev->perm_addr, dyn_port->hw_addr); |
| netdev->netdev_ops = &ice_sf_netdev_ops; |
| SET_NETDEV_DEVLINK_PORT(netdev, devlink_port); |
| |
| err = register_netdev(netdev); |
| if (err) { |
| free_netdev(netdev); |
| vsi->netdev = NULL; |
| return -ENOMEM; |
| } |
| set_bit(ICE_VSI_NETDEV_REGISTERED, vsi->state); |
| netif_carrier_off(netdev); |
| netif_tx_stop_all_queues(netdev); |
| |
| return 0; |
| } |
| |
| static void ice_sf_decfg_netdev(struct ice_vsi *vsi) |
| { |
| unregister_netdev(vsi->netdev); |
| clear_bit(ICE_VSI_NETDEV_REGISTERED, vsi->state); |
| free_netdev(vsi->netdev); |
| vsi->netdev = NULL; |
| clear_bit(ICE_VSI_NETDEV_ALLOCD, vsi->state); |
| } |
| |
| /** |
| * ice_sf_dev_probe - subfunction driver probe function |
| * @adev: pointer to the auxiliary device |
| * @id: pointer to the auxiliary_device id |
| * |
| * Configure VSI and netdev resources for the subfunction device. |
| * |
| * Return: zero on success or an error code on failure. |
| */ |
| static int ice_sf_dev_probe(struct auxiliary_device *adev, |
| const struct auxiliary_device_id *id) |
| { |
| struct ice_sf_dev *sf_dev = ice_adev_to_sf_dev(adev); |
| struct ice_dynamic_port *dyn_port = sf_dev->dyn_port; |
| struct ice_vsi *vsi = dyn_port->vsi; |
| struct ice_pf *pf = dyn_port->pf; |
| struct device *dev = &adev->dev; |
| struct ice_sf_priv *priv; |
| struct devlink *devlink; |
| int err; |
| |
| vsi->type = ICE_VSI_SF; |
| vsi->port_info = pf->hw.port_info; |
| vsi->flags = ICE_VSI_FLAG_INIT; |
| |
| priv = ice_allocate_sf(&adev->dev, pf); |
| if (IS_ERR(priv)) { |
| dev_err(dev, "Subfunction devlink alloc failed"); |
| return PTR_ERR(priv); |
| } |
| |
| priv->dev = sf_dev; |
| sf_dev->priv = priv; |
| devlink = priv_to_devlink(priv); |
| |
| devl_lock(devlink); |
| |
| err = ice_vsi_cfg(vsi); |
| if (err) { |
| dev_err(dev, "Subfunction vsi config failed"); |
| goto err_free_devlink; |
| } |
| vsi->sf = dyn_port; |
| |
| ice_eswitch_update_repr(&dyn_port->repr_id, vsi); |
| |
| err = ice_devlink_create_sf_dev_port(sf_dev); |
| if (err) { |
| dev_err(dev, "Cannot add ice virtual devlink port for subfunction"); |
| goto err_vsi_decfg; |
| } |
| |
| err = ice_sf_cfg_netdev(dyn_port, &sf_dev->priv->devlink_port); |
| if (err) { |
| dev_err(dev, "Subfunction netdev config failed"); |
| goto err_devlink_destroy; |
| } |
| |
| err = devl_port_fn_devlink_set(&dyn_port->devlink_port, devlink); |
| if (err) { |
| dev_err(dev, "Can't link devlink instance to SF devlink port"); |
| goto err_netdev_decfg; |
| } |
| |
| ice_napi_add(vsi); |
| |
| devl_register(devlink); |
| devl_unlock(devlink); |
| |
| dyn_port->attached = true; |
| |
| return 0; |
| |
| err_netdev_decfg: |
| ice_sf_decfg_netdev(vsi); |
| err_devlink_destroy: |
| ice_devlink_destroy_sf_dev_port(sf_dev); |
| err_vsi_decfg: |
| ice_vsi_decfg(vsi); |
| err_free_devlink: |
| devl_unlock(devlink); |
| devlink_free(devlink); |
| return err; |
| } |
| |
| /** |
| * ice_sf_dev_remove - subfunction driver remove function |
| * @adev: pointer to the auxiliary device |
| * |
| * Deinitalize VSI and netdev resources for the subfunction device. |
| */ |
| static void ice_sf_dev_remove(struct auxiliary_device *adev) |
| { |
| struct ice_sf_dev *sf_dev = ice_adev_to_sf_dev(adev); |
| struct ice_dynamic_port *dyn_port = sf_dev->dyn_port; |
| struct ice_vsi *vsi = dyn_port->vsi; |
| struct devlink *devlink; |
| |
| devlink = priv_to_devlink(sf_dev->priv); |
| devl_lock(devlink); |
| |
| ice_vsi_close(vsi); |
| |
| ice_sf_decfg_netdev(vsi); |
| ice_devlink_destroy_sf_dev_port(sf_dev); |
| devl_unregister(devlink); |
| devl_unlock(devlink); |
| devlink_free(devlink); |
| ice_vsi_decfg(vsi); |
| |
| dyn_port->attached = false; |
| } |
| |
| static const struct auxiliary_device_id ice_sf_dev_id_table[] = { |
| { .name = "ice.sf", }, |
| { }, |
| }; |
| |
| MODULE_DEVICE_TABLE(auxiliary, ice_sf_dev_id_table); |
| |
| static struct auxiliary_driver ice_sf_driver = { |
| .name = "sf", |
| .probe = ice_sf_dev_probe, |
| .remove = ice_sf_dev_remove, |
| .id_table = ice_sf_dev_id_table |
| }; |
| |
| static DEFINE_XARRAY_ALLOC1(ice_sf_aux_id); |
| |
| /** |
| * ice_sf_driver_register - Register new auxiliary subfunction driver |
| * |
| * Return: zero on success or an error code on failure. |
| */ |
| int ice_sf_driver_register(void) |
| { |
| return auxiliary_driver_register(&ice_sf_driver); |
| } |
| |
| /** |
| * ice_sf_driver_unregister - Unregister new auxiliary subfunction driver |
| * |
| */ |
| void ice_sf_driver_unregister(void) |
| { |
| auxiliary_driver_unregister(&ice_sf_driver); |
| } |
| |
| /** |
| * ice_sf_dev_release - Release device associated with auxiliary device |
| * @device: pointer to the device |
| * |
| * Since most of the code for subfunction deactivation is handled in |
| * the remove handler, here just free tracking resources. |
| */ |
| static void ice_sf_dev_release(struct device *device) |
| { |
| struct auxiliary_device *adev = to_auxiliary_dev(device); |
| struct ice_sf_dev *sf_dev = ice_adev_to_sf_dev(adev); |
| |
| xa_erase(&ice_sf_aux_id, adev->id); |
| kfree(sf_dev); |
| } |
| |
| /** |
| * ice_sf_eth_activate - Activate Ethernet subfunction port |
| * @dyn_port: the dynamic port instance for this subfunction |
| * @extack: extack for reporting error messages |
| * |
| * Activate the dynamic port as an Ethernet subfunction. Setup the netdev |
| * resources associated and initialize the auxiliary device. |
| * |
| * Return: zero on success or an error code on failure. |
| */ |
| int |
| ice_sf_eth_activate(struct ice_dynamic_port *dyn_port, |
| struct netlink_ext_ack *extack) |
| { |
| struct ice_pf *pf = dyn_port->pf; |
| struct ice_sf_dev *sf_dev; |
| struct pci_dev *pdev; |
| int err; |
| u32 id; |
| |
| err = xa_alloc(&ice_sf_aux_id, &id, NULL, xa_limit_32b, |
| GFP_KERNEL); |
| if (err) { |
| NL_SET_ERR_MSG_MOD(extack, "Could not allocate SF ID"); |
| return err; |
| } |
| |
| sf_dev = kzalloc(sizeof(*sf_dev), GFP_KERNEL); |
| if (!sf_dev) { |
| err = -ENOMEM; |
| NL_SET_ERR_MSG_MOD(extack, "Could not allocate SF memory"); |
| goto xa_erase; |
| } |
| pdev = pf->pdev; |
| |
| sf_dev->dyn_port = dyn_port; |
| sf_dev->adev.id = id; |
| sf_dev->adev.name = "sf"; |
| sf_dev->adev.dev.release = ice_sf_dev_release; |
| sf_dev->adev.dev.parent = &pdev->dev; |
| |
| err = auxiliary_device_init(&sf_dev->adev); |
| if (err) { |
| NL_SET_ERR_MSG_MOD(extack, "Failed to initialize SF device"); |
| goto sf_dev_free; |
| } |
| |
| err = auxiliary_device_add(&sf_dev->adev); |
| if (err) { |
| NL_SET_ERR_MSG_MOD(extack, "Failed to add SF device"); |
| goto aux_dev_uninit; |
| } |
| |
| dyn_port->sf_dev = sf_dev; |
| |
| return 0; |
| |
| aux_dev_uninit: |
| auxiliary_device_uninit(&sf_dev->adev); |
| sf_dev_free: |
| kfree(sf_dev); |
| xa_erase: |
| xa_erase(&ice_sf_aux_id, id); |
| |
| return err; |
| } |
| |
| /** |
| * ice_sf_eth_deactivate - Deactivate Ethernet subfunction port |
| * @dyn_port: the dynamic port instance for this subfunction |
| * |
| * Deactivate the Ethernet subfunction, removing its auxiliary device and the |
| * associated resources. |
| */ |
| void ice_sf_eth_deactivate(struct ice_dynamic_port *dyn_port) |
| { |
| struct ice_sf_dev *sf_dev = dyn_port->sf_dev; |
| |
| auxiliary_device_delete(&sf_dev->adev); |
| auxiliary_device_uninit(&sf_dev->adev); |
| } |