mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	ionic: Add the basic NDO callbacks for netdev support
Set up the initial NDO structure and callbacks for netdev to use, and register the netdev. This will allow us to do a few basic operations on the device, but no traffic yet. Signed-off-by: Shannon Nelson <snelson@pensando.io> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									77ceb68e29
								
							
						
					
					
						commit
						beead698b1
					
				| @ -28,6 +28,7 @@ struct ionic_lif; | |||||||
| struct ionic { | struct ionic { | ||||||
| 	struct pci_dev *pdev; | 	struct pci_dev *pdev; | ||||||
| 	struct device *dev; | 	struct device *dev; | ||||||
|  | 	struct devlink_port dl_port; | ||||||
| 	struct ionic_dev idev; | 	struct ionic_dev idev; | ||||||
| 	struct mutex dev_cmd_lock;	/* lock for dev_cmd operations */ | 	struct mutex dev_cmd_lock;	/* lock for dev_cmd operations */ | ||||||
| 	struct dentry *dentry; | 	struct dentry *dentry; | ||||||
| @ -35,6 +36,7 @@ struct ionic { | |||||||
| 	unsigned int num_bars; | 	unsigned int num_bars; | ||||||
| 	struct ionic_identity ident; | 	struct ionic_identity ident; | ||||||
| 	struct list_head lifs; | 	struct list_head lifs; | ||||||
|  | 	struct ionic_lif *master_lif; | ||||||
| 	unsigned int nnqs_per_lif; | 	unsigned int nnqs_per_lif; | ||||||
| 	unsigned int neqs_per_lif; | 	unsigned int neqs_per_lif; | ||||||
| 	unsigned int ntxqs_per_lif; | 	unsigned int ntxqs_per_lif; | ||||||
|  | |||||||
| @ -206,12 +206,24 @@ static int ionic_probe(struct pci_dev *pdev, const struct pci_device_id *ent) | |||||||
| 		goto err_out_free_lifs; | 		goto err_out_free_lifs; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	err = ionic_lifs_register(ionic); | ||||||
|  | 	if (err) { | ||||||
|  | 		dev_err(dev, "Cannot register LIFs: %d, aborting\n", err); | ||||||
|  | 		goto err_out_deinit_lifs; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	err = ionic_devlink_register(ionic); | 	err = ionic_devlink_register(ionic); | ||||||
| 	if (err) | 	if (err) { | ||||||
| 		dev_err(dev, "Cannot register devlink: %d\n", err); | 		dev_err(dev, "Cannot register devlink: %d\n", err); | ||||||
|  | 		goto err_out_deregister_lifs; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| 
 | 
 | ||||||
|  | err_out_deregister_lifs: | ||||||
|  | 	ionic_lifs_unregister(ionic); | ||||||
|  | err_out_deinit_lifs: | ||||||
|  | 	ionic_lifs_deinit(ionic); | ||||||
| err_out_free_lifs: | err_out_free_lifs: | ||||||
| 	ionic_lifs_free(ionic); | 	ionic_lifs_free(ionic); | ||||||
| err_out_free_irqs: | err_out_free_irqs: | ||||||
| @ -246,6 +258,7 @@ static void ionic_remove(struct pci_dev *pdev) | |||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	ionic_devlink_unregister(ionic); | 	ionic_devlink_unregister(ionic); | ||||||
|  | 	ionic_lifs_unregister(ionic); | ||||||
| 	ionic_lifs_deinit(ionic); | 	ionic_lifs_deinit(ionic); | ||||||
| 	ionic_lifs_free(ionic); | 	ionic_lifs_free(ionic); | ||||||
| 	ionic_bus_free_irq_vectors(ionic); | 	ionic_bus_free_irq_vectors(ionic); | ||||||
|  | |||||||
| @ -10,6 +10,8 @@ | |||||||
| #include "ionic_if.h" | #include "ionic_if.h" | ||||||
| #include "ionic_regs.h" | #include "ionic_regs.h" | ||||||
| 
 | 
 | ||||||
|  | #define IONIC_MIN_MTU			ETH_MIN_MTU | ||||||
|  | #define IONIC_MAX_MTU			9194 | ||||||
| #define IONIC_LIFS_MAX			1024 | #define IONIC_LIFS_MAX			1024 | ||||||
| 
 | 
 | ||||||
| struct ionic_dev_bar { | struct ionic_dev_bar { | ||||||
|  | |||||||
| @ -6,6 +6,7 @@ | |||||||
| 
 | 
 | ||||||
| #include "ionic.h" | #include "ionic.h" | ||||||
| #include "ionic_bus.h" | #include "ionic_bus.h" | ||||||
|  | #include "ionic_lif.h" | ||||||
| #include "ionic_devlink.h" | #include "ionic_devlink.h" | ||||||
| 
 | 
 | ||||||
| static int ionic_dl_info_get(struct devlink *dl, struct devlink_info_req *req, | static int ionic_dl_info_get(struct devlink *dl, struct devlink_info_req *req, | ||||||
| @ -72,8 +73,19 @@ int ionic_devlink_register(struct ionic *ionic) | |||||||
| 	int err; | 	int err; | ||||||
| 
 | 
 | ||||||
| 	err = devlink_register(dl, ionic->dev); | 	err = devlink_register(dl, ionic->dev); | ||||||
| 	if (err) | 	if (err) { | ||||||
| 		dev_warn(ionic->dev, "devlink_register failed: %d\n", err); | 		dev_warn(ionic->dev, "devlink_register failed: %d\n", err); | ||||||
|  | 		return err; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	devlink_port_attrs_set(&ionic->dl_port, DEVLINK_PORT_FLAVOUR_PHYSICAL, | ||||||
|  | 			       0, false, 0, NULL, 0); | ||||||
|  | 	err = devlink_port_register(dl, &ionic->dl_port, 0); | ||||||
|  | 	if (err) | ||||||
|  | 		dev_err(ionic->dev, "devlink_port_register failed: %d\n", err); | ||||||
|  | 	else | ||||||
|  | 		devlink_port_type_eth_set(&ionic->dl_port, | ||||||
|  | 					  ionic->master_lif->netdev); | ||||||
| 
 | 
 | ||||||
| 	return err; | 	return err; | ||||||
| } | } | ||||||
| @ -82,5 +94,6 @@ void ionic_devlink_unregister(struct ionic *ionic) | |||||||
| { | { | ||||||
| 	struct devlink *dl = priv_to_devlink(ionic); | 	struct devlink *dl = priv_to_devlink(ionic); | ||||||
| 
 | 
 | ||||||
|  | 	devlink_port_unregister(&ionic->dl_port); | ||||||
| 	devlink_unregister(dl); | 	devlink_unregister(dl); | ||||||
| } | } | ||||||
|  | |||||||
| @ -399,6 +399,305 @@ static int ionic_adminq_napi(struct napi_struct *napi, int budget) | |||||||
| 	return max(n_work, a_work); | 	return max(n_work, a_work); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static __le64 ionic_netdev_features_to_nic(netdev_features_t features) | ||||||
|  | { | ||||||
|  | 	u64 wanted = 0; | ||||||
|  | 
 | ||||||
|  | 	if (features & NETIF_F_HW_VLAN_CTAG_TX) | ||||||
|  | 		wanted |= IONIC_ETH_HW_VLAN_TX_TAG; | ||||||
|  | 	if (features & NETIF_F_HW_VLAN_CTAG_RX) | ||||||
|  | 		wanted |= IONIC_ETH_HW_VLAN_RX_STRIP; | ||||||
|  | 	if (features & NETIF_F_HW_VLAN_CTAG_FILTER) | ||||||
|  | 		wanted |= IONIC_ETH_HW_VLAN_RX_FILTER; | ||||||
|  | 	if (features & NETIF_F_RXHASH) | ||||||
|  | 		wanted |= IONIC_ETH_HW_RX_HASH; | ||||||
|  | 	if (features & NETIF_F_RXCSUM) | ||||||
|  | 		wanted |= IONIC_ETH_HW_RX_CSUM; | ||||||
|  | 	if (features & NETIF_F_SG) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TX_SG; | ||||||
|  | 	if (features & NETIF_F_HW_CSUM) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TX_CSUM; | ||||||
|  | 	if (features & NETIF_F_TSO) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO; | ||||||
|  | 	if (features & NETIF_F_TSO6) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_IPV6; | ||||||
|  | 	if (features & NETIF_F_TSO_ECN) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_ECN; | ||||||
|  | 	if (features & NETIF_F_GSO_GRE) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_GRE; | ||||||
|  | 	if (features & NETIF_F_GSO_GRE_CSUM) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_GRE_CSUM; | ||||||
|  | 	if (features & NETIF_F_GSO_IPXIP4) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_IPXIP4; | ||||||
|  | 	if (features & NETIF_F_GSO_IPXIP6) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_IPXIP6; | ||||||
|  | 	if (features & NETIF_F_GSO_UDP_TUNNEL) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_UDP; | ||||||
|  | 	if (features & NETIF_F_GSO_UDP_TUNNEL_CSUM) | ||||||
|  | 		wanted |= IONIC_ETH_HW_TSO_UDP_CSUM; | ||||||
|  | 
 | ||||||
|  | 	return cpu_to_le64(wanted); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_set_nic_features(struct ionic_lif *lif, | ||||||
|  | 				  netdev_features_t features) | ||||||
|  | { | ||||||
|  | 	struct device *dev = lif->ionic->dev; | ||||||
|  | 	struct ionic_admin_ctx ctx = { | ||||||
|  | 		.work = COMPLETION_INITIALIZER_ONSTACK(ctx.work), | ||||||
|  | 		.cmd.lif_setattr = { | ||||||
|  | 			.opcode = IONIC_CMD_LIF_SETATTR, | ||||||
|  | 			.index = cpu_to_le16(lif->index), | ||||||
|  | 			.attr = IONIC_LIF_ATTR_FEATURES, | ||||||
|  | 		}, | ||||||
|  | 	}; | ||||||
|  | 	u64 vlan_flags = IONIC_ETH_HW_VLAN_TX_TAG | | ||||||
|  | 			 IONIC_ETH_HW_VLAN_RX_STRIP | | ||||||
|  | 			 IONIC_ETH_HW_VLAN_RX_FILTER; | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	ctx.cmd.lif_setattr.features = ionic_netdev_features_to_nic(features); | ||||||
|  | 	err = ionic_adminq_post_wait(lif, &ctx); | ||||||
|  | 	if (err) | ||||||
|  | 		return err; | ||||||
|  | 
 | ||||||
|  | 	lif->hw_features = le64_to_cpu(ctx.cmd.lif_setattr.features & | ||||||
|  | 				       ctx.comp.lif_setattr.features); | ||||||
|  | 
 | ||||||
|  | 	if ((vlan_flags & features) && | ||||||
|  | 	    !(vlan_flags & le64_to_cpu(ctx.comp.lif_setattr.features))) | ||||||
|  | 		dev_info_once(lif->ionic->dev, "NIC is not supporting vlan offload, likely in SmartNIC mode\n"); | ||||||
|  | 
 | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_VLAN_TX_TAG) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_VLAN_TX_TAG\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_VLAN_RX_STRIP) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_VLAN_RX_STRIP\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_VLAN_RX_FILTER) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_VLAN_RX_FILTER\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_RX_HASH) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_RX_HASH\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TX_SG) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TX_SG\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TX_CSUM) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TX_CSUM\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_RX_CSUM) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_RX_CSUM\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_IPV6) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_IPV6\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_ECN) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_ECN\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_GRE) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_GRE\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_GRE_CSUM) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_GRE_CSUM\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_IPXIP4) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_IPXIP4\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_IPXIP6) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_IPXIP6\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_UDP) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_UDP\n"); | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_UDP_CSUM) | ||||||
|  | 		dev_dbg(dev, "feature ETH_HW_TSO_UDP_CSUM\n"); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_init_nic_features(struct ionic_lif *lif) | ||||||
|  | { | ||||||
|  | 	struct net_device *netdev = lif->netdev; | ||||||
|  | 	netdev_features_t features; | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	/* set up what we expect to support by default */ | ||||||
|  | 	features = NETIF_F_HW_VLAN_CTAG_TX | | ||||||
|  | 		   NETIF_F_HW_VLAN_CTAG_RX | | ||||||
|  | 		   NETIF_F_HW_VLAN_CTAG_FILTER | | ||||||
|  | 		   NETIF_F_RXHASH | | ||||||
|  | 		   NETIF_F_SG | | ||||||
|  | 		   NETIF_F_HW_CSUM | | ||||||
|  | 		   NETIF_F_RXCSUM | | ||||||
|  | 		   NETIF_F_TSO | | ||||||
|  | 		   NETIF_F_TSO6 | | ||||||
|  | 		   NETIF_F_TSO_ECN; | ||||||
|  | 
 | ||||||
|  | 	err = ionic_set_nic_features(lif, features); | ||||||
|  | 	if (err) | ||||||
|  | 		return err; | ||||||
|  | 
 | ||||||
|  | 	/* tell the netdev what we actually can support */ | ||||||
|  | 	netdev->features |= NETIF_F_HIGHDMA; | ||||||
|  | 
 | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_VLAN_TX_TAG) | ||||||
|  | 		netdev->hw_features |= NETIF_F_HW_VLAN_CTAG_TX; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_VLAN_RX_STRIP) | ||||||
|  | 		netdev->hw_features |= NETIF_F_HW_VLAN_CTAG_RX; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_VLAN_RX_FILTER) | ||||||
|  | 		netdev->hw_features |= NETIF_F_HW_VLAN_CTAG_FILTER; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_RX_HASH) | ||||||
|  | 		netdev->hw_features |= NETIF_F_RXHASH; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TX_SG) | ||||||
|  | 		netdev->hw_features |= NETIF_F_SG; | ||||||
|  | 
 | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TX_CSUM) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_HW_CSUM; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_RX_CSUM) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_RXCSUM; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_TSO; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_IPV6) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_TSO6; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_ECN) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_TSO_ECN; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_GRE) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_GSO_GRE; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_GRE_CSUM) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_GSO_GRE_CSUM; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_IPXIP4) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_GSO_IPXIP4; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_IPXIP6) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_GSO_IPXIP6; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_UDP) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_GSO_UDP_TUNNEL; | ||||||
|  | 	if (lif->hw_features & IONIC_ETH_HW_TSO_UDP_CSUM) | ||||||
|  | 		netdev->hw_enc_features |= NETIF_F_GSO_UDP_TUNNEL_CSUM; | ||||||
|  | 
 | ||||||
|  | 	netdev->hw_features |= netdev->hw_enc_features; | ||||||
|  | 	netdev->features |= netdev->hw_features; | ||||||
|  | 
 | ||||||
|  | 	netdev->priv_flags |= IFF_UNICAST_FLT; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_set_features(struct net_device *netdev, | ||||||
|  | 			      netdev_features_t features) | ||||||
|  | { | ||||||
|  | 	struct ionic_lif *lif = netdev_priv(netdev); | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	netdev_dbg(netdev, "%s: lif->features=0x%08llx new_features=0x%08llx\n", | ||||||
|  | 		   __func__, (u64)lif->netdev->features, (u64)features); | ||||||
|  | 
 | ||||||
|  | 	err = ionic_set_nic_features(lif, features); | ||||||
|  | 
 | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_set_mac_address(struct net_device *netdev, void *sa) | ||||||
|  | { | ||||||
|  | 	netdev_info(netdev, "%s: stubbed\n", __func__); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_change_mtu(struct net_device *netdev, int new_mtu) | ||||||
|  | { | ||||||
|  | 	struct ionic_lif *lif = netdev_priv(netdev); | ||||||
|  | 	struct ionic_admin_ctx ctx = { | ||||||
|  | 		.work = COMPLETION_INITIALIZER_ONSTACK(ctx.work), | ||||||
|  | 		.cmd.lif_setattr = { | ||||||
|  | 			.opcode = IONIC_CMD_LIF_SETATTR, | ||||||
|  | 			.index = cpu_to_le16(lif->index), | ||||||
|  | 			.attr = IONIC_LIF_ATTR_MTU, | ||||||
|  | 			.mtu = cpu_to_le32(new_mtu), | ||||||
|  | 		}, | ||||||
|  | 	}; | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	err = ionic_adminq_post_wait(lif, &ctx); | ||||||
|  | 	if (err) | ||||||
|  | 		return err; | ||||||
|  | 
 | ||||||
|  | 	netdev->mtu = new_mtu; | ||||||
|  | 	err = ionic_reset_queues(lif); | ||||||
|  | 
 | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void ionic_tx_timeout(struct net_device *netdev) | ||||||
|  | { | ||||||
|  | 	netdev_info(netdev, "%s: stubbed\n", __func__); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_vlan_rx_add_vid(struct net_device *netdev, __be16 proto, | ||||||
|  | 				 u16 vid) | ||||||
|  | { | ||||||
|  | 	netdev_info(netdev, "%s: stubbed\n", __func__); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int ionic_vlan_rx_kill_vid(struct net_device *netdev, __be16 proto, | ||||||
|  | 				  u16 vid) | ||||||
|  | { | ||||||
|  | 	netdev_info(netdev, "%s: stubbed\n", __func__); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int ionic_open(struct net_device *netdev) | ||||||
|  | { | ||||||
|  | 	struct ionic_lif *lif = netdev_priv(netdev); | ||||||
|  | 
 | ||||||
|  | 	netif_carrier_off(netdev); | ||||||
|  | 
 | ||||||
|  | 	set_bit(IONIC_LIF_UP, lif->state); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int ionic_stop(struct net_device *netdev) | ||||||
|  | { | ||||||
|  | 	struct ionic_lif *lif = netdev_priv(netdev); | ||||||
|  | 	int err = 0; | ||||||
|  | 
 | ||||||
|  | 	if (!test_bit(IONIC_LIF_UP, lif->state)) { | ||||||
|  | 		dev_dbg(lif->ionic->dev, "%s: %s state=DOWN\n", | ||||||
|  | 			__func__, lif->name); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	dev_dbg(lif->ionic->dev, "%s: %s state=UP\n", __func__, lif->name); | ||||||
|  | 	clear_bit(IONIC_LIF_UP, lif->state); | ||||||
|  | 
 | ||||||
|  | 	/* carrier off before disabling queues to avoid watchdog timeout */ | ||||||
|  | 	netif_carrier_off(netdev); | ||||||
|  | 
 | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct net_device_ops ionic_netdev_ops = { | ||||||
|  | 	.ndo_open               = ionic_open, | ||||||
|  | 	.ndo_stop               = ionic_stop, | ||||||
|  | 	.ndo_set_features	= ionic_set_features, | ||||||
|  | 	.ndo_set_mac_address	= ionic_set_mac_address, | ||||||
|  | 	.ndo_validate_addr	= eth_validate_addr, | ||||||
|  | 	.ndo_tx_timeout         = ionic_tx_timeout, | ||||||
|  | 	.ndo_change_mtu         = ionic_change_mtu, | ||||||
|  | 	.ndo_vlan_rx_add_vid    = ionic_vlan_rx_add_vid, | ||||||
|  | 	.ndo_vlan_rx_kill_vid   = ionic_vlan_rx_kill_vid, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | int ionic_reset_queues(struct ionic_lif *lif) | ||||||
|  | { | ||||||
|  | 	bool running; | ||||||
|  | 	int err = 0; | ||||||
|  | 
 | ||||||
|  | 	/* Put off the next watchdog timeout */ | ||||||
|  | 	netif_trans_update(lif->netdev); | ||||||
|  | 
 | ||||||
|  | 	if (!ionic_wait_for_bit(lif, IONIC_LIF_QUEUE_RESET)) | ||||||
|  | 		return -EBUSY; | ||||||
|  | 
 | ||||||
|  | 	running = netif_running(lif->netdev); | ||||||
|  | 	if (running) | ||||||
|  | 		err = ionic_stop(lif->netdev); | ||||||
|  | 	if (!err && running) | ||||||
|  | 		ionic_open(lif->netdev); | ||||||
|  | 
 | ||||||
|  | 	clear_bit(IONIC_LIF_QUEUE_RESET, lif->state); | ||||||
|  | 
 | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static struct ionic_lif *ionic_lif_alloc(struct ionic *ionic, unsigned int index) | static struct ionic_lif *ionic_lif_alloc(struct ionic *ionic, unsigned int index) | ||||||
| { | { | ||||||
| 	struct device *dev = ionic->dev; | 	struct device *dev = ionic->dev; | ||||||
| @ -417,6 +716,12 @@ static struct ionic_lif *ionic_lif_alloc(struct ionic *ionic, unsigned int index | |||||||
| 
 | 
 | ||||||
| 	lif = netdev_priv(netdev); | 	lif = netdev_priv(netdev); | ||||||
| 	lif->netdev = netdev; | 	lif->netdev = netdev; | ||||||
|  | 	ionic->master_lif = lif; | ||||||
|  | 	netdev->netdev_ops = &ionic_netdev_ops; | ||||||
|  | 
 | ||||||
|  | 	netdev->watchdog_timeo = 2 * HZ; | ||||||
|  | 	netdev->min_mtu = IONIC_MIN_MTU; | ||||||
|  | 	netdev->max_mtu = IONIC_MAX_MTU; | ||||||
| 
 | 
 | ||||||
| 	lif->neqs = ionic->neqs_per_lif; | 	lif->neqs = ionic->neqs_per_lif; | ||||||
| 	lif->nxqs = ionic->ntxqs_per_lif; | 	lif->nxqs = ionic->ntxqs_per_lif; | ||||||
| @ -700,6 +1005,10 @@ static int ionic_lif_init(struct ionic_lif *lif) | |||||||
| 			goto err_out_notifyq_deinit; | 			goto err_out_notifyq_deinit; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	err = ionic_init_nic_features(lif); | ||||||
|  | 	if (err) | ||||||
|  | 		goto err_out_notifyq_deinit; | ||||||
|  | 
 | ||||||
| 	set_bit(IONIC_LIF_INITED, lif->state); | 	set_bit(IONIC_LIF_INITED, lif->state); | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| @ -734,6 +1043,32 @@ int ionic_lifs_init(struct ionic *ionic) | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | int ionic_lifs_register(struct ionic *ionic) | ||||||
|  | { | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	/* only register LIF0 for now */ | ||||||
|  | 	err = register_netdev(ionic->master_lif->netdev); | ||||||
|  | 	if (err) { | ||||||
|  | 		dev_err(ionic->dev, "Cannot register net device, aborting\n"); | ||||||
|  | 		return err; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	ionic->master_lif->registered = true; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ionic_lifs_unregister(struct ionic *ionic) | ||||||
|  | { | ||||||
|  | 	/* There is only one lif ever registered in the
 | ||||||
|  | 	 * current model, so don't bother searching the | ||||||
|  | 	 * ionic->lif for candidates to unregister | ||||||
|  | 	 */ | ||||||
|  | 	if (ionic->master_lif->netdev->reg_state == NETREG_REGISTERED) | ||||||
|  | 		unregister_netdev(ionic->master_lif->netdev); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| int ionic_lif_identify(struct ionic *ionic, u8 lif_type, | int ionic_lif_identify(struct ionic *ionic, u8 lif_type, | ||||||
| 		       union ionic_lif_identity *lid) | 		       union ionic_lif_identity *lid) | ||||||
| { | { | ||||||
|  | |||||||
| @ -59,6 +59,8 @@ struct ionic_qcq { | |||||||
| 
 | 
 | ||||||
| enum ionic_lif_state_flags { | enum ionic_lif_state_flags { | ||||||
| 	IONIC_LIF_INITED, | 	IONIC_LIF_INITED, | ||||||
|  | 	IONIC_LIF_UP, | ||||||
|  | 	IONIC_LIF_QUEUE_RESET, | ||||||
| 
 | 
 | ||||||
| 	/* leave this as last */ | 	/* leave this as last */ | ||||||
| 	IONIC_LIF_STATE_SIZE | 	IONIC_LIF_STATE_SIZE | ||||||
| @ -82,6 +84,7 @@ struct ionic_lif { | |||||||
| 	u64 last_eid; | 	u64 last_eid; | ||||||
| 	unsigned int neqs; | 	unsigned int neqs; | ||||||
| 	unsigned int nxqs; | 	unsigned int nxqs; | ||||||
|  | 	u64 hw_features; | ||||||
| 
 | 
 | ||||||
| 	struct ionic_lif_info *info; | 	struct ionic_lif_info *info; | ||||||
| 	dma_addr_t info_pa; | 	dma_addr_t info_pa; | ||||||
| @ -93,14 +96,31 @@ struct ionic_lif { | |||||||
| 	u32 flags; | 	u32 flags; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | static inline int ionic_wait_for_bit(struct ionic_lif *lif, int bitname) | ||||||
|  | { | ||||||
|  | 	unsigned long tlimit = jiffies + HZ; | ||||||
|  | 
 | ||||||
|  | 	while (test_and_set_bit(bitname, lif->state) && | ||||||
|  | 	       time_before(jiffies, tlimit)) | ||||||
|  | 		usleep_range(100, 200); | ||||||
|  | 
 | ||||||
|  | 	return test_bit(bitname, lif->state); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| int ionic_lifs_alloc(struct ionic *ionic); | int ionic_lifs_alloc(struct ionic *ionic); | ||||||
| void ionic_lifs_free(struct ionic *ionic); | void ionic_lifs_free(struct ionic *ionic); | ||||||
| void ionic_lifs_deinit(struct ionic *ionic); | void ionic_lifs_deinit(struct ionic *ionic); | ||||||
| int ionic_lifs_init(struct ionic *ionic); | int ionic_lifs_init(struct ionic *ionic); | ||||||
|  | int ionic_lifs_register(struct ionic *ionic); | ||||||
|  | void ionic_lifs_unregister(struct ionic *ionic); | ||||||
| int ionic_lif_identify(struct ionic *ionic, u8 lif_type, | int ionic_lif_identify(struct ionic *ionic, u8 lif_type, | ||||||
| 		       union ionic_lif_identity *lif_ident); | 		       union ionic_lif_identity *lif_ident); | ||||||
| int ionic_lifs_size(struct ionic *ionic); | int ionic_lifs_size(struct ionic *ionic); | ||||||
| 
 | 
 | ||||||
|  | int ionic_open(struct net_device *netdev); | ||||||
|  | int ionic_stop(struct net_device *netdev); | ||||||
|  | int ionic_reset_queues(struct ionic_lif *lif); | ||||||
|  | 
 | ||||||
| static inline void debug_stats_napi_poll(struct ionic_qcq *qcq, | static inline void debug_stats_napi_poll(struct ionic_qcq *qcq, | ||||||
| 					 unsigned int work_done) | 					 unsigned int work_done) | ||||||
| { | { | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Shannon Nelson
						Shannon Nelson