blob: 7afd24ce79a558f44589b4ebf34c3d7e4a477653 [file] [log] [blame]
Jakub Kicinskic4c8f392018-05-21 22:12:47 -07001// SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause)
2/*
3 * Copyright (C) 2018 Netronome Systems, Inc.
4 *
5 * This software is dual licensed under the GNU General License Version 2,
6 * June 1991 as shown in the file COPYING in the top-level directory of this
7 * source tree or the BSD 2-Clause License provided below. You have the
8 * option to license this software under the complete terms of either license.
9 *
10 * The BSD 2-Clause License:
11 *
12 * Redistribution and use in source and binary forms, with or
13 * without modification, are permitted provided that the following
14 * conditions are met:
15 *
16 * 1. Redistributions of source code must retain the above
17 * copyright notice, this list of conditions and the following
18 * disclaimer.
19 *
20 * 2. Redistributions in binary form must reproduce the above
21 * copyright notice, this list of conditions and the following
22 * disclaimer in the documentation and/or other materials
23 * provided with the distribution.
24 *
25 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
26 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
27 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
28 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
29 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
30 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
31 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
32 * SOFTWARE.
33 */
34
Jakub Kicinskid05d9022018-05-21 22:12:52 -070035#include <linux/bitfield.h>
Jakub Kicinskicc54dc22018-05-21 22:12:48 -070036#include <linux/etherdevice.h>
Jakub Kicinskid05d9022018-05-21 22:12:52 -070037#include <linux/lockdep.h>
38#include <linux/netdevice.h>
39#include <linux/rcupdate.h>
40#include <linux/slab.h>
Jakub Kicinskicc54dc22018-05-21 22:12:48 -070041
42#include "../nfpcore/nfp.h"
Jakub Kicinskic4c8f392018-05-21 22:12:47 -070043#include "../nfpcore/nfp_cpp.h"
44#include "../nfpcore/nfp_nsp.h"
45#include "../nfp_app.h"
46#include "../nfp_main.h"
Jakub Kicinskicc54dc22018-05-21 22:12:48 -070047#include "../nfp_net.h"
Jakub Kicinskid05d9022018-05-21 22:12:52 -070048#include "../nfp_net_repr.h"
Jakub Kicinskicc54dc22018-05-21 22:12:48 -070049#include "../nfp_port.h"
Jakub Kicinskic4c8f392018-05-21 22:12:47 -070050#include "main.h"
51
Jakub Kicinskid05d9022018-05-21 22:12:52 -070052static u32 nfp_abm_portid(enum nfp_repr_type rtype, unsigned int id)
53{
54 return FIELD_PREP(NFP_ABM_PORTID_TYPE, rtype) |
55 FIELD_PREP(NFP_ABM_PORTID_ID, id);
56}
57
58static struct net_device *nfp_abm_repr_get(struct nfp_app *app, u32 port_id)
59{
60 enum nfp_repr_type rtype;
61 struct nfp_reprs *reprs;
62 u8 port;
63
64 rtype = FIELD_GET(NFP_ABM_PORTID_TYPE, port_id);
65 port = FIELD_GET(NFP_ABM_PORTID_ID, port_id);
66
67 reprs = rcu_dereference(app->reprs[rtype]);
68 if (!reprs)
69 return NULL;
70
71 if (port >= reprs->num_reprs)
72 return NULL;
73
74 return rcu_dereference(reprs->reprs[port]);
75}
76
77static int
78nfp_abm_spawn_repr(struct nfp_app *app, struct nfp_abm_link *alink,
79 enum nfp_port_type ptype)
80{
81 struct net_device *netdev;
82 enum nfp_repr_type rtype;
83 struct nfp_reprs *reprs;
84 struct nfp_repr *repr;
85 struct nfp_port *port;
86 int err;
87
88 if (ptype == NFP_PORT_PHYS_PORT)
89 rtype = NFP_REPR_TYPE_PHYS_PORT;
90 else
91 rtype = NFP_REPR_TYPE_PF;
92
93 netdev = nfp_repr_alloc(app);
94 if (!netdev)
95 return -ENOMEM;
96 repr = netdev_priv(netdev);
97 repr->app_priv = alink;
98
99 port = nfp_port_alloc(app, ptype, netdev);
100 if (IS_ERR(port)) {
101 err = PTR_ERR(port);
102 goto err_free_repr;
103 }
104
105 if (ptype == NFP_PORT_PHYS_PORT) {
Jakub Kicinski1f700362018-05-21 22:12:53 -0700106 port->eth_forced = true;
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700107 err = nfp_port_init_phy_port(app->pf, app, port, alink->id);
108 if (err)
109 goto err_free_port;
110 } else {
111 port->pf_id = alink->abm->pf_id;
112 port->vnic = alink->vnic->dp.ctrl_bar;
113 }
114
115 SET_NETDEV_DEV(netdev, &alink->vnic->pdev->dev);
116 eth_hw_addr_random(netdev);
117
118 err = nfp_repr_init(app, netdev, nfp_abm_portid(rtype, alink->id),
119 port, alink->vnic->dp.netdev);
120 if (err)
121 goto err_free_port;
122
123 reprs = nfp_reprs_get_locked(app, rtype);
124 WARN(nfp_repr_get_locked(app, reprs, alink->id), "duplicate repr");
125 rcu_assign_pointer(reprs->reprs[alink->id], netdev);
126
127 nfp_info(app->cpp, "%s Port %d Representor(%s) created\n",
128 ptype == NFP_PORT_PF_PORT ? "PCIe" : "Phys",
129 alink->id, netdev->name);
130
131 return 0;
132
133err_free_port:
134 nfp_port_free(port);
135err_free_repr:
136 nfp_repr_free(netdev);
137 return err;
138}
139
140static void
141nfp_abm_kill_repr(struct nfp_app *app, struct nfp_abm_link *alink,
142 enum nfp_repr_type rtype)
143{
144 struct net_device *netdev;
145 struct nfp_reprs *reprs;
146
147 reprs = nfp_reprs_get_locked(app, rtype);
148 netdev = nfp_repr_get_locked(app, reprs, alink->id);
149 if (!netdev)
150 return;
151 rcu_assign_pointer(reprs->reprs[alink->id], NULL);
152 synchronize_rcu();
153 /* Cast to make sure nfp_repr_clean_and_free() takes a nfp_repr */
154 nfp_repr_clean_and_free((struct nfp_repr *)netdev_priv(netdev));
155}
156
157static void
158nfp_abm_kill_reprs(struct nfp_abm *abm, struct nfp_abm_link *alink)
159{
160 nfp_abm_kill_repr(abm->app, alink, NFP_REPR_TYPE_PF);
161 nfp_abm_kill_repr(abm->app, alink, NFP_REPR_TYPE_PHYS_PORT);
162}
163
164static void nfp_abm_kill_reprs_all(struct nfp_abm *abm)
165{
166 struct nfp_pf *pf = abm->app->pf;
167 struct nfp_net *nn;
168
169 list_for_each_entry(nn, &pf->vnics, vnic_list)
170 nfp_abm_kill_reprs(abm, (struct nfp_abm_link *)nn->app_priv);
171}
172
173static enum devlink_eswitch_mode nfp_abm_eswitch_mode_get(struct nfp_app *app)
174{
175 struct nfp_abm *abm = app->priv;
176
177 return abm->eswitch_mode;
178}
179
180static int nfp_abm_eswitch_set_legacy(struct nfp_abm *abm)
181{
182 nfp_abm_kill_reprs_all(abm);
183
184 abm->eswitch_mode = DEVLINK_ESWITCH_MODE_LEGACY;
185 return 0;
186}
187
188static void nfp_abm_eswitch_clean_up(struct nfp_abm *abm)
189{
190 if (abm->eswitch_mode != DEVLINK_ESWITCH_MODE_LEGACY)
191 WARN_ON(nfp_abm_eswitch_set_legacy(abm));
192}
193
194static int nfp_abm_eswitch_set_switchdev(struct nfp_abm *abm)
195{
196 struct nfp_app *app = abm->app;
197 struct nfp_pf *pf = app->pf;
198 struct nfp_net *nn;
199 int err;
200
201 list_for_each_entry(nn, &pf->vnics, vnic_list) {
202 struct nfp_abm_link *alink = nn->app_priv;
203
204 err = nfp_abm_spawn_repr(app, alink, NFP_PORT_PHYS_PORT);
205 if (err)
206 goto err_kill_all_reprs;
207
208 err = nfp_abm_spawn_repr(app, alink, NFP_PORT_PF_PORT);
209 if (err)
210 goto err_kill_all_reprs;
211 }
212
213 abm->eswitch_mode = DEVLINK_ESWITCH_MODE_SWITCHDEV;
214 return 0;
215
216err_kill_all_reprs:
217 nfp_abm_kill_reprs_all(abm);
218 return err;
219}
220
221static int nfp_abm_eswitch_mode_set(struct nfp_app *app, u16 mode)
222{
223 struct nfp_abm *abm = app->priv;
224
225 if (abm->eswitch_mode == mode)
226 return 0;
227
228 switch (mode) {
229 case DEVLINK_ESWITCH_MODE_LEGACY:
230 return nfp_abm_eswitch_set_legacy(abm);
231 case DEVLINK_ESWITCH_MODE_SWITCHDEV:
232 return nfp_abm_eswitch_set_switchdev(abm);
233 default:
234 return -EINVAL;
235 }
236}
237
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700238static void
239nfp_abm_vnic_set_mac(struct nfp_pf *pf, struct nfp_abm *abm, struct nfp_net *nn,
240 unsigned int id)
241{
242 struct nfp_eth_table_port *eth_port = &pf->eth_tbl->ports[id];
243 u8 mac_addr[ETH_ALEN];
244 const char *mac_str;
245 char name[32];
246
247 if (id > pf->eth_tbl->count) {
248 nfp_warn(pf->cpp, "No entry for persistent MAC address\n");
249 eth_hw_addr_random(nn->dp.netdev);
250 return;
251 }
252
253 snprintf(name, sizeof(name), "eth%u.mac.pf%u",
254 eth_port->eth_index, abm->pf_id);
255
256 mac_str = nfp_hwinfo_lookup(pf->hwinfo, name);
257 if (!mac_str) {
258 nfp_warn(pf->cpp, "Can't lookup persistent MAC address (%s)\n",
259 name);
260 eth_hw_addr_random(nn->dp.netdev);
261 return;
262 }
263
264 if (sscanf(mac_str, "%02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx",
265 &mac_addr[0], &mac_addr[1], &mac_addr[2],
266 &mac_addr[3], &mac_addr[4], &mac_addr[5]) != 6) {
267 nfp_warn(pf->cpp, "Can't parse persistent MAC address (%s)\n",
268 mac_str);
269 eth_hw_addr_random(nn->dp.netdev);
270 return;
271 }
272
273 ether_addr_copy(nn->dp.netdev->dev_addr, mac_addr);
274 ether_addr_copy(nn->dp.netdev->perm_addr, mac_addr);
275}
276
277static int
278nfp_abm_vnic_alloc(struct nfp_app *app, struct nfp_net *nn, unsigned int id)
279{
Jakub Kicinski1f700362018-05-21 22:12:53 -0700280 struct nfp_eth_table_port *eth_port = &app->pf->eth_tbl->ports[id];
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700281 struct nfp_abm *abm = app->priv;
282 struct nfp_abm_link *alink;
Jakub Kicinski1f700362018-05-21 22:12:53 -0700283 int err;
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700284
285 alink = kzalloc(sizeof(*alink), GFP_KERNEL);
286 if (!alink)
287 return -ENOMEM;
288 nn->app_priv = alink;
289 alink->abm = abm;
290 alink->vnic = nn;
291 alink->id = id;
292
Jakub Kicinski1f700362018-05-21 22:12:53 -0700293 /* This is a multi-host app, make sure MAC/PHY is up, but don't
294 * make the MAC/PHY state follow the state of any of the ports.
295 */
296 err = nfp_eth_set_configured(app->cpp, eth_port->index, true);
297 if (err < 0)
298 goto err_free_alink;
299
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700300 netif_keep_dst(nn->dp.netdev);
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700301
302 nfp_abm_vnic_set_mac(app->pf, abm, nn, id);
303 nfp_abm_ctrl_read_params(alink);
304
305 return 0;
Jakub Kicinski1f700362018-05-21 22:12:53 -0700306
307err_free_alink:
308 kfree(alink);
309 return err;
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700310}
311
312static void nfp_abm_vnic_free(struct nfp_app *app, struct nfp_net *nn)
313{
314 struct nfp_abm_link *alink = nn->app_priv;
315
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700316 nfp_abm_kill_reprs(alink->abm, alink);
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700317 kfree(alink);
318}
319
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700320static int nfp_abm_init(struct nfp_app *app)
321{
322 struct nfp_pf *pf = app->pf;
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700323 struct nfp_reprs *reprs;
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700324 struct nfp_abm *abm;
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700325 int err;
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700326
327 if (!pf->eth_tbl) {
328 nfp_err(pf->cpp, "ABM NIC requires ETH table\n");
329 return -EINVAL;
330 }
331 if (pf->max_data_vnics != pf->eth_tbl->count) {
332 nfp_err(pf->cpp, "ETH entries don't match vNICs (%d vs %d)\n",
333 pf->max_data_vnics, pf->eth_tbl->count);
334 return -EINVAL;
335 }
336 if (!pf->mac_stats_bar) {
337 nfp_warn(app->cpp, "ABM NIC requires mac_stats symbol\n");
338 return -EINVAL;
339 }
340
341 abm = kzalloc(sizeof(*abm), GFP_KERNEL);
342 if (!abm)
343 return -ENOMEM;
344 app->priv = abm;
345 abm->app = app;
346
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700347 err = nfp_abm_ctrl_find_addrs(abm);
348 if (err)
349 goto err_free_abm;
350
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700351 err = -ENOMEM;
352 reprs = nfp_reprs_alloc(pf->max_data_vnics);
353 if (!reprs)
354 goto err_free_abm;
355 RCU_INIT_POINTER(app->reprs[NFP_REPR_TYPE_PHYS_PORT], reprs);
356
357 reprs = nfp_reprs_alloc(pf->max_data_vnics);
358 if (!reprs)
359 goto err_free_phys;
360 RCU_INIT_POINTER(app->reprs[NFP_REPR_TYPE_PF], reprs);
361
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700362 return 0;
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700363
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700364err_free_phys:
365 nfp_reprs_clean_and_free_by_type(app, NFP_REPR_TYPE_PHYS_PORT);
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700366err_free_abm:
367 kfree(abm);
368 app->priv = NULL;
369 return err;
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700370}
371
372static void nfp_abm_clean(struct nfp_app *app)
373{
374 struct nfp_abm *abm = app->priv;
375
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700376 nfp_abm_eswitch_clean_up(abm);
377 nfp_reprs_clean_and_free_by_type(app, NFP_REPR_TYPE_PF);
378 nfp_reprs_clean_and_free_by_type(app, NFP_REPR_TYPE_PHYS_PORT);
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700379 kfree(abm);
380 app->priv = NULL;
381}
382
383const struct nfp_app_type app_abm = {
384 .id = NFP_APP_ACTIVE_BUFFER_MGMT_NIC,
385 .name = "abm",
386
387 .init = nfp_abm_init,
388 .clean = nfp_abm_clean,
389
Jakub Kicinskicc54dc22018-05-21 22:12:48 -0700390 .vnic_alloc = nfp_abm_vnic_alloc,
391 .vnic_free = nfp_abm_vnic_free,
Jakub Kicinskid05d9022018-05-21 22:12:52 -0700392
393 .eswitch_mode_get = nfp_abm_eswitch_mode_get,
394 .eswitch_mode_set = nfp_abm_eswitch_mode_set,
395
396 .repr_get = nfp_abm_repr_get,
Jakub Kicinskic4c8f392018-05-21 22:12:47 -0700397};