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