xref: /openbmc/linux/drivers/net/ethernet/sfc/ef100_netdev.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
151b35a45SEdward Cree // SPDX-License-Identifier: GPL-2.0-only
251b35a45SEdward Cree /****************************************************************************
351b35a45SEdward Cree  * Driver for Solarflare network controllers and boards
451b35a45SEdward Cree  * Copyright 2018 Solarflare Communications Inc.
551b35a45SEdward Cree  * Copyright 2019-2020 Xilinx Inc.
651b35a45SEdward Cree  *
751b35a45SEdward Cree  * This program is free software; you can redistribute it and/or modify it
851b35a45SEdward Cree  * under the terms of the GNU General Public License version 2 as published
951b35a45SEdward Cree  * by the Free Software Foundation, incorporated herein by reference.
1051b35a45SEdward Cree  */
1151b35a45SEdward Cree #include "net_driver.h"
1251b35a45SEdward Cree #include "mcdi_port_common.h"
1351b35a45SEdward Cree #include "mcdi_functions.h"
1451b35a45SEdward Cree #include "efx_common.h"
1551b35a45SEdward Cree #include "efx_channels.h"
1651b35a45SEdward Cree #include "tx_common.h"
1751b35a45SEdward Cree #include "ef100_netdev.h"
1851b35a45SEdward Cree #include "ef100_ethtool.h"
1951b35a45SEdward Cree #include "nic_common.h"
2051b35a45SEdward Cree #include "ef100_nic.h"
2151b35a45SEdward Cree #include "ef100_tx.h"
2251b35a45SEdward Cree #include "ef100_regs.h"
2351b35a45SEdward Cree #include "mcdi_filters.h"
2451b35a45SEdward Cree #include "rx_common.h"
2598ff4c7cSJonathan Cooper #include "ef100_sriov.h"
269dc0cad2SEdward Cree #include "tc_bindings.h"
27fa34a514SAlejandro Lucero #include "tc_encap_actions.h"
2851b35a45SEdward Cree #include "efx_devlink.h"
2951b35a45SEdward Cree 
ef100_update_name(struct efx_nic * efx)3051b35a45SEdward Cree static void ef100_update_name(struct efx_nic *efx)
3151b35a45SEdward Cree {
3251b35a45SEdward Cree 	strcpy(efx->name, efx->net_dev->name);
3351b35a45SEdward Cree }
34965b549fSEdward Cree 
ef100_alloc_vis(struct efx_nic * efx,unsigned int * allocated_vis)35965b549fSEdward Cree static int ef100_alloc_vis(struct efx_nic *efx, unsigned int *allocated_vis)
36965b549fSEdward Cree {
37965b549fSEdward Cree 	/* EF100 uses a single TXQ per channel, as all checksum offloading
38965b549fSEdward Cree 	 * is configured in the TX descriptor, and there is no TX Pacer for
39965b549fSEdward Cree 	 * HIGHPRI queues.
40965b549fSEdward Cree 	 */
41965b549fSEdward Cree 	unsigned int tx_vis = efx->n_tx_channels + efx->n_extra_tx_channels;
42965b549fSEdward Cree 	unsigned int rx_vis = efx->n_rx_channels;
43965b549fSEdward Cree 	unsigned int min_vis, max_vis;
44965b549fSEdward Cree 	int rc;
45965b549fSEdward Cree 
46965b549fSEdward Cree 	EFX_WARN_ON_PARANOID(efx->tx_queues_per_channel != 1);
47965b549fSEdward Cree 
48965b549fSEdward Cree 	tx_vis += efx->n_xdp_channels * efx->xdp_tx_per_channel;
49965b549fSEdward Cree 
50965b549fSEdward Cree 	max_vis = max(rx_vis, tx_vis);
51965b549fSEdward Cree 	/* We require at least a single complete TX channel worth of queues. */
52965b549fSEdward Cree 	min_vis = efx->tx_queues_per_channel;
53965b549fSEdward Cree 
54965b549fSEdward Cree 	rc = efx_mcdi_alloc_vis(efx, min_vis, max_vis,
55965b549fSEdward Cree 				NULL, allocated_vis);
56965b549fSEdward Cree 
57965b549fSEdward Cree 	/* We retry allocating VIs by reallocating channels when we have not
58965b549fSEdward Cree 	 * been able to allocate the maximum VIs.
59965b549fSEdward Cree 	 */
60965b549fSEdward Cree 	if (!rc && *allocated_vis < max_vis)
61965b549fSEdward Cree 		rc = -EAGAIN;
62965b549fSEdward Cree 
63965b549fSEdward Cree 	return rc;
64965b549fSEdward Cree }
65965b549fSEdward Cree 
ef100_remap_bar(struct efx_nic * efx,int max_vis)66965b549fSEdward Cree static int ef100_remap_bar(struct efx_nic *efx, int max_vis)
67965b549fSEdward Cree {
68965b549fSEdward Cree 	unsigned int uc_mem_map_size;
69965b549fSEdward Cree 	void __iomem *membase;
70965b549fSEdward Cree 
71965b549fSEdward Cree 	efx->max_vis = max_vis;
72965b549fSEdward Cree 	uc_mem_map_size = PAGE_ALIGN(max_vis * efx->vi_stride);
73965b549fSEdward Cree 
74965b549fSEdward Cree 	/* Extend the original UC mapping of the memory BAR */
75965b549fSEdward Cree 	membase = ioremap(efx->membase_phys, uc_mem_map_size);
76965b549fSEdward Cree 	if (!membase) {
77965b549fSEdward Cree 		netif_err(efx, probe, efx->net_dev,
78965b549fSEdward Cree 			  "could not extend memory BAR to %x\n",
79965b549fSEdward Cree 			  uc_mem_map_size);
80965b549fSEdward Cree 		return -ENOMEM;
81965b549fSEdward Cree 	}
82965b549fSEdward Cree 	iounmap(efx->membase);
83965b549fSEdward Cree 	efx->membase = membase;
84965b549fSEdward Cree 	return 0;
858cb03f4eSJonathan Cooper }
86965b549fSEdward Cree 
87965b549fSEdward Cree /* Context: process, rtnl_lock() held.
88965b549fSEdward Cree  * Note that the kernel will ignore our return code; this method
89965b549fSEdward Cree  * should really be a void.
9084e7fc25SEdward Cree  */
ef100_net_stop(struct net_device * net_dev)91965b549fSEdward Cree static int ef100_net_stop(struct net_device *net_dev)
92965b549fSEdward Cree {
93b593b6f1SEdward Cree 	struct efx_nic *efx = efx_netdev_priv(net_dev);
94965b549fSEdward Cree 
95965b549fSEdward Cree 	netif_dbg(efx, ifdown, efx->net_dev, "closing on CPU %d\n",
96965b549fSEdward Cree 		  raw_smp_processor_id());
97a9dc3d56SEdward Cree 
98965b549fSEdward Cree 	efx_detach_reps(efx);
99965b549fSEdward Cree 	netif_stop_queue(net_dev);
100965b549fSEdward Cree 	efx_stop_all(efx);
101965b549fSEdward Cree 	efx_mcdi_mac_fini_stats(efx);
102965b549fSEdward Cree 	efx_disable_interrupts(efx);
103813cf9d1SJonathan Cooper 	efx_clear_interrupt_affinity(efx);
104813cf9d1SJonathan Cooper 	efx_nic_fini_interrupt(efx);
105965b549fSEdward Cree 	efx_remove_filters(efx);
106965b549fSEdward Cree 	efx_fini_napi(efx);
107965b549fSEdward Cree 	efx_remove_channels(efx);
108965b549fSEdward Cree 	efx_mcdi_free_vis(efx);
109965b549fSEdward Cree 	efx_remove_interrupts(efx);
110965b549fSEdward Cree 
1118cb03f4eSJonathan Cooper 	efx->state = STATE_NET_DOWN;
112965b549fSEdward Cree 
113965b549fSEdward Cree 	return 0;
114965b549fSEdward Cree }
115965b549fSEdward Cree 
116965b549fSEdward Cree /* Context: process, rtnl_lock() held. */
ef100_net_open(struct net_device * net_dev)117965b549fSEdward Cree static int ef100_net_open(struct net_device *net_dev)
118965b549fSEdward Cree {
119965b549fSEdward Cree 	struct efx_nic *efx = efx_netdev_priv(net_dev);
120965b549fSEdward Cree 	unsigned int allocated_vis;
121965b549fSEdward Cree 	int rc;
122965b549fSEdward Cree 
123965b549fSEdward Cree 	ef100_update_name(efx);
124965b549fSEdward Cree 	netif_dbg(efx, ifup, net_dev, "opening device on CPU %d\n",
125965b549fSEdward Cree 		  raw_smp_processor_id());
126965b549fSEdward Cree 
127965b549fSEdward Cree 	rc = efx_check_disabled(efx);
128965b549fSEdward Cree 	if (rc)
129965b549fSEdward Cree 		goto fail;
130965b549fSEdward Cree 
131965b549fSEdward Cree 	rc = efx_probe_interrupts(efx);
132965b549fSEdward Cree 	if (rc)
133965b549fSEdward Cree 		goto fail;
134965b549fSEdward Cree 
135965b549fSEdward Cree 	rc = efx_set_channels(efx);
136965b549fSEdward Cree 	if (rc)
137965b549fSEdward Cree 		goto fail;
138965b549fSEdward Cree 
139965b549fSEdward Cree 	rc = efx_mcdi_free_vis(efx);
140965b549fSEdward Cree 	if (rc)
141965b549fSEdward Cree 		goto fail;
142965b549fSEdward Cree 
143965b549fSEdward Cree 	rc = ef100_alloc_vis(efx, &allocated_vis);
144965b549fSEdward Cree 	if (rc && rc != -EAGAIN)
145965b549fSEdward Cree 		goto fail;
146965b549fSEdward Cree 
147965b549fSEdward Cree 	/* Try one more time but with the maximum number of channels
148965b549fSEdward Cree 	 * equal to the allocated VIs, which would more likely succeed.
149a9dc3d56SEdward Cree 	 */
150a9dc3d56SEdward Cree 	if (rc == -EAGAIN) {
151a9dc3d56SEdward Cree 		rc = efx_mcdi_free_vis(efx);
152a9dc3d56SEdward Cree 		if (rc)
153965b549fSEdward Cree 			goto fail;
154965b549fSEdward Cree 
155965b549fSEdward Cree 		efx_remove_interrupts(efx);
156965b549fSEdward Cree 		efx->max_channels = allocated_vis;
157965b549fSEdward Cree 
158965b549fSEdward Cree 		rc = efx_probe_interrupts(efx);
159965b549fSEdward Cree 		if (rc)
160965b549fSEdward Cree 			goto fail;
161965b549fSEdward Cree 
162965b549fSEdward Cree 		rc = efx_set_channels(efx);
163965b549fSEdward Cree 		if (rc)
164965b549fSEdward Cree 			goto fail;
165965b549fSEdward Cree 
166965b549fSEdward Cree 		rc = ef100_alloc_vis(efx, &allocated_vis);
167b593b6f1SEdward Cree 		if (rc && rc != -EAGAIN)
168b593b6f1SEdward Cree 			goto fail;
169b593b6f1SEdward Cree 
170b593b6f1SEdward Cree 		/* It should be very unlikely that we failed here again, but in
171965b549fSEdward Cree 		 * such a case we return ENOSPC.
172965b549fSEdward Cree 		 */
173965b549fSEdward Cree 		if (rc == -EAGAIN) {
174965b549fSEdward Cree 			rc = -ENOSPC;
175965b549fSEdward Cree 			goto fail;
176965b549fSEdward Cree 		}
177965b549fSEdward Cree 	}
178965b549fSEdward Cree 
179965b549fSEdward Cree 	rc = efx_probe_channels(efx);
180965b549fSEdward Cree 	if (rc)
181813cf9d1SJonathan Cooper 		return rc;
18284e7fc25SEdward Cree 
18384e7fc25SEdward Cree 	rc = ef100_remap_bar(efx, allocated_vis);
184813cf9d1SJonathan Cooper 	if (rc)
185965b549fSEdward Cree 		goto fail;
186965b549fSEdward Cree 
187965b549fSEdward Cree 	efx_init_napi(efx);
188965b549fSEdward Cree 
189965b549fSEdward Cree 	rc = efx_probe_filters(efx);
190965b549fSEdward Cree 	if (rc)
191965b549fSEdward Cree 		goto fail;
19251b35a45SEdward Cree 
19351b35a45SEdward Cree 	rc = efx_nic_init_interrupt(efx);
19451b35a45SEdward Cree 	if (rc)
19551b35a45SEdward Cree 		goto fail;
19651b35a45SEdward Cree 	efx_set_interrupt_affinity(efx);
19751b35a45SEdward Cree 
19851b35a45SEdward Cree 	rc = efx_enable_interrupts(efx);
19951b35a45SEdward Cree 	if (rc)
20051b35a45SEdward Cree 		goto fail;
20151b35a45SEdward Cree 
2028cb03f4eSJonathan Cooper 	/* in case the MC rebooted while we were stopped, consume the change
203f72c38faSEdward Cree 	 * to the warm reboot count
204f72c38faSEdward Cree 	 */
205f72c38faSEdward Cree 	(void) efx_mcdi_poll_reboot(efx);
206f72c38faSEdward Cree 
207f72c38faSEdward Cree 	rc = efx_mcdi_mac_init_stats(efx);
208f72c38faSEdward Cree 	if (rc)
209f72c38faSEdward Cree 		goto fail;
210f72c38faSEdward Cree 
211f72c38faSEdward Cree 	efx_start_all(efx);
21251b35a45SEdward Cree 
21351b35a45SEdward Cree 	/* Link state detection is normally event-driven; we have
21451b35a45SEdward Cree 	 * to poll now because we could have missed a change
21551b35a45SEdward Cree 	 */
21651b35a45SEdward Cree 	mutex_lock(&efx->mac_lock);
21751b35a45SEdward Cree 	if (efx_mcdi_phy_poll(efx))
21851b35a45SEdward Cree 		efx_link_status_changed(efx);
21951b35a45SEdward Cree 	mutex_unlock(&efx->mac_lock);
22051b35a45SEdward Cree 
22151b35a45SEdward Cree 	efx->state = STATE_NET_UP;
222aad98abdSZhang Changzhong 	if (netif_running(efx->net_dev))
22351b35a45SEdward Cree 		efx_attach_reps(efx);
22451b35a45SEdward Cree 
22551b35a45SEdward Cree 	return 0;
22651b35a45SEdward Cree 
227f72c38faSEdward Cree fail:
22851b35a45SEdward Cree 	ef100_net_stop(net_dev);
22951b35a45SEdward Cree 	return rc;
23051b35a45SEdward Cree }
23151b35a45SEdward Cree 
23251b35a45SEdward Cree /* Initiate a packet transmission.  We use one channel per CPU
23351b35a45SEdward Cree  * (sharing when we have more CPUs than channels).
23451b35a45SEdward Cree  *
23551b35a45SEdward Cree  * Context: non-blocking.
23651b35a45SEdward Cree  * Note that returning anything other than NETDEV_TX_OK will cause the
237965b549fSEdward Cree  * OS to free the skb.
238965b549fSEdward Cree  */
ef100_hard_start_xmit(struct sk_buff * skb,struct net_device * net_dev)23951b35a45SEdward Cree static netdev_tx_t ef100_hard_start_xmit(struct sk_buff *skb,
2404404c089SEdward Cree 					 struct net_device *net_dev)
241b593b6f1SEdward Cree {
2424404c089SEdward Cree 	struct efx_nic *efx = efx_netdev_priv(net_dev);
243a9dc3d56SEdward Cree 
2444404c089SEdward Cree 	return __ef100_hard_start_xmit(skb, efx, net_dev, NULL);
245a9dc3d56SEdward Cree }
2464404c089SEdward Cree 
__ef100_hard_start_xmit(struct sk_buff * skb,struct efx_nic * efx,struct net_device * net_dev,struct efx_rep * efv)2471c748843SEdward Cree netdev_tx_t __ef100_hard_start_xmit(struct sk_buff *skb,
2481c748843SEdward Cree 				    struct efx_nic *efx,
249a9dc3d56SEdward Cree 				    struct net_device *net_dev,
250a9dc3d56SEdward Cree 				    struct efx_rep *efv)
251a9dc3d56SEdward Cree {
2529dc0cad2SEdward Cree 	struct efx_tx_queue *tx_queue;
2539dc0cad2SEdward Cree 	struct efx_channel *channel;
2549dc0cad2SEdward Cree 	int rc;
25551b35a45SEdward Cree 
25651b35a45SEdward Cree 	channel = efx_get_tx_channel(efx, skb_get_queue_mapping(skb));
25751b35a45SEdward Cree 	netif_vdbg(efx, tx_queued, efx->net_dev,
25851b35a45SEdward Cree 		   "%s len %d data %d channel %d\n", __func__,
25951b35a45SEdward Cree 		   skb->len, skb->data_len, channel->channel);
26051b35a45SEdward Cree 	if (!efx->n_channels || !efx->n_tx_channels || !channel) {
26151b35a45SEdward Cree 		netif_stop_queue(net_dev);
26251b35a45SEdward Cree 		dev_kfree_skb_any(skb);
26351b35a45SEdward Cree 		goto err;
26451b35a45SEdward Cree 	}
26598ff4c7cSJonathan Cooper 
26698ff4c7cSJonathan Cooper 	tx_queue = &channel->tx_queue[0];
26751b35a45SEdward Cree 	rc = __ef100_enqueue_skb(tx_queue, skb, efv);
26851b35a45SEdward Cree 	if (rc == 0)
26951b35a45SEdward Cree 		return NETDEV_TX_OK;
27051b35a45SEdward Cree 
27151b35a45SEdward Cree err:
27298ff4c7cSJonathan Cooper 	net_dev->stats.tx_dropped++;
27351b35a45SEdward Cree 	return NETDEV_TX_OK;
27451b35a45SEdward Cree }
27551b35a45SEdward Cree 
27651b35a45SEdward Cree static const struct net_device_ops ef100_netdev_ops = {
27751b35a45SEdward Cree 	.ndo_open               = ef100_net_open,
27851b35a45SEdward Cree 	.ndo_stop               = ef100_net_stop,
27951b35a45SEdward Cree 	.ndo_start_xmit         = ef100_hard_start_xmit,
28051b35a45SEdward Cree 	.ndo_tx_timeout         = efx_watchdog,
28151b35a45SEdward Cree 	.ndo_get_stats64        = efx_net_stats,
28251b35a45SEdward Cree 	.ndo_change_mtu         = efx_change_mtu,
28351b35a45SEdward Cree 	.ndo_validate_addr      = eth_validate_addr,
28451b35a45SEdward Cree 	.ndo_set_mac_address    = efx_set_mac_address,
28551b35a45SEdward Cree 	.ndo_set_rx_mode        = efx_set_rx_mode, /* Lookout */
28651b35a45SEdward Cree 	.ndo_set_features       = efx_set_features,
28751b35a45SEdward Cree 	.ndo_get_phys_port_id   = efx_get_phys_port_id,
28851b35a45SEdward Cree 	.ndo_get_phys_port_name = efx_get_phys_port_name,
28951b35a45SEdward Cree #ifdef CONFIG_RFS_ACCEL
29051b35a45SEdward Cree 	.ndo_rx_flow_steer      = efx_filter_rfs,
29151b35a45SEdward Cree #endif
29251b35a45SEdward Cree #ifdef CONFIG_SFC_SRIOV
29351b35a45SEdward Cree 	.ndo_setup_tc		= efx_tc_setup,
29451b35a45SEdward Cree #endif
29551b35a45SEdward Cree };
29651b35a45SEdward Cree 
29751b35a45SEdward Cree /*	Netdev registration
298813cf9d1SJonathan Cooper  */
ef100_netdev_event(struct notifier_block * this,unsigned long event,void * ptr)29951b35a45SEdward Cree int ef100_netdev_event(struct notifier_block *this,
30051b35a45SEdward Cree 		       unsigned long event, void *ptr)
30151b35a45SEdward Cree {
30251b35a45SEdward Cree 	struct efx_nic *efx = container_of(this, struct efx_nic, netdev_notifier);
30351b35a45SEdward Cree 	struct net_device *net_dev = netdev_notifier_info_to_dev(ptr);
30451b35a45SEdward Cree 	struct ef100_nic_data *nic_data = efx->nic_data;
30551b35a45SEdward Cree 	int err;
30651b35a45SEdward Cree 
30751b35a45SEdward Cree 	if (efx->net_dev == net_dev &&
30851b35a45SEdward Cree 	    (event == NETDEV_CHANGENAME || event == NETDEV_REGISTER))
30951b35a45SEdward Cree 		ef100_update_name(efx);
31098ff4c7cSJonathan Cooper 
31151b35a45SEdward Cree 	if (!nic_data->grp_mae)
31251b35a45SEdward Cree 		return NOTIFY_DONE;
31351b35a45SEdward Cree 	err = efx_tc_netdev_event(efx, event, net_dev);
3148b39db19SJonathan Cooper 	if (err & NOTIFY_STOP_MASK)
31551b35a45SEdward Cree 		return err;
31651b35a45SEdward Cree 
31751b35a45SEdward Cree 	return NOTIFY_DONE;
31898ff4c7cSJonathan Cooper }
31998ff4c7cSJonathan Cooper 
ef100_netevent_event(struct notifier_block * this,unsigned long event,void * ptr)32098ff4c7cSJonathan Cooper static int ef100_netevent_event(struct notifier_block *this,
32198ff4c7cSJonathan Cooper 				unsigned long event, void *ptr)
32298ff4c7cSJonathan Cooper {
32398ff4c7cSJonathan Cooper 	struct efx_nic *efx = container_of(this, struct efx_nic, netevent_notifier);
32498ff4c7cSJonathan Cooper 	struct ef100_nic_data *nic_data = efx->nic_data;
32598ff4c7cSJonathan Cooper 	int err;
32698ff4c7cSJonathan Cooper 
32798ff4c7cSJonathan Cooper 	if (!nic_data->grp_mae)
32898ff4c7cSJonathan Cooper 		return NOTIFY_DONE;
32998ff4c7cSJonathan Cooper 	err = efx_tc_netevent_event(efx, event, ptr);
33098ff4c7cSJonathan Cooper 	if (err & NOTIFY_STOP_MASK)
33198ff4c7cSJonathan Cooper 		return err;
33298ff4c7cSJonathan Cooper 
33308135eecSEdward Cree 	return NOTIFY_DONE;
33498ff4c7cSJonathan Cooper };
33598ff4c7cSJonathan Cooper 
ef100_register_netdev(struct efx_nic * efx)336fa34a514SAlejandro Lucero static int ef100_register_netdev(struct efx_nic *efx)
33798ff4c7cSJonathan Cooper {
33898ff4c7cSJonathan Cooper 	struct net_device *net_dev = efx->net_dev;
33967ab160eSEdward Cree 	int rc;
34025414b2aSAlejandro Lucero 
34167ab160eSEdward Cree 	net_dev->watchdog_timeo = 5 * HZ;
34267ab160eSEdward Cree 	net_dev->irq = efx->pci_dev->irq;
34367ab160eSEdward Cree 	net_dev->netdev_ops = &ef100_netdev_ops;
34498ff4c7cSJonathan Cooper 	net_dev->min_mtu = EFX_MIN_MTU;
34598ff4c7cSJonathan Cooper 	net_dev->max_mtu = EFX_MAX_MTU;
34698ff4c7cSJonathan Cooper 	net_dev->ethtool_ops = &ef100_ethtool_ops;
34798ff4c7cSJonathan Cooper 
34898ff4c7cSJonathan Cooper 	rtnl_lock();
34998ff4c7cSJonathan Cooper 
35098ff4c7cSJonathan Cooper 	rc = dev_alloc_name(net_dev, net_dev->name);
351fa34a514SAlejandro Lucero 	if (rc < 0)
352fa34a514SAlejandro Lucero 		goto fail_locked;
35398ff4c7cSJonathan Cooper 	ef100_update_name(efx);
35498ff4c7cSJonathan Cooper 
35598ff4c7cSJonathan Cooper 	rc = register_netdevice(net_dev);
35698ff4c7cSJonathan Cooper 	if (rc)
35798ff4c7cSJonathan Cooper 		goto fail_locked;
35898ff4c7cSJonathan Cooper 
35998ff4c7cSJonathan Cooper 	/* Always start with carrier off; PHY events will detect the link */
36098ff4c7cSJonathan Cooper 	netif_carrier_off(net_dev);
36198ff4c7cSJonathan Cooper 
3627e056e23SAlejandro Lucero 	efx->state = STATE_NET_DOWN;
36398ff4c7cSJonathan Cooper 	rtnl_unlock();
36498ff4c7cSJonathan Cooper 	efx_init_mcdi_logging(efx);
36598ff4c7cSJonathan Cooper 
36698ff4c7cSJonathan Cooper 	return 0;
36798ff4c7cSJonathan Cooper 
36898ff4c7cSJonathan Cooper fail_locked:
36998ff4c7cSJonathan Cooper 	rtnl_unlock();
37098ff4c7cSJonathan Cooper 	netif_err(efx, drv, efx->net_dev, "could not register net dev\n");
37198ff4c7cSJonathan Cooper 	return rc;
37298ff4c7cSJonathan Cooper }
37398ff4c7cSJonathan Cooper 
ef100_unregister_netdev(struct efx_nic * efx)37498ff4c7cSJonathan Cooper static void ef100_unregister_netdev(struct efx_nic *efx)
37598ff4c7cSJonathan Cooper {
37698ff4c7cSJonathan Cooper 	if (efx_dev_registered(efx)) {
37798ff4c7cSJonathan Cooper 		efx_fini_mcdi_logging(efx);
37898ff4c7cSJonathan Cooper 		efx->state = STATE_PROBED;
37998ff4c7cSJonathan Cooper 		unregister_netdev(efx->net_dev);
38098ff4c7cSJonathan Cooper 	}
381*134120b0SPieter Jansen van Vuuren }
382*134120b0SPieter Jansen van Vuuren 
ef100_remove_netdev(struct efx_probe_data * probe_data)383*134120b0SPieter Jansen van Vuuren void ef100_remove_netdev(struct efx_probe_data *probe_data)
38498ff4c7cSJonathan Cooper {
38598ff4c7cSJonathan Cooper 	struct efx_nic *efx = &probe_data->efx;
38698ff4c7cSJonathan Cooper 
38798ff4c7cSJonathan Cooper 	if (!efx->net_dev)
38898ff4c7cSJonathan Cooper 		return;
38998ff4c7cSJonathan Cooper 
39098ff4c7cSJonathan Cooper 	rtnl_lock();
39198ff4c7cSJonathan Cooper 	dev_close(efx->net_dev);
39298ff4c7cSJonathan Cooper 	rtnl_unlock();
39398ff4c7cSJonathan Cooper 
39498ff4c7cSJonathan Cooper 	unregister_netdevice_notifier(&efx->netdev_notifier);
39598ff4c7cSJonathan Cooper 	unregister_netevent_notifier(&efx->netevent_notifier);
39698ff4c7cSJonathan Cooper #if defined(CONFIG_SFC_SRIOV)
39798ff4c7cSJonathan Cooper 	if (!efx->type->is_vf)
39898ff4c7cSJonathan Cooper 		efx_ef100_pci_sriov_disable(efx, true);
39998ff4c7cSJonathan Cooper #endif
40098ff4c7cSJonathan Cooper 
40198ff4c7cSJonathan Cooper 	efx_fini_devlink_lock(efx);
40298ff4c7cSJonathan Cooper 	ef100_unregister_netdev(efx);
40398ff4c7cSJonathan Cooper 
40498ff4c7cSJonathan Cooper #ifdef CONFIG_SFC_SRIOV
40598ff4c7cSJonathan Cooper 	ef100_pf_unset_devlink_port(efx);
40698ff4c7cSJonathan Cooper 	efx_fini_tc(efx);
40798ff4c7cSJonathan Cooper #endif
40898ff4c7cSJonathan Cooper 
40998ff4c7cSJonathan Cooper 	down_write(&efx->filter_sem);
41098ff4c7cSJonathan Cooper 	efx_mcdi_filter_table_remove(efx);
41198ff4c7cSJonathan Cooper 	up_write(&efx->filter_sem);
41298ff4c7cSJonathan Cooper 	efx_fini_channels(efx);
41398ff4c7cSJonathan Cooper 	kfree(efx->phy_data);
41498ff4c7cSJonathan Cooper 	efx->phy_data = NULL;
41598ff4c7cSJonathan Cooper 
4167e056e23SAlejandro Lucero 	efx_fini_devlink_and_unlock(efx);
4177e056e23SAlejandro Lucero 
4187e056e23SAlejandro Lucero 	free_netdev(efx->net_dev);
4197e056e23SAlejandro Lucero 	efx->net_dev = NULL;
4207e056e23SAlejandro Lucero 	efx->state = STATE_PROBED;
4217e056e23SAlejandro Lucero }
4227e056e23SAlejandro Lucero 
ef100_probe_netdev(struct efx_probe_data * probe_data)4237e056e23SAlejandro Lucero int ef100_probe_netdev(struct efx_probe_data *probe_data)
4247e056e23SAlejandro Lucero {
425fa34a514SAlejandro Lucero 	struct efx_nic *efx = &probe_data->efx;
426fa34a514SAlejandro Lucero 	struct efx_probe_data **probe_ptr;
427fa34a514SAlejandro Lucero 	struct ef100_nic_data *nic_data;
428fa34a514SAlejandro Lucero 	struct net_device *net_dev;
429fa34a514SAlejandro Lucero 	int rc;
43098ff4c7cSJonathan Cooper 
43198ff4c7cSJonathan Cooper 	if (efx->mcdi->fn_flags &
43298ff4c7cSJonathan Cooper 			(1 << MC_CMD_DRV_ATTACH_EXT_OUT_FLAG_NO_ACTIVE_PORT)) {
43398ff4c7cSJonathan Cooper 		pci_info(efx->pci_dev, "No network port on this PCI function");
43498ff4c7cSJonathan Cooper 		return 0;
43598ff4c7cSJonathan Cooper 	}
43698ff4c7cSJonathan Cooper 
43798ff4c7cSJonathan Cooper 	/* Allocate and initialise a struct net_device */
43825414b2aSAlejandro Lucero 	net_dev = alloc_etherdev_mq(sizeof(probe_data), EFX_MAX_CORE_TX_QUEUES);
43925414b2aSAlejandro Lucero 	if (!net_dev)
44025414b2aSAlejandro Lucero 		return -ENOMEM;
44198ff4c7cSJonathan Cooper 	probe_ptr = netdev_priv(net_dev);
44298ff4c7cSJonathan Cooper 	*probe_ptr = probe_data;
44398ff4c7cSJonathan Cooper 	efx->net_dev = net_dev;
44498ff4c7cSJonathan Cooper 	SET_NETDEV_DEV(net_dev, &efx->pci_dev->dev);
44598ff4c7cSJonathan Cooper 
44698ff4c7cSJonathan Cooper 	/* enable all supported features except rx-fcs and rx-all */
44798ff4c7cSJonathan Cooper 	net_dev->features |= efx->type->offload_features &
44898ff4c7cSJonathan Cooper 			     ~(NETIF_F_RXFCS | NETIF_F_RXALL);
44998ff4c7cSJonathan Cooper 	net_dev->hw_features |= efx->type->offload_features;
45098ff4c7cSJonathan Cooper 	net_dev->hw_enc_features |= efx->type->offload_features;
45125414b2aSAlejandro Lucero 	net_dev->vlan_features |= NETIF_F_HW_CSUM | NETIF_F_SG |
45225414b2aSAlejandro Lucero 				  NETIF_F_HIGHDMA | NETIF_F_ALL_TSO;
45398ff4c7cSJonathan Cooper 	netif_set_tso_max_segs(net_dev,
45425414b2aSAlejandro Lucero 			       ESE_EF100_DP_GZ_TSO_MAX_HDR_NUM_SEGS_DEFAULT);
45525414b2aSAlejandro Lucero 	efx->mdio.dev = net_dev;
45625414b2aSAlejandro Lucero 
45725414b2aSAlejandro Lucero 	rc = efx_ef100_init_datapath_caps(efx);
458fa34a514SAlejandro Lucero 	if (rc < 0)
45998ff4c7cSJonathan Cooper 		goto fail;
46098ff4c7cSJonathan Cooper 
461 	rc = ef100_phy_probe(efx);
462 	if (rc)
463 		goto fail;
464 
465 	rc = efx_init_channels(efx);
466 	if (rc)
467 		goto fail;
468 
469 	down_write(&efx->filter_sem);
470 	rc = ef100_filter_table_probe(efx);
471 	up_write(&efx->filter_sem);
472 	if (rc)
473 		goto fail;
474 
475 	netdev_rss_key_fill(efx->rss_context.rx_hash_key,
476 			    sizeof(efx->rss_context.rx_hash_key));
477 
478 	/* Don't fail init if RSS setup doesn't work. */
479 	efx_mcdi_push_default_indir_table(efx, efx->n_rx_channels);
480 
481 	nic_data = efx->nic_data;
482 	rc = ef100_get_mac_address(efx, net_dev->perm_addr, CLIENT_HANDLE_SELF,
483 				   efx->type->is_vf);
484 	if (rc)
485 		return rc;
486 	/* Assign MAC address */
487 	eth_hw_addr_set(net_dev, net_dev->perm_addr);
488 	ether_addr_copy(nic_data->port_id, net_dev->perm_addr);
489 
490 	/* devlink creation, registration and lock */
491 	rc = efx_probe_devlink_and_lock(efx);
492 	if (rc)
493 		pci_info(efx->pci_dev, "devlink registration failed");
494 
495 	rc = ef100_register_netdev(efx);
496 	if (rc)
497 		goto fail;
498 
499 	if (!efx->type->is_vf) {
500 		rc = ef100_probe_netdev_pf(efx);
501 		if (rc)
502 			goto fail;
503 #ifdef CONFIG_SFC_SRIOV
504 		ef100_pf_set_devlink_port(efx);
505 #endif
506 	}
507 
508 	efx->netdev_notifier.notifier_call = ef100_netdev_event;
509 	rc = register_netdevice_notifier(&efx->netdev_notifier);
510 	if (rc) {
511 		netif_err(efx, probe, efx->net_dev,
512 			  "Failed to register netdevice notifier, rc=%d\n", rc);
513 		goto fail;
514 	}
515 
516 	efx->netevent_notifier.notifier_call = ef100_netevent_event;
517 	rc = register_netevent_notifier(&efx->netevent_notifier);
518 	if (rc) {
519 		netif_err(efx, probe, efx->net_dev,
520 			  "Failed to register netevent notifier, rc=%d\n", rc);
521 		goto fail;
522 	}
523 
524 	efx_probe_devlink_unlock(efx);
525 	return rc;
526 fail:
527 #ifdef CONFIG_SFC_SRIOV
528 	/* remove devlink port if does exist */
529 	ef100_pf_unset_devlink_port(efx);
530 #endif
531 	efx_probe_devlink_unlock(efx);
532 	return rc;
533 }
534