blob: b9545bf03798bcf01269da916689097e39085336 [file] [log] [blame]
Shradha Shah834e23d2015-05-06 00:55:58 +01001/****************************************************************************
2 * Driver for Solarflare network controllers and boards
3 * Copyright 2015 Solarflare Communications Inc.
4 *
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 as published
7 * by the Free Software Foundation, incorporated herein by reference.
8 */
9#include <linux/pci.h>
10#include <linux/module.h>
11#include "net_driver.h"
12#include "ef10_sriov.h"
13#include "efx.h"
14#include "nic.h"
15#include "mcdi_pcol.h"
16
Shradha Shah3c5eb872015-05-06 00:58:31 +010017static int efx_ef10_evb_port_assign(struct efx_nic *efx, unsigned int port_id,
18 unsigned int vf_fn)
Shradha Shah834e23d2015-05-06 00:55:58 +010019{
Shradha Shah3c5eb872015-05-06 00:58:31 +010020 MCDI_DECLARE_BUF(inbuf, MC_CMD_EVB_PORT_ASSIGN_IN_LEN);
21 struct efx_ef10_nic_data *nic_data = efx->nic_data;
Shradha Shah834e23d2015-05-06 00:55:58 +010022
Shradha Shah3c5eb872015-05-06 00:58:31 +010023 MCDI_SET_DWORD(inbuf, EVB_PORT_ASSIGN_IN_PORT_ID, port_id);
24 MCDI_POPULATE_DWORD_2(inbuf, EVB_PORT_ASSIGN_IN_FUNCTION,
25 EVB_PORT_ASSIGN_IN_PF, nic_data->pf_index,
26 EVB_PORT_ASSIGN_IN_VF, vf_fn);
27
28 return efx_mcdi_rpc(efx, MC_CMD_EVB_PORT_ASSIGN, inbuf, sizeof(inbuf),
29 NULL, 0, NULL);
Shradha Shah834e23d2015-05-06 00:55:58 +010030}
31
Shradha Shah3c5eb872015-05-06 00:58:31 +010032static int efx_ef10_vport_add_mac(struct efx_nic *efx,
33 unsigned int port_id, u8 *mac)
Shradha Shah834e23d2015-05-06 00:55:58 +010034{
Shradha Shah3c5eb872015-05-06 00:58:31 +010035 MCDI_DECLARE_BUF(inbuf, MC_CMD_VPORT_ADD_MAC_ADDRESS_IN_LEN);
Shradha Shah834e23d2015-05-06 00:55:58 +010036
Shradha Shah3c5eb872015-05-06 00:58:31 +010037 MCDI_SET_DWORD(inbuf, VPORT_ADD_MAC_ADDRESS_IN_VPORT_ID, port_id);
38 ether_addr_copy(MCDI_PTR(inbuf, VPORT_ADD_MAC_ADDRESS_IN_MACADDR), mac);
39
40 return efx_mcdi_rpc(efx, MC_CMD_VPORT_ADD_MAC_ADDRESS, inbuf,
41 sizeof(inbuf), NULL, 0, NULL);
Shradha Shah834e23d2015-05-06 00:55:58 +010042}
43
Shradha Shah3c5eb872015-05-06 00:58:31 +010044static int efx_ef10_vport_del_mac(struct efx_nic *efx,
45 unsigned int port_id, u8 *mac)
Shradha Shah834e23d2015-05-06 00:55:58 +010046{
Shradha Shah3c5eb872015-05-06 00:58:31 +010047 MCDI_DECLARE_BUF(inbuf, MC_CMD_VPORT_DEL_MAC_ADDRESS_IN_LEN);
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +010048
Shradha Shah3c5eb872015-05-06 00:58:31 +010049 MCDI_SET_DWORD(inbuf, VPORT_DEL_MAC_ADDRESS_IN_VPORT_ID, port_id);
50 ether_addr_copy(MCDI_PTR(inbuf, VPORT_DEL_MAC_ADDRESS_IN_MACADDR), mac);
Shradha Shah02246a72015-05-06 00:58:14 +010051
Shradha Shah3c5eb872015-05-06 00:58:31 +010052 return efx_mcdi_rpc(efx, MC_CMD_VPORT_DEL_MAC_ADDRESS, inbuf,
53 sizeof(inbuf), NULL, 0, NULL);
Shradha Shah02246a72015-05-06 00:58:14 +010054}
55
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +010056static int efx_ef10_vswitch_alloc(struct efx_nic *efx, unsigned int port_id,
57 unsigned int vswitch_type)
58{
59 MCDI_DECLARE_BUF(inbuf, MC_CMD_VSWITCH_ALLOC_IN_LEN);
60
61 MCDI_SET_DWORD(inbuf, VSWITCH_ALLOC_IN_UPSTREAM_PORT_ID, port_id);
62 MCDI_SET_DWORD(inbuf, VSWITCH_ALLOC_IN_TYPE, vswitch_type);
63 MCDI_SET_DWORD(inbuf, VSWITCH_ALLOC_IN_NUM_VLAN_TAGS, 0);
64 MCDI_POPULATE_DWORD_1(inbuf, VSWITCH_ALLOC_IN_FLAGS,
65 VSWITCH_ALLOC_IN_FLAG_AUTO_PORT, 0);
66
67 return efx_mcdi_rpc(efx, MC_CMD_VSWITCH_ALLOC, inbuf, sizeof(inbuf),
68 NULL, 0, NULL);
69}
70
71static int efx_ef10_vswitch_free(struct efx_nic *efx, unsigned int port_id)
72{
73 MCDI_DECLARE_BUF(inbuf, MC_CMD_VSWITCH_FREE_IN_LEN);
74
75 MCDI_SET_DWORD(inbuf, VSWITCH_FREE_IN_UPSTREAM_PORT_ID, port_id);
76
77 return efx_mcdi_rpc(efx, MC_CMD_VSWITCH_FREE, inbuf, sizeof(inbuf),
78 NULL, 0, NULL);
79}
80
81static int efx_ef10_vport_alloc(struct efx_nic *efx,
82 unsigned int port_id_in,
83 unsigned int vport_type,
84 unsigned int *port_id_out)
85{
86 MCDI_DECLARE_BUF(inbuf, MC_CMD_VPORT_ALLOC_IN_LEN);
87 MCDI_DECLARE_BUF(outbuf, MC_CMD_VPORT_ALLOC_OUT_LEN);
88 size_t outlen;
89 int rc;
90
91 EFX_WARN_ON_PARANOID(!port_id_out);
92
93 MCDI_SET_DWORD(inbuf, VPORT_ALLOC_IN_UPSTREAM_PORT_ID, port_id_in);
94 MCDI_SET_DWORD(inbuf, VPORT_ALLOC_IN_TYPE, vport_type);
95 MCDI_SET_DWORD(inbuf, VPORT_ALLOC_IN_NUM_VLAN_TAGS, 0);
96 MCDI_POPULATE_DWORD_1(inbuf, VPORT_ALLOC_IN_FLAGS,
97 VPORT_ALLOC_IN_FLAG_AUTO_PORT, 0);
98
99 rc = efx_mcdi_rpc(efx, MC_CMD_VPORT_ALLOC, inbuf, sizeof(inbuf),
100 outbuf, sizeof(outbuf), &outlen);
101 if (rc)
102 return rc;
103 if (outlen < MC_CMD_VPORT_ALLOC_OUT_LEN)
104 return -EIO;
105
106 *port_id_out = MCDI_DWORD(outbuf, VPORT_ALLOC_OUT_VPORT_ID);
107 return 0;
108}
109
110static int efx_ef10_vport_free(struct efx_nic *efx, unsigned int port_id)
111{
112 MCDI_DECLARE_BUF(inbuf, MC_CMD_VPORT_FREE_IN_LEN);
113
114 MCDI_SET_DWORD(inbuf, VPORT_FREE_IN_VPORT_ID, port_id);
115
116 return efx_mcdi_rpc(efx, MC_CMD_VPORT_FREE, inbuf, sizeof(inbuf),
117 NULL, 0, NULL);
118}
119
Shradha Shah3c5eb872015-05-06 00:58:31 +0100120static void efx_ef10_sriov_free_vf_vports(struct efx_nic *efx)
121{
122 struct efx_ef10_nic_data *nic_data = efx->nic_data;
123 int i;
124
125 if (!nic_data->vf)
126 return;
127
128 for (i = 0; i < efx->vf_count; i++) {
129 struct ef10_vf *vf = nic_data->vf + i;
130
131 if (vf->vport_assigned) {
132 efx_ef10_evb_port_assign(efx, EVB_PORT_ID_NULL, i);
133 vf->vport_assigned = 0;
134 }
135
136 if (!is_zero_ether_addr(vf->mac)) {
137 efx_ef10_vport_del_mac(efx, vf->vport_id, vf->mac);
138 eth_zero_addr(vf->mac);
139 }
140
141 if (vf->vport_id) {
142 efx_ef10_vport_free(efx, vf->vport_id);
143 vf->vport_id = 0;
144 }
145 }
146}
147
148static void efx_ef10_sriov_free_vf_vswitching(struct efx_nic *efx)
149{
150 struct efx_ef10_nic_data *nic_data = efx->nic_data;
151
152 efx_ef10_sriov_free_vf_vports(efx);
153 kfree(nic_data->vf);
154 nic_data->vf = NULL;
155}
156
157static int efx_ef10_sriov_assign_vf_vport(struct efx_nic *efx,
158 unsigned int vf_i)
159{
160 struct efx_ef10_nic_data *nic_data = efx->nic_data;
161 struct ef10_vf *vf = nic_data->vf + vf_i;
162 int rc;
163
164 if (WARN_ON_ONCE(!nic_data->vf))
165 return -EOPNOTSUPP;
166
167 rc = efx_ef10_vport_alloc(efx, EVB_PORT_ID_ASSIGNED,
168 MC_CMD_VPORT_ALLOC_IN_VPORT_TYPE_NORMAL,
169 &vf->vport_id);
170 if (rc)
171 return rc;
172
173 rc = efx_ef10_vport_add_mac(efx, vf->vport_id, vf->mac);
174 if (rc) {
175 eth_zero_addr(vf->mac);
176 return rc;
177 }
178
179 rc = efx_ef10_evb_port_assign(efx, vf->vport_id, vf_i);
180 if (rc)
181 return rc;
182
183 vf->vport_assigned = 1;
184 return 0;
185}
186
187static int efx_ef10_sriov_alloc_vf_vswitching(struct efx_nic *efx)
188{
189 struct efx_ef10_nic_data *nic_data = efx->nic_data;
190 unsigned int i;
191 int rc;
192
193 nic_data->vf = kcalloc(efx->vf_count, sizeof(struct ef10_vf),
194 GFP_KERNEL);
195 if (!nic_data->vf)
196 return -ENOMEM;
197
198 for (i = 0; i < efx->vf_count; i++) {
199 random_ether_addr(nic_data->vf[i].mac);
200
201 rc = efx_ef10_sriov_assign_vf_vport(efx, i);
202 if (rc)
203 goto fail;
204 }
205
206 return 0;
207fail:
208 efx_ef10_sriov_free_vf_vports(efx);
209 kfree(nic_data->vf);
210 nic_data->vf = NULL;
211 return rc;
212}
213
214static int efx_ef10_sriov_restore_vf_vswitching(struct efx_nic *efx)
215{
216 unsigned int i;
217 int rc;
218
219 for (i = 0; i < efx->vf_count; i++) {
220 rc = efx_ef10_sriov_assign_vf_vport(efx, i);
221 if (rc)
222 goto fail;
223 }
224
225 return 0;
226fail:
227 efx_ef10_sriov_free_vf_vswitching(efx);
228 return rc;
229}
230
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100231/* On top of the default firmware vswitch setup, create a VEB vswitch and
232 * expansion vport for use by this function.
233 */
234int efx_ef10_vswitching_probe(struct efx_nic *efx)
235{
236 struct efx_ef10_nic_data *nic_data = efx->nic_data;
Shradha Shah3c5eb872015-05-06 00:58:31 +0100237 struct net_device *net_dev = efx->net_dev;
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100238 int rc;
239
240 if (pci_sriov_get_totalvfs(efx->pci_dev) <= 0)
241 return 0; /* vswitch not needed as we have no VFs */
242
243 rc = efx_ef10_vswitch_alloc(efx, EVB_PORT_ID_ASSIGNED,
244 MC_CMD_VSWITCH_ALLOC_IN_VSWITCH_TYPE_VEB);
245 if (rc)
246 goto fail1;
247
248 rc = efx_ef10_vport_alloc(efx, EVB_PORT_ID_ASSIGNED,
249 MC_CMD_VPORT_ALLOC_IN_VPORT_TYPE_NORMAL,
250 &nic_data->vport_id);
251 if (rc)
252 goto fail2;
253
Shradha Shah3c5eb872015-05-06 00:58:31 +0100254 rc = efx_ef10_vport_add_mac(efx, nic_data->vport_id, net_dev->dev_addr);
255 if (rc)
256 goto fail3;
257
258 ether_addr_copy(nic_data->vport_mac, net_dev->dev_addr);
259
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100260 return 0;
Shradha Shah3c5eb872015-05-06 00:58:31 +0100261fail3:
262 efx_ef10_vport_free(efx, nic_data->vport_id);
263 nic_data->vport_id = EVB_PORT_ID_ASSIGNED;
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100264fail2:
265 efx_ef10_vswitch_free(efx, EVB_PORT_ID_ASSIGNED);
266fail1:
267 return rc;
268}
269
270int efx_ef10_vswitching_restore(struct efx_nic *efx)
271{
272 struct efx_ef10_nic_data *nic_data = efx->nic_data;
273 int rc;
274
275 if (!nic_data->must_probe_vswitching)
276 return 0;
277
278 rc = efx_ef10_vswitching_probe(efx);
Shradha Shah3c5eb872015-05-06 00:58:31 +0100279 if (rc)
280 goto fail;
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100281
Shradha Shah3c5eb872015-05-06 00:58:31 +0100282 rc = efx_ef10_sriov_restore_vf_vswitching(efx);
283 if (rc)
284 goto fail;
285
286 nic_data->must_probe_vswitching = false;
287fail:
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100288 return rc;
289}
290
291void efx_ef10_vswitching_remove(struct efx_nic *efx)
292{
293 struct efx_ef10_nic_data *nic_data = efx->nic_data;
294
Shradha Shah3c5eb872015-05-06 00:58:31 +0100295 efx_ef10_sriov_free_vf_vswitching(efx);
296
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100297 if (nic_data->vport_id == EVB_PORT_ID_ASSIGNED)
298 return; /* No vswitch was ever created */
299
Shradha Shah3c5eb872015-05-06 00:58:31 +0100300 if (!is_zero_ether_addr(nic_data->vport_mac)) {
301 efx_ef10_vport_del_mac(efx, nic_data->vport_id,
302 efx->net_dev->dev_addr);
303 eth_zero_addr(nic_data->vport_mac);
304 }
Daniel Pieczko6d8aaaf2015-05-06 00:57:34 +0100305 efx_ef10_vport_free(efx, nic_data->vport_id);
306 nic_data->vport_id = EVB_PORT_ID_ASSIGNED;
307
308 efx_ef10_vswitch_free(efx, nic_data->vport_id);
309}
Shradha Shah3c5eb872015-05-06 00:58:31 +0100310
311static int efx_ef10_pci_sriov_enable(struct efx_nic *efx, int num_vfs)
312{
313 int rc = 0;
314 struct pci_dev *dev = efx->pci_dev;
315
316 efx->vf_count = num_vfs;
317
318 rc = efx_ef10_sriov_alloc_vf_vswitching(efx);
319 if (rc)
320 goto fail1;
321
322 rc = pci_enable_sriov(dev, num_vfs);
323 if (rc)
324 goto fail2;
325
326 return 0;
327fail2:
328 efx_ef10_sriov_free_vf_vswitching(efx);
329fail1:
330 efx->vf_count = 0;
331 netif_err(efx, probe, efx->net_dev,
332 "Failed to enable SRIOV VFs\n");
333 return rc;
334}
335
336static int efx_ef10_pci_sriov_disable(struct efx_nic *efx)
337{
338 struct pci_dev *dev = efx->pci_dev;
339
340 pci_disable_sriov(dev);
341 efx_ef10_sriov_free_vf_vswitching(efx);
342 efx->vf_count = 0;
343 return 0;
344}
345
346int efx_ef10_sriov_configure(struct efx_nic *efx, int num_vfs)
347{
348 if (num_vfs == 0)
349 return efx_ef10_pci_sriov_disable(efx);
350 else
351 return efx_ef10_pci_sriov_enable(efx, num_vfs);
352}
353
354int efx_ef10_sriov_init(struct efx_nic *efx)
355{
356 return 0;
357}
358
359void efx_ef10_sriov_fini(struct efx_nic *efx)
360{
361 struct efx_ef10_nic_data *nic_data = efx->nic_data;
362 int rc;
363
364 if (!nic_data->vf)
365 return;
366
367 rc = efx_ef10_pci_sriov_disable(efx);
368 if (rc)
369 netif_dbg(efx, drv, efx->net_dev,
370 "Disabling SRIOV was not successful rc=%d\n", rc);
371 else
372 netif_dbg(efx, drv, efx->net_dev, "SRIOV disabled\n");
373}