1ee6373ddSDimitris Michailidis // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
2ee6373ddSDimitris Michailidis 
3ee6373ddSDimitris Michailidis #include <linux/bpf.h>
4ee6373ddSDimitris Michailidis #include <linux/crash_dump.h>
5ee6373ddSDimitris Michailidis #include <linux/etherdevice.h>
6ee6373ddSDimitris Michailidis #include <linux/ethtool.h>
7ee6373ddSDimitris Michailidis #include <linux/filter.h>
8ee6373ddSDimitris Michailidis #include <linux/idr.h>
9ee6373ddSDimitris Michailidis #include <linux/if_vlan.h>
10ee6373ddSDimitris Michailidis #include <linux/module.h>
11ee6373ddSDimitris Michailidis #include <linux/netdevice.h>
12ee6373ddSDimitris Michailidis #include <linux/pci.h>
13ee6373ddSDimitris Michailidis #include <linux/rtnetlink.h>
14ee6373ddSDimitris Michailidis #include <linux/inetdevice.h>
15ee6373ddSDimitris Michailidis 
16ee6373ddSDimitris Michailidis #include "funeth.h"
17ee6373ddSDimitris Michailidis #include "funeth_devlink.h"
18ee6373ddSDimitris Michailidis #include "funeth_ktls.h"
19ee6373ddSDimitris Michailidis #include "fun_port.h"
20ee6373ddSDimitris Michailidis #include "fun_queue.h"
21ee6373ddSDimitris Michailidis #include "funeth_txrx.h"
22ee6373ddSDimitris Michailidis 
23ee6373ddSDimitris Michailidis #define ADMIN_SQ_DEPTH 32
24ee6373ddSDimitris Michailidis #define ADMIN_CQ_DEPTH 64
25ee6373ddSDimitris Michailidis #define ADMIN_RQ_DEPTH 16
26ee6373ddSDimitris Michailidis 
27ee6373ddSDimitris Michailidis /* Default number of Tx/Rx queues. */
28ee6373ddSDimitris Michailidis #define FUN_DFLT_QUEUES 16U
29ee6373ddSDimitris Michailidis 
30ee6373ddSDimitris Michailidis enum {
31ee6373ddSDimitris Michailidis 	FUN_SERV_RES_CHANGE = FUN_SERV_FIRST_AVAIL,
32ee6373ddSDimitris Michailidis 	FUN_SERV_DEL_PORTS,
33ee6373ddSDimitris Michailidis };
34ee6373ddSDimitris Michailidis 
35ee6373ddSDimitris Michailidis static const struct pci_device_id funeth_id_table[] = {
36ee6373ddSDimitris Michailidis 	{ PCI_VDEVICE(FUNGIBLE, 0x0101) },
37ee6373ddSDimitris Michailidis 	{ PCI_VDEVICE(FUNGIBLE, 0x0181) },
38ee6373ddSDimitris Michailidis 	{ 0, }
39ee6373ddSDimitris Michailidis };
40ee6373ddSDimitris Michailidis 
41ee6373ddSDimitris Michailidis /* Issue a port write admin command with @n key/value pairs. */
fun_port_write_cmds(struct funeth_priv * fp,unsigned int n,const int * keys,const u64 * data)42ee6373ddSDimitris Michailidis static int fun_port_write_cmds(struct funeth_priv *fp, unsigned int n,
43ee6373ddSDimitris Michailidis 			       const int *keys, const u64 *data)
44ee6373ddSDimitris Michailidis {
45ee6373ddSDimitris Michailidis 	unsigned int cmd_size, i;
46ee6373ddSDimitris Michailidis 	union {
47ee6373ddSDimitris Michailidis 		struct fun_admin_port_req req;
48ee6373ddSDimitris Michailidis 		struct fun_admin_port_rsp rsp;
49ee6373ddSDimitris Michailidis 		u8 v[ADMIN_SQE_SIZE];
50ee6373ddSDimitris Michailidis 	} cmd;
51ee6373ddSDimitris Michailidis 
52ee6373ddSDimitris Michailidis 	cmd_size = offsetof(struct fun_admin_port_req, u.write.write48) +
53ee6373ddSDimitris Michailidis 		n * sizeof(struct fun_admin_write48_req);
54ee6373ddSDimitris Michailidis 	if (cmd_size > sizeof(cmd) || cmd_size > ADMIN_RSP_MAX_LEN)
55ee6373ddSDimitris Michailidis 		return -EINVAL;
56ee6373ddSDimitris Michailidis 
57ee6373ddSDimitris Michailidis 	cmd.req.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_PORT,
58ee6373ddSDimitris Michailidis 						    cmd_size);
59ee6373ddSDimitris Michailidis 	cmd.req.u.write =
60ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_WRITE_REQ_INIT(FUN_ADMIN_SUBOP_WRITE, 0,
61ee6373ddSDimitris Michailidis 					      fp->netdev->dev_port);
62ee6373ddSDimitris Michailidis 	for (i = 0; i < n; i++)
63ee6373ddSDimitris Michailidis 		cmd.req.u.write.write48[i] =
64ee6373ddSDimitris Michailidis 			FUN_ADMIN_WRITE48_REQ_INIT(keys[i], data[i]);
65ee6373ddSDimitris Michailidis 
66ee6373ddSDimitris Michailidis 	return fun_submit_admin_sync_cmd(fp->fdev, &cmd.req.common,
67ee6373ddSDimitris Michailidis 					 &cmd.rsp, cmd_size, 0);
68ee6373ddSDimitris Michailidis }
69ee6373ddSDimitris Michailidis 
fun_port_write_cmd(struct funeth_priv * fp,int key,u64 data)70ee6373ddSDimitris Michailidis int fun_port_write_cmd(struct funeth_priv *fp, int key, u64 data)
71ee6373ddSDimitris Michailidis {
72ee6373ddSDimitris Michailidis 	return fun_port_write_cmds(fp, 1, &key, &data);
73ee6373ddSDimitris Michailidis }
74ee6373ddSDimitris Michailidis 
75ee6373ddSDimitris Michailidis /* Issue a port read admin command with @n key/value pairs. */
fun_port_read_cmds(struct funeth_priv * fp,unsigned int n,const int * keys,u64 * data)76ee6373ddSDimitris Michailidis static int fun_port_read_cmds(struct funeth_priv *fp, unsigned int n,
77ee6373ddSDimitris Michailidis 			      const int *keys, u64 *data)
78ee6373ddSDimitris Michailidis {
79ee6373ddSDimitris Michailidis 	const struct fun_admin_read48_rsp *r48rsp;
80ee6373ddSDimitris Michailidis 	unsigned int cmd_size, i;
81ee6373ddSDimitris Michailidis 	int rc;
82ee6373ddSDimitris Michailidis 	union {
83ee6373ddSDimitris Michailidis 		struct fun_admin_port_req req;
84ee6373ddSDimitris Michailidis 		struct fun_admin_port_rsp rsp;
85ee6373ddSDimitris Michailidis 		u8 v[ADMIN_SQE_SIZE];
86ee6373ddSDimitris Michailidis 	} cmd;
87ee6373ddSDimitris Michailidis 
88ee6373ddSDimitris Michailidis 	cmd_size = offsetof(struct fun_admin_port_req, u.read.read48) +
89ee6373ddSDimitris Michailidis 		n * sizeof(struct fun_admin_read48_req);
90ee6373ddSDimitris Michailidis 	if (cmd_size > sizeof(cmd) || cmd_size > ADMIN_RSP_MAX_LEN)
91ee6373ddSDimitris Michailidis 		return -EINVAL;
92ee6373ddSDimitris Michailidis 
93ee6373ddSDimitris Michailidis 	cmd.req.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_PORT,
94ee6373ddSDimitris Michailidis 						    cmd_size);
95ee6373ddSDimitris Michailidis 	cmd.req.u.read =
96ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_READ_REQ_INIT(FUN_ADMIN_SUBOP_READ, 0,
97ee6373ddSDimitris Michailidis 					     fp->netdev->dev_port);
98ee6373ddSDimitris Michailidis 	for (i = 0; i < n; i++)
99ee6373ddSDimitris Michailidis 		cmd.req.u.read.read48[i] = FUN_ADMIN_READ48_REQ_INIT(keys[i]);
100ee6373ddSDimitris Michailidis 
101ee6373ddSDimitris Michailidis 	rc = fun_submit_admin_sync_cmd(fp->fdev, &cmd.req.common,
102ee6373ddSDimitris Michailidis 				       &cmd.rsp, cmd_size, 0);
103ee6373ddSDimitris Michailidis 	if (rc)
104ee6373ddSDimitris Michailidis 		return rc;
105ee6373ddSDimitris Michailidis 
106ee6373ddSDimitris Michailidis 	for (r48rsp = cmd.rsp.u.read.read48, i = 0; i < n; i++, r48rsp++) {
107ee6373ddSDimitris Michailidis 		data[i] = FUN_ADMIN_READ48_RSP_DATA_G(r48rsp->key_to_data);
108ee6373ddSDimitris Michailidis 		dev_dbg(fp->fdev->dev,
109ee6373ddSDimitris Michailidis 			"port_read_rsp lport=%u (key_to_data=0x%llx) key=%d data:%lld retval:%lld",
110ee6373ddSDimitris Michailidis 			fp->lport, r48rsp->key_to_data, keys[i], data[i],
111ee6373ddSDimitris Michailidis 			FUN_ADMIN_READ48_RSP_RET_G(r48rsp->key_to_data));
112ee6373ddSDimitris Michailidis 	}
113ee6373ddSDimitris Michailidis 	return 0;
114ee6373ddSDimitris Michailidis }
115ee6373ddSDimitris Michailidis 
fun_port_read_cmd(struct funeth_priv * fp,int key,u64 * data)116ee6373ddSDimitris Michailidis int fun_port_read_cmd(struct funeth_priv *fp, int key, u64 *data)
117ee6373ddSDimitris Michailidis {
118ee6373ddSDimitris Michailidis 	return fun_port_read_cmds(fp, 1, &key, data);
119ee6373ddSDimitris Michailidis }
120ee6373ddSDimitris Michailidis 
fun_report_link(struct net_device * netdev)121ee6373ddSDimitris Michailidis static void fun_report_link(struct net_device *netdev)
122ee6373ddSDimitris Michailidis {
123ee6373ddSDimitris Michailidis 	if (netif_carrier_ok(netdev)) {
124ee6373ddSDimitris Michailidis 		const struct funeth_priv *fp = netdev_priv(netdev);
125ee6373ddSDimitris Michailidis 		const char *fec = "", *pause = "";
126ee6373ddSDimitris Michailidis 		int speed = fp->link_speed;
127ee6373ddSDimitris Michailidis 		char unit = 'M';
128ee6373ddSDimitris Michailidis 
129ee6373ddSDimitris Michailidis 		if (fp->link_speed >= SPEED_1000) {
130ee6373ddSDimitris Michailidis 			speed /= 1000;
131ee6373ddSDimitris Michailidis 			unit = 'G';
132ee6373ddSDimitris Michailidis 		}
133ee6373ddSDimitris Michailidis 
134ee6373ddSDimitris Michailidis 		if (fp->active_fec & FUN_PORT_FEC_RS)
135ee6373ddSDimitris Michailidis 			fec = ", RS-FEC";
136ee6373ddSDimitris Michailidis 		else if (fp->active_fec & FUN_PORT_FEC_FC)
137ee6373ddSDimitris Michailidis 			fec = ", BASER-FEC";
138ee6373ddSDimitris Michailidis 
139ee6373ddSDimitris Michailidis 		if ((fp->active_fc & FUN_PORT_CAP_PAUSE_MASK) == FUN_PORT_CAP_PAUSE_MASK)
140ee6373ddSDimitris Michailidis 			pause = ", Tx/Rx PAUSE";
141ee6373ddSDimitris Michailidis 		else if (fp->active_fc & FUN_PORT_CAP_RX_PAUSE)
142ee6373ddSDimitris Michailidis 			pause = ", Rx PAUSE";
143ee6373ddSDimitris Michailidis 		else if (fp->active_fc & FUN_PORT_CAP_TX_PAUSE)
144ee6373ddSDimitris Michailidis 			pause = ", Tx PAUSE";
145ee6373ddSDimitris Michailidis 
146ee6373ddSDimitris Michailidis 		netdev_info(netdev, "Link up at %d %cb/s full-duplex%s%s\n",
147ee6373ddSDimitris Michailidis 			    speed, unit, pause, fec);
148ee6373ddSDimitris Michailidis 	} else {
149ee6373ddSDimitris Michailidis 		netdev_info(netdev, "Link down\n");
150ee6373ddSDimitris Michailidis 	}
151ee6373ddSDimitris Michailidis }
152ee6373ddSDimitris Michailidis 
fun_adi_write(struct fun_dev * fdev,enum fun_admin_adi_attr attr,unsigned int adi_id,const struct fun_adi_param * param)153ee6373ddSDimitris Michailidis static int fun_adi_write(struct fun_dev *fdev, enum fun_admin_adi_attr attr,
154ee6373ddSDimitris Michailidis 			 unsigned int adi_id, const struct fun_adi_param *param)
155ee6373ddSDimitris Michailidis {
156ee6373ddSDimitris Michailidis 	struct fun_admin_adi_req req = {
157ee6373ddSDimitris Michailidis 		.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_ADI,
158ee6373ddSDimitris Michailidis 						     sizeof(req)),
159ee6373ddSDimitris Michailidis 		.u.write.subop = FUN_ADMIN_SUBOP_WRITE,
160ee6373ddSDimitris Michailidis 		.u.write.attribute = attr,
161ee6373ddSDimitris Michailidis 		.u.write.id = cpu_to_be32(adi_id),
162ee6373ddSDimitris Michailidis 		.u.write.param = *param
163ee6373ddSDimitris Michailidis 	};
164ee6373ddSDimitris Michailidis 
165ee6373ddSDimitris Michailidis 	return fun_submit_admin_sync_cmd(fdev, &req.common, NULL, 0, 0);
166ee6373ddSDimitris Michailidis }
167ee6373ddSDimitris Michailidis 
168ee6373ddSDimitris Michailidis /* Configure RSS for the given port. @op determines whether a new RSS context
169ee6373ddSDimitris Michailidis  * is to be created or whether an existing one should be reconfigured. The
170ee6373ddSDimitris Michailidis  * remaining parameters specify the hashing algorithm, key, and indirection
171ee6373ddSDimitris Michailidis  * table.
172ee6373ddSDimitris Michailidis  *
173ee6373ddSDimitris Michailidis  * This initiates packet delivery to the Rx queues set in the indirection
174ee6373ddSDimitris Michailidis  * table.
175ee6373ddSDimitris Michailidis  */
fun_config_rss(struct net_device * dev,int algo,const u8 * key,const u32 * qtable,u8 op)176ee6373ddSDimitris Michailidis int fun_config_rss(struct net_device *dev, int algo, const u8 *key,
177ee6373ddSDimitris Michailidis 		   const u32 *qtable, u8 op)
178ee6373ddSDimitris Michailidis {
179ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
180ee6373ddSDimitris Michailidis 	unsigned int table_len = fp->indir_table_nentries;
181ee6373ddSDimitris Michailidis 	unsigned int len = FUN_ETH_RSS_MAX_KEY_SIZE + sizeof(u32) * table_len;
182ee6373ddSDimitris Michailidis 	struct funeth_rxq **rxqs = rtnl_dereference(fp->rxqs);
183ee6373ddSDimitris Michailidis 	union {
184ee6373ddSDimitris Michailidis 		struct {
185ee6373ddSDimitris Michailidis 			struct fun_admin_rss_req req;
186ee6373ddSDimitris Michailidis 			struct fun_dataop_gl gl;
187ee6373ddSDimitris Michailidis 		};
188ee6373ddSDimitris Michailidis 		struct fun_admin_generic_create_rsp rsp;
189ee6373ddSDimitris Michailidis 	} cmd;
190ee6373ddSDimitris Michailidis 	__be32 *indir_tab;
191ee6373ddSDimitris Michailidis 	u16 flags;
192ee6373ddSDimitris Michailidis 	int rc;
193ee6373ddSDimitris Michailidis 
194ee6373ddSDimitris Michailidis 	if (op != FUN_ADMIN_SUBOP_CREATE && fp->rss_hw_id == FUN_HCI_ID_INVALID)
195ee6373ddSDimitris Michailidis 		return -EINVAL;
196ee6373ddSDimitris Michailidis 
197ee6373ddSDimitris Michailidis 	flags = op == FUN_ADMIN_SUBOP_CREATE ?
198ee6373ddSDimitris Michailidis 			FUN_ADMIN_RES_CREATE_FLAG_ALLOCATOR : 0;
199ee6373ddSDimitris Michailidis 	cmd.req.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_RSS,
200ee6373ddSDimitris Michailidis 						    sizeof(cmd));
201ee6373ddSDimitris Michailidis 	cmd.req.u.create =
202ee6373ddSDimitris Michailidis 		FUN_ADMIN_RSS_CREATE_REQ_INIT(op, flags, fp->rss_hw_id,
203ee6373ddSDimitris Michailidis 					      dev->dev_port, algo,
204ee6373ddSDimitris Michailidis 					      FUN_ETH_RSS_MAX_KEY_SIZE,
205ee6373ddSDimitris Michailidis 					      table_len, 0,
206ee6373ddSDimitris Michailidis 					      FUN_ETH_RSS_MAX_KEY_SIZE);
207ee6373ddSDimitris Michailidis 	cmd.req.u.create.dataop = FUN_DATAOP_HDR_INIT(1, 0, 1, 0, len);
208ee6373ddSDimitris Michailidis 	fun_dataop_gl_init(&cmd.gl, 0, 0, len, fp->rss_dma_addr);
209ee6373ddSDimitris Michailidis 
210ee6373ddSDimitris Michailidis 	/* write the key and indirection table into the RSS DMA area */
211ee6373ddSDimitris Michailidis 	memcpy(fp->rss_cfg, key, FUN_ETH_RSS_MAX_KEY_SIZE);
212ee6373ddSDimitris Michailidis 	indir_tab = fp->rss_cfg + FUN_ETH_RSS_MAX_KEY_SIZE;
213ee6373ddSDimitris Michailidis 	for (rc = 0; rc < table_len; rc++)
214ee6373ddSDimitris Michailidis 		*indir_tab++ = cpu_to_be32(rxqs[*qtable++]->hw_cqid);
215ee6373ddSDimitris Michailidis 
216ee6373ddSDimitris Michailidis 	rc = fun_submit_admin_sync_cmd(fp->fdev, &cmd.req.common,
217ee6373ddSDimitris Michailidis 				       &cmd.rsp, sizeof(cmd.rsp), 0);
218ee6373ddSDimitris Michailidis 	if (!rc && op == FUN_ADMIN_SUBOP_CREATE)
219ee6373ddSDimitris Michailidis 		fp->rss_hw_id = be32_to_cpu(cmd.rsp.id);
220ee6373ddSDimitris Michailidis 	return rc;
221ee6373ddSDimitris Michailidis }
222ee6373ddSDimitris Michailidis 
223ee6373ddSDimitris Michailidis /* Destroy the HW RSS conntext associated with the given port. This also stops
224ee6373ddSDimitris Michailidis  * all packet delivery to our Rx queues.
225ee6373ddSDimitris Michailidis  */
fun_destroy_rss(struct funeth_priv * fp)226ee6373ddSDimitris Michailidis static void fun_destroy_rss(struct funeth_priv *fp)
227ee6373ddSDimitris Michailidis {
228ee6373ddSDimitris Michailidis 	if (fp->rss_hw_id != FUN_HCI_ID_INVALID) {
229ee6373ddSDimitris Michailidis 		fun_res_destroy(fp->fdev, FUN_ADMIN_OP_RSS, 0, fp->rss_hw_id);
230ee6373ddSDimitris Michailidis 		fp->rss_hw_id = FUN_HCI_ID_INVALID;
231ee6373ddSDimitris Michailidis 	}
232ee6373ddSDimitris Michailidis }
233ee6373ddSDimitris Michailidis 
fun_irq_aff_notify(struct irq_affinity_notify * notify,const cpumask_t * mask)234ee6373ddSDimitris Michailidis static void fun_irq_aff_notify(struct irq_affinity_notify *notify,
235ee6373ddSDimitris Michailidis 			       const cpumask_t *mask)
236ee6373ddSDimitris Michailidis {
237ee6373ddSDimitris Michailidis 	struct fun_irq *p = container_of(notify, struct fun_irq, aff_notify);
238ee6373ddSDimitris Michailidis 
239ee6373ddSDimitris Michailidis 	cpumask_copy(&p->affinity_mask, mask);
240ee6373ddSDimitris Michailidis }
241ee6373ddSDimitris Michailidis 
fun_irq_aff_release(struct kref __always_unused * ref)242ee6373ddSDimitris Michailidis static void fun_irq_aff_release(struct kref __always_unused *ref)
243ee6373ddSDimitris Michailidis {
244ee6373ddSDimitris Michailidis }
245ee6373ddSDimitris Michailidis 
246ee6373ddSDimitris Michailidis /* Allocate an IRQ structure, assign an MSI-X index and initial affinity to it,
247ee6373ddSDimitris Michailidis  * and add it to the IRQ XArray.
248ee6373ddSDimitris Michailidis  */
fun_alloc_qirq(struct funeth_priv * fp,unsigned int idx,int node,unsigned int xa_idx_offset)249ee6373ddSDimitris Michailidis static struct fun_irq *fun_alloc_qirq(struct funeth_priv *fp, unsigned int idx,
250ee6373ddSDimitris Michailidis 				      int node, unsigned int xa_idx_offset)
251ee6373ddSDimitris Michailidis {
252ee6373ddSDimitris Michailidis 	struct fun_irq *irq;
253ee6373ddSDimitris Michailidis 	int cpu, res;
254ee6373ddSDimitris Michailidis 
255ee6373ddSDimitris Michailidis 	cpu = cpumask_local_spread(idx, node);
256cdba2490SDimitris Michailidis 	node = cpu_to_mem(cpu);
257ee6373ddSDimitris Michailidis 
258ee6373ddSDimitris Michailidis 	irq = kzalloc_node(sizeof(*irq), GFP_KERNEL, node);
259ee6373ddSDimitris Michailidis 	if (!irq)
260ee6373ddSDimitris Michailidis 		return ERR_PTR(-ENOMEM);
261ee6373ddSDimitris Michailidis 
262ee6373ddSDimitris Michailidis 	res = fun_reserve_irqs(fp->fdev, 1, &irq->irq_idx);
263ee6373ddSDimitris Michailidis 	if (res != 1)
264ee6373ddSDimitris Michailidis 		goto free_irq;
265ee6373ddSDimitris Michailidis 
266ee6373ddSDimitris Michailidis 	res = xa_insert(&fp->irqs, idx + xa_idx_offset, irq, GFP_KERNEL);
267ee6373ddSDimitris Michailidis 	if (res)
268ee6373ddSDimitris Michailidis 		goto release_irq;
269ee6373ddSDimitris Michailidis 
270ee6373ddSDimitris Michailidis 	irq->irq = pci_irq_vector(fp->pdev, irq->irq_idx);
271ee6373ddSDimitris Michailidis 	cpumask_set_cpu(cpu, &irq->affinity_mask);
272ee6373ddSDimitris Michailidis 	irq->aff_notify.notify = fun_irq_aff_notify;
273ee6373ddSDimitris Michailidis 	irq->aff_notify.release = fun_irq_aff_release;
274ee6373ddSDimitris Michailidis 	irq->state = FUN_IRQ_INIT;
275ee6373ddSDimitris Michailidis 	return irq;
276ee6373ddSDimitris Michailidis 
277ee6373ddSDimitris Michailidis release_irq:
278ee6373ddSDimitris Michailidis 	fun_release_irqs(fp->fdev, 1, &irq->irq_idx);
279ee6373ddSDimitris Michailidis free_irq:
280ee6373ddSDimitris Michailidis 	kfree(irq);
281ee6373ddSDimitris Michailidis 	return ERR_PTR(res);
282ee6373ddSDimitris Michailidis }
283ee6373ddSDimitris Michailidis 
fun_free_qirq(struct funeth_priv * fp,struct fun_irq * irq)284ee6373ddSDimitris Michailidis static void fun_free_qirq(struct funeth_priv *fp, struct fun_irq *irq)
285ee6373ddSDimitris Michailidis {
286ee6373ddSDimitris Michailidis 	netif_napi_del(&irq->napi);
287ee6373ddSDimitris Michailidis 	fun_release_irqs(fp->fdev, 1, &irq->irq_idx);
288ee6373ddSDimitris Michailidis 	kfree(irq);
289ee6373ddSDimitris Michailidis }
290ee6373ddSDimitris Michailidis 
291ee6373ddSDimitris Michailidis /* Release the IRQs reserved for Tx/Rx queues that aren't being used. */
fun_prune_queue_irqs(struct net_device * dev)292ee6373ddSDimitris Michailidis static void fun_prune_queue_irqs(struct net_device *dev)
293ee6373ddSDimitris Michailidis {
294ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
295ee6373ddSDimitris Michailidis 	unsigned int nreleased = 0;
296ee6373ddSDimitris Michailidis 	struct fun_irq *irq;
297ee6373ddSDimitris Michailidis 	unsigned long idx;
298ee6373ddSDimitris Michailidis 
299ee6373ddSDimitris Michailidis 	xa_for_each(&fp->irqs, idx, irq) {
300ee6373ddSDimitris Michailidis 		if (irq->txq || irq->rxq)  /* skip those in use */
301ee6373ddSDimitris Michailidis 			continue;
302ee6373ddSDimitris Michailidis 
303ee6373ddSDimitris Michailidis 		xa_erase(&fp->irqs, idx);
304ee6373ddSDimitris Michailidis 		fun_free_qirq(fp, irq);
305ee6373ddSDimitris Michailidis 		nreleased++;
306ee6373ddSDimitris Michailidis 		if (idx < fp->rx_irq_ofst)
307ee6373ddSDimitris Michailidis 			fp->num_tx_irqs--;
308ee6373ddSDimitris Michailidis 		else
309ee6373ddSDimitris Michailidis 			fp->num_rx_irqs--;
310ee6373ddSDimitris Michailidis 	}
311ee6373ddSDimitris Michailidis 	netif_info(fp, intr, dev, "Released %u queue IRQs\n", nreleased);
312ee6373ddSDimitris Michailidis }
313ee6373ddSDimitris Michailidis 
314ee6373ddSDimitris Michailidis /* Reserve IRQs, one per queue, to acommodate the requested queue numbers @ntx
315ee6373ddSDimitris Michailidis  * and @nrx. IRQs are added incrementally to those we already have.
316ee6373ddSDimitris Michailidis  * We hold on to allocated IRQs until garbage collection of unused IRQs is
317ee6373ddSDimitris Michailidis  * separately requested.
318ee6373ddSDimitris Michailidis  */
fun_alloc_queue_irqs(struct net_device * dev,unsigned int ntx,unsigned int nrx)319ee6373ddSDimitris Michailidis static int fun_alloc_queue_irqs(struct net_device *dev, unsigned int ntx,
320ee6373ddSDimitris Michailidis 				unsigned int nrx)
321ee6373ddSDimitris Michailidis {
322ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
323ee6373ddSDimitris Michailidis 	int node = dev_to_node(&fp->pdev->dev);
324ee6373ddSDimitris Michailidis 	struct fun_irq *irq;
325ee6373ddSDimitris Michailidis 	unsigned int i;
326ee6373ddSDimitris Michailidis 
327ee6373ddSDimitris Michailidis 	for (i = fp->num_tx_irqs; i < ntx; i++) {
328ee6373ddSDimitris Michailidis 		irq = fun_alloc_qirq(fp, i, node, 0);
329ee6373ddSDimitris Michailidis 		if (IS_ERR(irq))
330ee6373ddSDimitris Michailidis 			return PTR_ERR(irq);
331ee6373ddSDimitris Michailidis 
332ee6373ddSDimitris Michailidis 		fp->num_tx_irqs++;
33316d083e2SJakub Kicinski 		netif_napi_add_tx(dev, &irq->napi, fun_txq_napi_poll);
334ee6373ddSDimitris Michailidis 	}
335ee6373ddSDimitris Michailidis 
336ee6373ddSDimitris Michailidis 	for (i = fp->num_rx_irqs; i < nrx; i++) {
337ee6373ddSDimitris Michailidis 		irq = fun_alloc_qirq(fp, i, node, fp->rx_irq_ofst);
338ee6373ddSDimitris Michailidis 		if (IS_ERR(irq))
339ee6373ddSDimitris Michailidis 			return PTR_ERR(irq);
340ee6373ddSDimitris Michailidis 
341ee6373ddSDimitris Michailidis 		fp->num_rx_irqs++;
342b48b89f9SJakub Kicinski 		netif_napi_add(dev, &irq->napi, fun_rxq_napi_poll);
343ee6373ddSDimitris Michailidis 	}
344ee6373ddSDimitris Michailidis 
345ee6373ddSDimitris Michailidis 	netif_info(fp, intr, dev, "Reserved %u/%u IRQs for Tx/Rx queues\n",
346ee6373ddSDimitris Michailidis 		   ntx, nrx);
347ee6373ddSDimitris Michailidis 	return 0;
348ee6373ddSDimitris Michailidis }
349ee6373ddSDimitris Michailidis 
free_txqs(struct funeth_txq ** txqs,unsigned int nqs,unsigned int start,int state)350ee6373ddSDimitris Michailidis static void free_txqs(struct funeth_txq **txqs, unsigned int nqs,
351ee6373ddSDimitris Michailidis 		      unsigned int start, int state)
352ee6373ddSDimitris Michailidis {
353ee6373ddSDimitris Michailidis 	unsigned int i;
354ee6373ddSDimitris Michailidis 
355ee6373ddSDimitris Michailidis 	for (i = start; i < nqs && txqs[i]; i++)
356ee6373ddSDimitris Michailidis 		txqs[i] = funeth_txq_free(txqs[i], state);
357ee6373ddSDimitris Michailidis }
358ee6373ddSDimitris Michailidis 
alloc_txqs(struct net_device * dev,struct funeth_txq ** txqs,unsigned int nqs,unsigned int depth,unsigned int start,int state)359ee6373ddSDimitris Michailidis static int alloc_txqs(struct net_device *dev, struct funeth_txq **txqs,
360ee6373ddSDimitris Michailidis 		      unsigned int nqs, unsigned int depth, unsigned int start,
361ee6373ddSDimitris Michailidis 		      int state)
362ee6373ddSDimitris Michailidis {
363ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
364ee6373ddSDimitris Michailidis 	unsigned int i;
365ee6373ddSDimitris Michailidis 	int err;
366ee6373ddSDimitris Michailidis 
367ee6373ddSDimitris Michailidis 	for (i = start; i < nqs; i++) {
368ee6373ddSDimitris Michailidis 		err = funeth_txq_create(dev, i, depth, xa_load(&fp->irqs, i),
369ee6373ddSDimitris Michailidis 					state, &txqs[i]);
370ee6373ddSDimitris Michailidis 		if (err) {
371ee6373ddSDimitris Michailidis 			free_txqs(txqs, nqs, start, FUN_QSTATE_DESTROYED);
372ee6373ddSDimitris Michailidis 			return err;
373ee6373ddSDimitris Michailidis 		}
374ee6373ddSDimitris Michailidis 	}
375ee6373ddSDimitris Michailidis 	return 0;
376ee6373ddSDimitris Michailidis }
377ee6373ddSDimitris Michailidis 
free_rxqs(struct funeth_rxq ** rxqs,unsigned int nqs,unsigned int start,int state)378ee6373ddSDimitris Michailidis static void free_rxqs(struct funeth_rxq **rxqs, unsigned int nqs,
379ee6373ddSDimitris Michailidis 		      unsigned int start, int state)
380ee6373ddSDimitris Michailidis {
381ee6373ddSDimitris Michailidis 	unsigned int i;
382ee6373ddSDimitris Michailidis 
383ee6373ddSDimitris Michailidis 	for (i = start; i < nqs && rxqs[i]; i++)
384ee6373ddSDimitris Michailidis 		rxqs[i] = funeth_rxq_free(rxqs[i], state);
385ee6373ddSDimitris Michailidis }
386ee6373ddSDimitris Michailidis 
alloc_rxqs(struct net_device * dev,struct funeth_rxq ** rxqs,unsigned int nqs,unsigned int ncqe,unsigned int nrqe,unsigned int start,int state)387ee6373ddSDimitris Michailidis static int alloc_rxqs(struct net_device *dev, struct funeth_rxq **rxqs,
388ee6373ddSDimitris Michailidis 		      unsigned int nqs, unsigned int ncqe, unsigned int nrqe,
389ee6373ddSDimitris Michailidis 		      unsigned int start, int state)
390ee6373ddSDimitris Michailidis {
391ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
392ee6373ddSDimitris Michailidis 	unsigned int i;
393ee6373ddSDimitris Michailidis 	int err;
394ee6373ddSDimitris Michailidis 
395ee6373ddSDimitris Michailidis 	for (i = start; i < nqs; i++) {
396ee6373ddSDimitris Michailidis 		err = funeth_rxq_create(dev, i, ncqe, nrqe,
397ee6373ddSDimitris Michailidis 					xa_load(&fp->irqs, i + fp->rx_irq_ofst),
398ee6373ddSDimitris Michailidis 					state, &rxqs[i]);
399ee6373ddSDimitris Michailidis 		if (err) {
400ee6373ddSDimitris Michailidis 			free_rxqs(rxqs, nqs, start, FUN_QSTATE_DESTROYED);
401ee6373ddSDimitris Michailidis 			return err;
402ee6373ddSDimitris Michailidis 		}
403ee6373ddSDimitris Michailidis 	}
404ee6373ddSDimitris Michailidis 	return 0;
405ee6373ddSDimitris Michailidis }
406ee6373ddSDimitris Michailidis 
free_xdpqs(struct funeth_txq ** xdpqs,unsigned int nqs,unsigned int start,int state)407ee6373ddSDimitris Michailidis static void free_xdpqs(struct funeth_txq **xdpqs, unsigned int nqs,
408ee6373ddSDimitris Michailidis 		       unsigned int start, int state)
409ee6373ddSDimitris Michailidis {
410ee6373ddSDimitris Michailidis 	unsigned int i;
411ee6373ddSDimitris Michailidis 
412ee6373ddSDimitris Michailidis 	for (i = start; i < nqs && xdpqs[i]; i++)
413ee6373ddSDimitris Michailidis 		xdpqs[i] = funeth_txq_free(xdpqs[i], state);
414ee6373ddSDimitris Michailidis 
415ee6373ddSDimitris Michailidis 	if (state == FUN_QSTATE_DESTROYED)
416ee6373ddSDimitris Michailidis 		kfree(xdpqs);
417ee6373ddSDimitris Michailidis }
418ee6373ddSDimitris Michailidis 
alloc_xdpqs(struct net_device * dev,unsigned int nqs,unsigned int depth,unsigned int start,int state)419ee6373ddSDimitris Michailidis static struct funeth_txq **alloc_xdpqs(struct net_device *dev, unsigned int nqs,
420ee6373ddSDimitris Michailidis 				       unsigned int depth, unsigned int start,
421ee6373ddSDimitris Michailidis 				       int state)
422ee6373ddSDimitris Michailidis {
423ee6373ddSDimitris Michailidis 	struct funeth_txq **xdpqs;
424ee6373ddSDimitris Michailidis 	unsigned int i;
425ee6373ddSDimitris Michailidis 	int err;
426ee6373ddSDimitris Michailidis 
427ee6373ddSDimitris Michailidis 	xdpqs = kcalloc(nqs, sizeof(*xdpqs), GFP_KERNEL);
428ee6373ddSDimitris Michailidis 	if (!xdpqs)
429ee6373ddSDimitris Michailidis 		return ERR_PTR(-ENOMEM);
430ee6373ddSDimitris Michailidis 
431ee6373ddSDimitris Michailidis 	for (i = start; i < nqs; i++) {
432ee6373ddSDimitris Michailidis 		err = funeth_txq_create(dev, i, depth, NULL, state, &xdpqs[i]);
433ee6373ddSDimitris Michailidis 		if (err) {
434ee6373ddSDimitris Michailidis 			free_xdpqs(xdpqs, nqs, start, FUN_QSTATE_DESTROYED);
435ee6373ddSDimitris Michailidis 			return ERR_PTR(err);
436ee6373ddSDimitris Michailidis 		}
437ee6373ddSDimitris Michailidis 	}
438ee6373ddSDimitris Michailidis 	return xdpqs;
439ee6373ddSDimitris Michailidis }
440ee6373ddSDimitris Michailidis 
fun_free_rings(struct net_device * netdev,struct fun_qset * qset)441ee6373ddSDimitris Michailidis static void fun_free_rings(struct net_device *netdev, struct fun_qset *qset)
442ee6373ddSDimitris Michailidis {
443ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
444ee6373ddSDimitris Michailidis 	struct funeth_txq **xdpqs = qset->xdpqs;
445ee6373ddSDimitris Michailidis 	struct funeth_rxq **rxqs = qset->rxqs;
446ee6373ddSDimitris Michailidis 
447ee6373ddSDimitris Michailidis 	/* qset may not specify any queues to operate on. In that case the
448ee6373ddSDimitris Michailidis 	 * currently installed queues are implied.
449ee6373ddSDimitris Michailidis 	 */
450ee6373ddSDimitris Michailidis 	if (!rxqs) {
451ee6373ddSDimitris Michailidis 		rxqs = rtnl_dereference(fp->rxqs);
452ee6373ddSDimitris Michailidis 		xdpqs = rtnl_dereference(fp->xdpqs);
453ee6373ddSDimitris Michailidis 		qset->txqs = fp->txqs;
454ee6373ddSDimitris Michailidis 		qset->nrxqs = netdev->real_num_rx_queues;
455ee6373ddSDimitris Michailidis 		qset->ntxqs = netdev->real_num_tx_queues;
456ee6373ddSDimitris Michailidis 		qset->nxdpqs = fp->num_xdpqs;
457ee6373ddSDimitris Michailidis 	}
458ee6373ddSDimitris Michailidis 	if (!rxqs)
459ee6373ddSDimitris Michailidis 		return;
460ee6373ddSDimitris Michailidis 
461ee6373ddSDimitris Michailidis 	if (rxqs == rtnl_dereference(fp->rxqs)) {
462ee6373ddSDimitris Michailidis 		rcu_assign_pointer(fp->rxqs, NULL);
463ee6373ddSDimitris Michailidis 		rcu_assign_pointer(fp->xdpqs, NULL);
464ee6373ddSDimitris Michailidis 		synchronize_net();
465ee6373ddSDimitris Michailidis 		fp->txqs = NULL;
466ee6373ddSDimitris Michailidis 	}
467ee6373ddSDimitris Michailidis 
468ee6373ddSDimitris Michailidis 	free_rxqs(rxqs, qset->nrxqs, qset->rxq_start, qset->state);
469ee6373ddSDimitris Michailidis 	free_txqs(qset->txqs, qset->ntxqs, qset->txq_start, qset->state);
470ee6373ddSDimitris Michailidis 	free_xdpqs(xdpqs, qset->nxdpqs, qset->xdpq_start, qset->state);
471ee6373ddSDimitris Michailidis 	if (qset->state == FUN_QSTATE_DESTROYED)
472ee6373ddSDimitris Michailidis 		kfree(rxqs);
473ee6373ddSDimitris Michailidis 
474ee6373ddSDimitris Michailidis 	/* Tell the caller which queues were operated on. */
475ee6373ddSDimitris Michailidis 	qset->rxqs = rxqs;
476ee6373ddSDimitris Michailidis 	qset->xdpqs = xdpqs;
477ee6373ddSDimitris Michailidis }
478ee6373ddSDimitris Michailidis 
fun_alloc_rings(struct net_device * netdev,struct fun_qset * qset)479ee6373ddSDimitris Michailidis static int fun_alloc_rings(struct net_device *netdev, struct fun_qset *qset)
480ee6373ddSDimitris Michailidis {
481ee6373ddSDimitris Michailidis 	struct funeth_txq **xdpqs = NULL, **txqs;
482ee6373ddSDimitris Michailidis 	struct funeth_rxq **rxqs;
483ee6373ddSDimitris Michailidis 	int err;
484ee6373ddSDimitris Michailidis 
485ee6373ddSDimitris Michailidis 	err = fun_alloc_queue_irqs(netdev, qset->ntxqs, qset->nrxqs);
486ee6373ddSDimitris Michailidis 	if (err)
487ee6373ddSDimitris Michailidis 		return err;
488ee6373ddSDimitris Michailidis 
489ee6373ddSDimitris Michailidis 	rxqs = kcalloc(qset->ntxqs + qset->nrxqs, sizeof(*rxqs), GFP_KERNEL);
490ee6373ddSDimitris Michailidis 	if (!rxqs)
491ee6373ddSDimitris Michailidis 		return -ENOMEM;
492ee6373ddSDimitris Michailidis 
493ee6373ddSDimitris Michailidis 	if (qset->nxdpqs) {
494ee6373ddSDimitris Michailidis 		xdpqs = alloc_xdpqs(netdev, qset->nxdpqs, qset->sq_depth,
495ee6373ddSDimitris Michailidis 				    qset->xdpq_start, qset->state);
496ee6373ddSDimitris Michailidis 		if (IS_ERR(xdpqs)) {
497ee6373ddSDimitris Michailidis 			err = PTR_ERR(xdpqs);
498ee6373ddSDimitris Michailidis 			goto free_qvec;
499ee6373ddSDimitris Michailidis 		}
500ee6373ddSDimitris Michailidis 	}
501ee6373ddSDimitris Michailidis 
502ee6373ddSDimitris Michailidis 	txqs = (struct funeth_txq **)&rxqs[qset->nrxqs];
503ee6373ddSDimitris Michailidis 	err = alloc_txqs(netdev, txqs, qset->ntxqs, qset->sq_depth,
504ee6373ddSDimitris Michailidis 			 qset->txq_start, qset->state);
505ee6373ddSDimitris Michailidis 	if (err)
506ee6373ddSDimitris Michailidis 		goto free_xdpqs;
507ee6373ddSDimitris Michailidis 
508ee6373ddSDimitris Michailidis 	err = alloc_rxqs(netdev, rxqs, qset->nrxqs, qset->cq_depth,
509ee6373ddSDimitris Michailidis 			 qset->rq_depth, qset->rxq_start, qset->state);
510ee6373ddSDimitris Michailidis 	if (err)
511ee6373ddSDimitris Michailidis 		goto free_txqs;
512ee6373ddSDimitris Michailidis 
513ee6373ddSDimitris Michailidis 	qset->rxqs = rxqs;
514ee6373ddSDimitris Michailidis 	qset->txqs = txqs;
515ee6373ddSDimitris Michailidis 	qset->xdpqs = xdpqs;
516ee6373ddSDimitris Michailidis 	return 0;
517ee6373ddSDimitris Michailidis 
518ee6373ddSDimitris Michailidis free_txqs:
519ee6373ddSDimitris Michailidis 	free_txqs(txqs, qset->ntxqs, qset->txq_start, FUN_QSTATE_DESTROYED);
520ee6373ddSDimitris Michailidis free_xdpqs:
521ee6373ddSDimitris Michailidis 	free_xdpqs(xdpqs, qset->nxdpqs, qset->xdpq_start, FUN_QSTATE_DESTROYED);
522ee6373ddSDimitris Michailidis free_qvec:
523ee6373ddSDimitris Michailidis 	kfree(rxqs);
524ee6373ddSDimitris Michailidis 	return err;
525ee6373ddSDimitris Michailidis }
526ee6373ddSDimitris Michailidis 
527ee6373ddSDimitris Michailidis /* Take queues to the next level. Presently this means creating them on the
528ee6373ddSDimitris Michailidis  * device.
529ee6373ddSDimitris Michailidis  */
fun_advance_ring_state(struct net_device * dev,struct fun_qset * qset)530ee6373ddSDimitris Michailidis static int fun_advance_ring_state(struct net_device *dev, struct fun_qset *qset)
531ee6373ddSDimitris Michailidis {
532ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
533ee6373ddSDimitris Michailidis 	int i, err;
534ee6373ddSDimitris Michailidis 
535ee6373ddSDimitris Michailidis 	for (i = 0; i < qset->nrxqs; i++) {
536ee6373ddSDimitris Michailidis 		err = fun_rxq_create_dev(qset->rxqs[i],
537ee6373ddSDimitris Michailidis 					 xa_load(&fp->irqs,
538ee6373ddSDimitris Michailidis 						 i + fp->rx_irq_ofst));
539ee6373ddSDimitris Michailidis 		if (err)
540ee6373ddSDimitris Michailidis 			goto out;
541ee6373ddSDimitris Michailidis 	}
542ee6373ddSDimitris Michailidis 
543ee6373ddSDimitris Michailidis 	for (i = 0; i < qset->ntxqs; i++) {
544ee6373ddSDimitris Michailidis 		err = fun_txq_create_dev(qset->txqs[i], xa_load(&fp->irqs, i));
545ee6373ddSDimitris Michailidis 		if (err)
546ee6373ddSDimitris Michailidis 			goto out;
547ee6373ddSDimitris Michailidis 	}
548ee6373ddSDimitris Michailidis 
549ee6373ddSDimitris Michailidis 	for (i = 0; i < qset->nxdpqs; i++) {
550ee6373ddSDimitris Michailidis 		err = fun_txq_create_dev(qset->xdpqs[i], NULL);
551ee6373ddSDimitris Michailidis 		if (err)
552ee6373ddSDimitris Michailidis 			goto out;
553ee6373ddSDimitris Michailidis 	}
554ee6373ddSDimitris Michailidis 
555ee6373ddSDimitris Michailidis 	return 0;
556ee6373ddSDimitris Michailidis 
557ee6373ddSDimitris Michailidis out:
558ee6373ddSDimitris Michailidis 	fun_free_rings(dev, qset);
559ee6373ddSDimitris Michailidis 	return err;
560ee6373ddSDimitris Michailidis }
561ee6373ddSDimitris Michailidis 
fun_port_create(struct net_device * netdev)562ee6373ddSDimitris Michailidis static int fun_port_create(struct net_device *netdev)
563ee6373ddSDimitris Michailidis {
564ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
565ee6373ddSDimitris Michailidis 	union {
566ee6373ddSDimitris Michailidis 		struct fun_admin_port_req req;
567ee6373ddSDimitris Michailidis 		struct fun_admin_port_rsp rsp;
568ee6373ddSDimitris Michailidis 	} cmd;
569ee6373ddSDimitris Michailidis 	int rc;
570ee6373ddSDimitris Michailidis 
571ee6373ddSDimitris Michailidis 	if (fp->lport != INVALID_LPORT)
572ee6373ddSDimitris Michailidis 		return 0;
573ee6373ddSDimitris Michailidis 
574ee6373ddSDimitris Michailidis 	cmd.req.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_PORT,
575ee6373ddSDimitris Michailidis 						    sizeof(cmd.req));
576ee6373ddSDimitris Michailidis 	cmd.req.u.create =
577ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_CREATE_REQ_INIT(FUN_ADMIN_SUBOP_CREATE, 0,
578ee6373ddSDimitris Michailidis 					       netdev->dev_port);
579ee6373ddSDimitris Michailidis 
580ee6373ddSDimitris Michailidis 	rc = fun_submit_admin_sync_cmd(fp->fdev, &cmd.req.common, &cmd.rsp,
581ee6373ddSDimitris Michailidis 				       sizeof(cmd.rsp), 0);
582ee6373ddSDimitris Michailidis 
583ee6373ddSDimitris Michailidis 	if (!rc)
584ee6373ddSDimitris Michailidis 		fp->lport = be16_to_cpu(cmd.rsp.u.create.lport);
585ee6373ddSDimitris Michailidis 	return rc;
586ee6373ddSDimitris Michailidis }
587ee6373ddSDimitris Michailidis 
fun_port_destroy(struct net_device * netdev)588ee6373ddSDimitris Michailidis static int fun_port_destroy(struct net_device *netdev)
589ee6373ddSDimitris Michailidis {
590ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
591ee6373ddSDimitris Michailidis 
592ee6373ddSDimitris Michailidis 	if (fp->lport == INVALID_LPORT)
593ee6373ddSDimitris Michailidis 		return 0;
594ee6373ddSDimitris Michailidis 
595ee6373ddSDimitris Michailidis 	fp->lport = INVALID_LPORT;
596ee6373ddSDimitris Michailidis 	return fun_res_destroy(fp->fdev, FUN_ADMIN_OP_PORT, 0,
597ee6373ddSDimitris Michailidis 			       netdev->dev_port);
598ee6373ddSDimitris Michailidis }
599ee6373ddSDimitris Michailidis 
fun_eth_create(struct funeth_priv * fp)600ee6373ddSDimitris Michailidis static int fun_eth_create(struct funeth_priv *fp)
601ee6373ddSDimitris Michailidis {
602ee6373ddSDimitris Michailidis 	union {
603ee6373ddSDimitris Michailidis 		struct fun_admin_eth_req req;
604ee6373ddSDimitris Michailidis 		struct fun_admin_generic_create_rsp rsp;
605ee6373ddSDimitris Michailidis 	} cmd;
606ee6373ddSDimitris Michailidis 	int rc;
607ee6373ddSDimitris Michailidis 
608ee6373ddSDimitris Michailidis 	cmd.req.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_ETH,
609ee6373ddSDimitris Michailidis 						    sizeof(cmd.req));
610ee6373ddSDimitris Michailidis 	cmd.req.u.create = FUN_ADMIN_ETH_CREATE_REQ_INIT(
611ee6373ddSDimitris Michailidis 				FUN_ADMIN_SUBOP_CREATE,
612ee6373ddSDimitris Michailidis 				FUN_ADMIN_RES_CREATE_FLAG_ALLOCATOR,
613ee6373ddSDimitris Michailidis 				0, fp->netdev->dev_port);
614ee6373ddSDimitris Michailidis 
615ee6373ddSDimitris Michailidis 	rc = fun_submit_admin_sync_cmd(fp->fdev, &cmd.req.common, &cmd.rsp,
616ee6373ddSDimitris Michailidis 				       sizeof(cmd.rsp), 0);
617ee6373ddSDimitris Michailidis 	return rc ? rc : be32_to_cpu(cmd.rsp.id);
618ee6373ddSDimitris Michailidis }
619ee6373ddSDimitris Michailidis 
fun_vi_create(struct funeth_priv * fp)620ee6373ddSDimitris Michailidis static int fun_vi_create(struct funeth_priv *fp)
621ee6373ddSDimitris Michailidis {
622ee6373ddSDimitris Michailidis 	struct fun_admin_vi_req req = {
623ee6373ddSDimitris Michailidis 		.common = FUN_ADMIN_REQ_COMMON_INIT2(FUN_ADMIN_OP_VI,
624ee6373ddSDimitris Michailidis 						     sizeof(req)),
625ee6373ddSDimitris Michailidis 		.u.create = FUN_ADMIN_VI_CREATE_REQ_INIT(FUN_ADMIN_SUBOP_CREATE,
626ee6373ddSDimitris Michailidis 							 0,
627ee6373ddSDimitris Michailidis 							 fp->netdev->dev_port,
628ee6373ddSDimitris Michailidis 							 fp->netdev->dev_port)
629ee6373ddSDimitris Michailidis 	};
630ee6373ddSDimitris Michailidis 
631ee6373ddSDimitris Michailidis 	return fun_submit_admin_sync_cmd(fp->fdev, &req.common, NULL, 0, 0);
632ee6373ddSDimitris Michailidis }
633ee6373ddSDimitris Michailidis 
634ee6373ddSDimitris Michailidis /* Helper to create an ETH flow and bind an SQ to it.
635ee6373ddSDimitris Michailidis  * Returns the ETH id (>= 0) on success or a negative error.
636ee6373ddSDimitris Michailidis  */
fun_create_and_bind_tx(struct funeth_priv * fp,u32 sqid)637ee6373ddSDimitris Michailidis int fun_create_and_bind_tx(struct funeth_priv *fp, u32 sqid)
638ee6373ddSDimitris Michailidis {
639ee6373ddSDimitris Michailidis 	int rc, ethid;
640ee6373ddSDimitris Michailidis 
641ee6373ddSDimitris Michailidis 	ethid = fun_eth_create(fp);
642ee6373ddSDimitris Michailidis 	if (ethid >= 0) {
643ee6373ddSDimitris Michailidis 		rc = fun_bind(fp->fdev, FUN_ADMIN_BIND_TYPE_EPSQ, sqid,
644ee6373ddSDimitris Michailidis 			      FUN_ADMIN_BIND_TYPE_ETH, ethid);
645ee6373ddSDimitris Michailidis 		if (rc) {
646ee6373ddSDimitris Michailidis 			fun_res_destroy(fp->fdev, FUN_ADMIN_OP_ETH, 0, ethid);
647ee6373ddSDimitris Michailidis 			ethid = rc;
648ee6373ddSDimitris Michailidis 		}
649ee6373ddSDimitris Michailidis 	}
650ee6373ddSDimitris Michailidis 	return ethid;
651ee6373ddSDimitris Michailidis }
652ee6373ddSDimitris Michailidis 
fun_queue_irq_handler(int irq,void * data)653ee6373ddSDimitris Michailidis static irqreturn_t fun_queue_irq_handler(int irq, void *data)
654ee6373ddSDimitris Michailidis {
655ee6373ddSDimitris Michailidis 	struct fun_irq *p = data;
656ee6373ddSDimitris Michailidis 
657ee6373ddSDimitris Michailidis 	if (p->rxq) {
658ee6373ddSDimitris Michailidis 		prefetch(p->rxq->next_cqe_info);
659ee6373ddSDimitris Michailidis 		p->rxq->irq_cnt++;
660ee6373ddSDimitris Michailidis 	}
661ee6373ddSDimitris Michailidis 	napi_schedule_irqoff(&p->napi);
662ee6373ddSDimitris Michailidis 	return IRQ_HANDLED;
663ee6373ddSDimitris Michailidis }
664ee6373ddSDimitris Michailidis 
fun_enable_irqs(struct net_device * dev)665ee6373ddSDimitris Michailidis static int fun_enable_irqs(struct net_device *dev)
666ee6373ddSDimitris Michailidis {
667ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
668ee6373ddSDimitris Michailidis 	unsigned long idx, last;
669ee6373ddSDimitris Michailidis 	unsigned int qidx;
670ee6373ddSDimitris Michailidis 	struct fun_irq *p;
671ee6373ddSDimitris Michailidis 	const char *qtype;
672ee6373ddSDimitris Michailidis 	int err;
673ee6373ddSDimitris Michailidis 
674ee6373ddSDimitris Michailidis 	xa_for_each(&fp->irqs, idx, p) {
675ee6373ddSDimitris Michailidis 		if (p->txq) {
676ee6373ddSDimitris Michailidis 			qtype = "tx";
677ee6373ddSDimitris Michailidis 			qidx = p->txq->qidx;
678ee6373ddSDimitris Michailidis 		} else if (p->rxq) {
679ee6373ddSDimitris Michailidis 			qtype = "rx";
680ee6373ddSDimitris Michailidis 			qidx = p->rxq->qidx;
681ee6373ddSDimitris Michailidis 		} else {
682ee6373ddSDimitris Michailidis 			continue;
683ee6373ddSDimitris Michailidis 		}
684ee6373ddSDimitris Michailidis 
685ee6373ddSDimitris Michailidis 		if (p->state != FUN_IRQ_INIT)
686ee6373ddSDimitris Michailidis 			continue;
687ee6373ddSDimitris Michailidis 
688ee6373ddSDimitris Michailidis 		snprintf(p->name, sizeof(p->name) - 1, "%s-%s-%u", dev->name,
689ee6373ddSDimitris Michailidis 			 qtype, qidx);
690ee6373ddSDimitris Michailidis 		err = request_irq(p->irq, fun_queue_irq_handler, 0, p->name, p);
691ee6373ddSDimitris Michailidis 		if (err) {
692ee6373ddSDimitris Michailidis 			netdev_err(dev, "Failed to allocate IRQ %u, err %d\n",
693ee6373ddSDimitris Michailidis 				   p->irq, err);
694ee6373ddSDimitris Michailidis 			goto unroll;
695ee6373ddSDimitris Michailidis 		}
696ee6373ddSDimitris Michailidis 		p->state = FUN_IRQ_REQUESTED;
697ee6373ddSDimitris Michailidis 	}
698ee6373ddSDimitris Michailidis 
699ee6373ddSDimitris Michailidis 	xa_for_each(&fp->irqs, idx, p) {
700ee6373ddSDimitris Michailidis 		if (p->state != FUN_IRQ_REQUESTED)
701ee6373ddSDimitris Michailidis 			continue;
702ee6373ddSDimitris Michailidis 		irq_set_affinity_notifier(p->irq, &p->aff_notify);
703ee6373ddSDimitris Michailidis 		irq_set_affinity_and_hint(p->irq, &p->affinity_mask);
704ee6373ddSDimitris Michailidis 		napi_enable(&p->napi);
705ee6373ddSDimitris Michailidis 		p->state = FUN_IRQ_ENABLED;
706ee6373ddSDimitris Michailidis 	}
707ee6373ddSDimitris Michailidis 
708ee6373ddSDimitris Michailidis 	return 0;
709ee6373ddSDimitris Michailidis 
710ee6373ddSDimitris Michailidis unroll:
711ee6373ddSDimitris Michailidis 	last = idx - 1;
712ee6373ddSDimitris Michailidis 	xa_for_each_range(&fp->irqs, idx, p, 0, last)
713ee6373ddSDimitris Michailidis 		if (p->state == FUN_IRQ_REQUESTED) {
714ee6373ddSDimitris Michailidis 			free_irq(p->irq, p);
715ee6373ddSDimitris Michailidis 			p->state = FUN_IRQ_INIT;
716ee6373ddSDimitris Michailidis 		}
717ee6373ddSDimitris Michailidis 
718ee6373ddSDimitris Michailidis 	return err;
719ee6373ddSDimitris Michailidis }
720ee6373ddSDimitris Michailidis 
fun_disable_one_irq(struct fun_irq * irq)721ee6373ddSDimitris Michailidis static void fun_disable_one_irq(struct fun_irq *irq)
722ee6373ddSDimitris Michailidis {
723ee6373ddSDimitris Michailidis 	napi_disable(&irq->napi);
724ee6373ddSDimitris Michailidis 	irq_set_affinity_notifier(irq->irq, NULL);
725ee6373ddSDimitris Michailidis 	irq_update_affinity_hint(irq->irq, NULL);
726ee6373ddSDimitris Michailidis 	free_irq(irq->irq, irq);
727ee6373ddSDimitris Michailidis 	irq->state = FUN_IRQ_INIT;
728ee6373ddSDimitris Michailidis }
729ee6373ddSDimitris Michailidis 
fun_disable_irqs(struct net_device * dev)730ee6373ddSDimitris Michailidis static void fun_disable_irqs(struct net_device *dev)
731ee6373ddSDimitris Michailidis {
732ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
733ee6373ddSDimitris Michailidis 	struct fun_irq *p;
734ee6373ddSDimitris Michailidis 	unsigned long idx;
735ee6373ddSDimitris Michailidis 
736ee6373ddSDimitris Michailidis 	xa_for_each(&fp->irqs, idx, p)
737ee6373ddSDimitris Michailidis 		if (p->state == FUN_IRQ_ENABLED)
738ee6373ddSDimitris Michailidis 			fun_disable_one_irq(p);
739ee6373ddSDimitris Michailidis }
740ee6373ddSDimitris Michailidis 
fun_down(struct net_device * dev,struct fun_qset * qset)741ee6373ddSDimitris Michailidis static void fun_down(struct net_device *dev, struct fun_qset *qset)
742ee6373ddSDimitris Michailidis {
743ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
744ee6373ddSDimitris Michailidis 
745ee6373ddSDimitris Michailidis 	/* If we don't have queues the data path is already down.
746ee6373ddSDimitris Michailidis 	 * Note netif_running(dev) may be true.
747ee6373ddSDimitris Michailidis 	 */
748ee6373ddSDimitris Michailidis 	if (!rcu_access_pointer(fp->rxqs))
749ee6373ddSDimitris Michailidis 		return;
750ee6373ddSDimitris Michailidis 
751ee6373ddSDimitris Michailidis 	/* It is also down if the queues aren't on the device. */
752ee6373ddSDimitris Michailidis 	if (fp->txqs[0]->init_state >= FUN_QSTATE_INIT_FULL) {
753ee6373ddSDimitris Michailidis 		netif_info(fp, ifdown, dev,
754ee6373ddSDimitris Michailidis 			   "Tearing down data path on device\n");
755ee6373ddSDimitris Michailidis 		fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_DISABLE, 0);
756ee6373ddSDimitris Michailidis 
757ee6373ddSDimitris Michailidis 		netif_carrier_off(dev);
758ee6373ddSDimitris Michailidis 		netif_tx_disable(dev);
759ee6373ddSDimitris Michailidis 
760ee6373ddSDimitris Michailidis 		fun_destroy_rss(fp);
761ee6373ddSDimitris Michailidis 		fun_res_destroy(fp->fdev, FUN_ADMIN_OP_VI, 0, dev->dev_port);
762ee6373ddSDimitris Michailidis 		fun_disable_irqs(dev);
763ee6373ddSDimitris Michailidis 	}
764ee6373ddSDimitris Michailidis 
765ee6373ddSDimitris Michailidis 	fun_free_rings(dev, qset);
766ee6373ddSDimitris Michailidis }
767ee6373ddSDimitris Michailidis 
fun_up(struct net_device * dev,struct fun_qset * qset)768ee6373ddSDimitris Michailidis static int fun_up(struct net_device *dev, struct fun_qset *qset)
769ee6373ddSDimitris Michailidis {
770ee6373ddSDimitris Michailidis 	static const int port_keys[] = {
771ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_KEY_STATS_DMA_LOW,
772ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_KEY_STATS_DMA_HIGH,
773ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_KEY_ENABLE
774ee6373ddSDimitris Michailidis 	};
775ee6373ddSDimitris Michailidis 
776ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
777ee6373ddSDimitris Michailidis 	u64 vals[] = {
778ee6373ddSDimitris Michailidis 		lower_32_bits(fp->stats_dma_addr),
779ee6373ddSDimitris Michailidis 		upper_32_bits(fp->stats_dma_addr),
780ee6373ddSDimitris Michailidis 		FUN_PORT_FLAG_ENABLE_NOTIFY
781ee6373ddSDimitris Michailidis 	};
782ee6373ddSDimitris Michailidis 	int err;
783ee6373ddSDimitris Michailidis 
784ee6373ddSDimitris Michailidis 	netif_info(fp, ifup, dev, "Setting up data path on device\n");
785ee6373ddSDimitris Michailidis 
786ee6373ddSDimitris Michailidis 	if (qset->rxqs[0]->init_state < FUN_QSTATE_INIT_FULL) {
787ee6373ddSDimitris Michailidis 		err = fun_advance_ring_state(dev, qset);
788ee6373ddSDimitris Michailidis 		if (err)
789ee6373ddSDimitris Michailidis 			return err;
790ee6373ddSDimitris Michailidis 	}
791ee6373ddSDimitris Michailidis 
792ee6373ddSDimitris Michailidis 	err = fun_vi_create(fp);
793ee6373ddSDimitris Michailidis 	if (err)
794ee6373ddSDimitris Michailidis 		goto free_queues;
795ee6373ddSDimitris Michailidis 
796ee6373ddSDimitris Michailidis 	fp->txqs = qset->txqs;
797ee6373ddSDimitris Michailidis 	rcu_assign_pointer(fp->rxqs, qset->rxqs);
798ee6373ddSDimitris Michailidis 	rcu_assign_pointer(fp->xdpqs, qset->xdpqs);
799ee6373ddSDimitris Michailidis 
800ee6373ddSDimitris Michailidis 	err = fun_enable_irqs(dev);
801ee6373ddSDimitris Michailidis 	if (err)
802ee6373ddSDimitris Michailidis 		goto destroy_vi;
803ee6373ddSDimitris Michailidis 
804ee6373ddSDimitris Michailidis 	if (fp->rss_cfg) {
805ee6373ddSDimitris Michailidis 		err = fun_config_rss(dev, fp->hash_algo, fp->rss_key,
806ee6373ddSDimitris Michailidis 				     fp->indir_table, FUN_ADMIN_SUBOP_CREATE);
807ee6373ddSDimitris Michailidis 	} else {
808ee6373ddSDimitris Michailidis 		/* The non-RSS case has only 1 queue. */
809ee6373ddSDimitris Michailidis 		err = fun_bind(fp->fdev, FUN_ADMIN_BIND_TYPE_VI, dev->dev_port,
810ee6373ddSDimitris Michailidis 			       FUN_ADMIN_BIND_TYPE_EPCQ,
811ee6373ddSDimitris Michailidis 			       qset->rxqs[0]->hw_cqid);
812ee6373ddSDimitris Michailidis 	}
813ee6373ddSDimitris Michailidis 	if (err)
814ee6373ddSDimitris Michailidis 		goto disable_irqs;
815ee6373ddSDimitris Michailidis 
816ee6373ddSDimitris Michailidis 	err = fun_port_write_cmds(fp, 3, port_keys, vals);
817ee6373ddSDimitris Michailidis 	if (err)
818ee6373ddSDimitris Michailidis 		goto free_rss;
819ee6373ddSDimitris Michailidis 
820ee6373ddSDimitris Michailidis 	netif_tx_start_all_queues(dev);
821ee6373ddSDimitris Michailidis 	return 0;
822ee6373ddSDimitris Michailidis 
823ee6373ddSDimitris Michailidis free_rss:
824ee6373ddSDimitris Michailidis 	fun_destroy_rss(fp);
825ee6373ddSDimitris Michailidis disable_irqs:
826ee6373ddSDimitris Michailidis 	fun_disable_irqs(dev);
827ee6373ddSDimitris Michailidis destroy_vi:
828ee6373ddSDimitris Michailidis 	fun_res_destroy(fp->fdev, FUN_ADMIN_OP_VI, 0, dev->dev_port);
829ee6373ddSDimitris Michailidis free_queues:
830ee6373ddSDimitris Michailidis 	fun_free_rings(dev, qset);
831ee6373ddSDimitris Michailidis 	return err;
832ee6373ddSDimitris Michailidis }
833ee6373ddSDimitris Michailidis 
funeth_open(struct net_device * netdev)834ee6373ddSDimitris Michailidis static int funeth_open(struct net_device *netdev)
835ee6373ddSDimitris Michailidis {
836ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
837ee6373ddSDimitris Michailidis 	struct fun_qset qset = {
838ee6373ddSDimitris Michailidis 		.nrxqs = netdev->real_num_rx_queues,
839ee6373ddSDimitris Michailidis 		.ntxqs = netdev->real_num_tx_queues,
840ee6373ddSDimitris Michailidis 		.nxdpqs = fp->num_xdpqs,
841ee6373ddSDimitris Michailidis 		.cq_depth = fp->cq_depth,
842ee6373ddSDimitris Michailidis 		.rq_depth = fp->rq_depth,
843ee6373ddSDimitris Michailidis 		.sq_depth = fp->sq_depth,
844ee6373ddSDimitris Michailidis 		.state = FUN_QSTATE_INIT_FULL,
845ee6373ddSDimitris Michailidis 	};
846ee6373ddSDimitris Michailidis 	int rc;
847ee6373ddSDimitris Michailidis 
848ee6373ddSDimitris Michailidis 	rc = fun_alloc_rings(netdev, &qset);
849ee6373ddSDimitris Michailidis 	if (rc)
850ee6373ddSDimitris Michailidis 		return rc;
851ee6373ddSDimitris Michailidis 
852ee6373ddSDimitris Michailidis 	rc = fun_up(netdev, &qset);
853ee6373ddSDimitris Michailidis 	if (rc) {
854ee6373ddSDimitris Michailidis 		qset.state = FUN_QSTATE_DESTROYED;
855ee6373ddSDimitris Michailidis 		fun_free_rings(netdev, &qset);
856ee6373ddSDimitris Michailidis 	}
857ee6373ddSDimitris Michailidis 
858ee6373ddSDimitris Michailidis 	return rc;
859ee6373ddSDimitris Michailidis }
860ee6373ddSDimitris Michailidis 
funeth_close(struct net_device * netdev)861ee6373ddSDimitris Michailidis static int funeth_close(struct net_device *netdev)
862ee6373ddSDimitris Michailidis {
863ee6373ddSDimitris Michailidis 	struct fun_qset qset = { .state = FUN_QSTATE_DESTROYED };
864ee6373ddSDimitris Michailidis 
865ee6373ddSDimitris Michailidis 	fun_down(netdev, &qset);
866ee6373ddSDimitris Michailidis 	return 0;
867ee6373ddSDimitris Michailidis }
868ee6373ddSDimitris Michailidis 
fun_get_stats64(struct net_device * netdev,struct rtnl_link_stats64 * stats)869ee6373ddSDimitris Michailidis static void fun_get_stats64(struct net_device *netdev,
870ee6373ddSDimitris Michailidis 			    struct rtnl_link_stats64 *stats)
871ee6373ddSDimitris Michailidis {
872ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
873ee6373ddSDimitris Michailidis 	struct funeth_txq **xdpqs;
874ee6373ddSDimitris Michailidis 	struct funeth_rxq **rxqs;
875ee6373ddSDimitris Michailidis 	unsigned int i, start;
876ee6373ddSDimitris Michailidis 
877ee6373ddSDimitris Michailidis 	stats->tx_packets = fp->tx_packets;
878ee6373ddSDimitris Michailidis 	stats->tx_bytes   = fp->tx_bytes;
879ee6373ddSDimitris Michailidis 	stats->tx_dropped = fp->tx_dropped;
880ee6373ddSDimitris Michailidis 
881ee6373ddSDimitris Michailidis 	stats->rx_packets = fp->rx_packets;
882ee6373ddSDimitris Michailidis 	stats->rx_bytes   = fp->rx_bytes;
883ee6373ddSDimitris Michailidis 	stats->rx_dropped = fp->rx_dropped;
884ee6373ddSDimitris Michailidis 
885ee6373ddSDimitris Michailidis 	rcu_read_lock();
886ee6373ddSDimitris Michailidis 	rxqs = rcu_dereference(fp->rxqs);
887ee6373ddSDimitris Michailidis 	if (!rxqs)
888ee6373ddSDimitris Michailidis 		goto unlock;
889ee6373ddSDimitris Michailidis 
890ee6373ddSDimitris Michailidis 	for (i = 0; i < netdev->real_num_tx_queues; i++) {
891ee6373ddSDimitris Michailidis 		struct funeth_txq_stats txs;
892ee6373ddSDimitris Michailidis 
893ee6373ddSDimitris Michailidis 		FUN_QSTAT_READ(fp->txqs[i], start, txs);
894ee6373ddSDimitris Michailidis 		stats->tx_packets += txs.tx_pkts;
895ee6373ddSDimitris Michailidis 		stats->tx_bytes   += txs.tx_bytes;
896ee6373ddSDimitris Michailidis 		stats->tx_dropped += txs.tx_map_err;
897ee6373ddSDimitris Michailidis 	}
898ee6373ddSDimitris Michailidis 
899ee6373ddSDimitris Michailidis 	for (i = 0; i < netdev->real_num_rx_queues; i++) {
900ee6373ddSDimitris Michailidis 		struct funeth_rxq_stats rxs;
901ee6373ddSDimitris Michailidis 
902ee6373ddSDimitris Michailidis 		FUN_QSTAT_READ(rxqs[i], start, rxs);
903ee6373ddSDimitris Michailidis 		stats->rx_packets += rxs.rx_pkts;
904ee6373ddSDimitris Michailidis 		stats->rx_bytes   += rxs.rx_bytes;
905ee6373ddSDimitris Michailidis 		stats->rx_dropped += rxs.rx_map_err + rxs.rx_mem_drops;
906ee6373ddSDimitris Michailidis 	}
907ee6373ddSDimitris Michailidis 
908ee6373ddSDimitris Michailidis 	xdpqs = rcu_dereference(fp->xdpqs);
909ee6373ddSDimitris Michailidis 	if (!xdpqs)
910ee6373ddSDimitris Michailidis 		goto unlock;
911ee6373ddSDimitris Michailidis 
912ee6373ddSDimitris Michailidis 	for (i = 0; i < fp->num_xdpqs; i++) {
913ee6373ddSDimitris Michailidis 		struct funeth_txq_stats txs;
914ee6373ddSDimitris Michailidis 
915ee6373ddSDimitris Michailidis 		FUN_QSTAT_READ(xdpqs[i], start, txs);
916ee6373ddSDimitris Michailidis 		stats->tx_packets += txs.tx_pkts;
917ee6373ddSDimitris Michailidis 		stats->tx_bytes   += txs.tx_bytes;
918ee6373ddSDimitris Michailidis 	}
919ee6373ddSDimitris Michailidis unlock:
920ee6373ddSDimitris Michailidis 	rcu_read_unlock();
921ee6373ddSDimitris Michailidis }
922ee6373ddSDimitris Michailidis 
fun_change_mtu(struct net_device * netdev,int new_mtu)923ee6373ddSDimitris Michailidis static int fun_change_mtu(struct net_device *netdev, int new_mtu)
924ee6373ddSDimitris Michailidis {
925ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
926ee6373ddSDimitris Michailidis 	int rc;
927ee6373ddSDimitris Michailidis 
928ee6373ddSDimitris Michailidis 	rc = fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_MTU, new_mtu);
929ee6373ddSDimitris Michailidis 	if (!rc)
930*1eb2cdedSEric Dumazet 		WRITE_ONCE(netdev->mtu, new_mtu);
931ee6373ddSDimitris Michailidis 	return rc;
932ee6373ddSDimitris Michailidis }
933ee6373ddSDimitris Michailidis 
fun_set_macaddr(struct net_device * netdev,void * addr)934ee6373ddSDimitris Michailidis static int fun_set_macaddr(struct net_device *netdev, void *addr)
935ee6373ddSDimitris Michailidis {
936ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
937ee6373ddSDimitris Michailidis 	struct sockaddr *saddr = addr;
938ee6373ddSDimitris Michailidis 	int rc;
939ee6373ddSDimitris Michailidis 
940ee6373ddSDimitris Michailidis 	if (!is_valid_ether_addr(saddr->sa_data))
941ee6373ddSDimitris Michailidis 		return -EADDRNOTAVAIL;
942ee6373ddSDimitris Michailidis 
943ee6373ddSDimitris Michailidis 	if (ether_addr_equal(netdev->dev_addr, saddr->sa_data))
944ee6373ddSDimitris Michailidis 		return 0;
945ee6373ddSDimitris Michailidis 
946ee6373ddSDimitris Michailidis 	rc = fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_MACADDR,
947ee6373ddSDimitris Michailidis 				ether_addr_to_u64(saddr->sa_data));
948ee6373ddSDimitris Michailidis 	if (!rc)
949ee6373ddSDimitris Michailidis 		eth_hw_addr_set(netdev, saddr->sa_data);
950ee6373ddSDimitris Michailidis 	return rc;
951ee6373ddSDimitris Michailidis }
952ee6373ddSDimitris Michailidis 
fun_get_port_attributes(struct net_device * netdev)953ee6373ddSDimitris Michailidis static int fun_get_port_attributes(struct net_device *netdev)
954ee6373ddSDimitris Michailidis {
955ee6373ddSDimitris Michailidis 	static const int keys[] = {
956ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_KEY_MACADDR, FUN_ADMIN_PORT_KEY_CAPABILITIES,
957ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_KEY_ADVERT, FUN_ADMIN_PORT_KEY_MTU
958ee6373ddSDimitris Michailidis 	};
959ee6373ddSDimitris Michailidis 	static const int phys_keys[] = {
960ee6373ddSDimitris Michailidis 		FUN_ADMIN_PORT_KEY_LANE_ATTRS,
961ee6373ddSDimitris Michailidis 	};
962ee6373ddSDimitris Michailidis 
963ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
964ee6373ddSDimitris Michailidis 	u64 data[ARRAY_SIZE(keys)];
965ee6373ddSDimitris Michailidis 	u8 mac[ETH_ALEN];
966ee6373ddSDimitris Michailidis 	int i, rc;
967ee6373ddSDimitris Michailidis 
968ee6373ddSDimitris Michailidis 	rc = fun_port_read_cmds(fp, ARRAY_SIZE(keys), keys, data);
969ee6373ddSDimitris Michailidis 	if (rc)
970ee6373ddSDimitris Michailidis 		return rc;
971ee6373ddSDimitris Michailidis 
972ee6373ddSDimitris Michailidis 	for (i = 0; i < ARRAY_SIZE(keys); i++) {
973ee6373ddSDimitris Michailidis 		switch (keys[i]) {
974ee6373ddSDimitris Michailidis 		case FUN_ADMIN_PORT_KEY_MACADDR:
975ee6373ddSDimitris Michailidis 			u64_to_ether_addr(data[i], mac);
976ee6373ddSDimitris Michailidis 			if (is_zero_ether_addr(mac)) {
977ee6373ddSDimitris Michailidis 				eth_hw_addr_random(netdev);
978ee6373ddSDimitris Michailidis 			} else if (is_valid_ether_addr(mac)) {
979ee6373ddSDimitris Michailidis 				eth_hw_addr_set(netdev, mac);
980ee6373ddSDimitris Michailidis 			} else {
981ee6373ddSDimitris Michailidis 				netdev_err(netdev,
982ee6373ddSDimitris Michailidis 					   "device provided a bad MAC address %pM\n",
983ee6373ddSDimitris Michailidis 					   mac);
984ee6373ddSDimitris Michailidis 				return -EINVAL;
985ee6373ddSDimitris Michailidis 			}
986ee6373ddSDimitris Michailidis 			break;
987ee6373ddSDimitris Michailidis 
988ee6373ddSDimitris Michailidis 		case FUN_ADMIN_PORT_KEY_CAPABILITIES:
989ee6373ddSDimitris Michailidis 			fp->port_caps = data[i];
990ee6373ddSDimitris Michailidis 			break;
991ee6373ddSDimitris Michailidis 
992ee6373ddSDimitris Michailidis 		case FUN_ADMIN_PORT_KEY_ADVERT:
993ee6373ddSDimitris Michailidis 			fp->advertising = data[i];
994ee6373ddSDimitris Michailidis 			break;
995ee6373ddSDimitris Michailidis 
996ee6373ddSDimitris Michailidis 		case FUN_ADMIN_PORT_KEY_MTU:
997ee6373ddSDimitris Michailidis 			netdev->mtu = data[i];
998ee6373ddSDimitris Michailidis 			break;
999ee6373ddSDimitris Michailidis 		}
1000ee6373ddSDimitris Michailidis 	}
1001ee6373ddSDimitris Michailidis 
1002ee6373ddSDimitris Michailidis 	if (!(fp->port_caps & FUN_PORT_CAP_VPORT)) {
1003ee6373ddSDimitris Michailidis 		rc = fun_port_read_cmds(fp, ARRAY_SIZE(phys_keys), phys_keys,
1004ee6373ddSDimitris Michailidis 					data);
1005ee6373ddSDimitris Michailidis 		if (rc)
1006ee6373ddSDimitris Michailidis 			return rc;
1007ee6373ddSDimitris Michailidis 
1008ee6373ddSDimitris Michailidis 		fp->lane_attrs = data[0];
1009ee6373ddSDimitris Michailidis 	}
1010ee6373ddSDimitris Michailidis 
1011ee6373ddSDimitris Michailidis 	if (netdev->addr_assign_type == NET_ADDR_RANDOM)
1012ee6373ddSDimitris Michailidis 		return fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_MACADDR,
1013ee6373ddSDimitris Michailidis 					  ether_addr_to_u64(netdev->dev_addr));
1014ee6373ddSDimitris Michailidis 	return 0;
1015ee6373ddSDimitris Michailidis }
1016ee6373ddSDimitris Michailidis 
fun_hwtstamp_get(struct net_device * dev,struct ifreq * ifr)1017ee6373ddSDimitris Michailidis static int fun_hwtstamp_get(struct net_device *dev, struct ifreq *ifr)
1018ee6373ddSDimitris Michailidis {
1019ee6373ddSDimitris Michailidis 	const struct funeth_priv *fp = netdev_priv(dev);
1020ee6373ddSDimitris Michailidis 
1021ee6373ddSDimitris Michailidis 	return copy_to_user(ifr->ifr_data, &fp->hwtstamp_cfg,
1022ee6373ddSDimitris Michailidis 			    sizeof(fp->hwtstamp_cfg)) ? -EFAULT : 0;
1023ee6373ddSDimitris Michailidis }
1024ee6373ddSDimitris Michailidis 
fun_hwtstamp_set(struct net_device * dev,struct ifreq * ifr)1025ee6373ddSDimitris Michailidis static int fun_hwtstamp_set(struct net_device *dev, struct ifreq *ifr)
1026ee6373ddSDimitris Michailidis {
1027ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1028ee6373ddSDimitris Michailidis 	struct hwtstamp_config cfg;
1029ee6373ddSDimitris Michailidis 
1030ee6373ddSDimitris Michailidis 	if (copy_from_user(&cfg, ifr->ifr_data, sizeof(cfg)))
1031ee6373ddSDimitris Michailidis 		return -EFAULT;
1032ee6373ddSDimitris Michailidis 
1033ee6373ddSDimitris Michailidis 	/* no TX HW timestamps */
1034ee6373ddSDimitris Michailidis 	cfg.tx_type = HWTSTAMP_TX_OFF;
1035ee6373ddSDimitris Michailidis 
1036ee6373ddSDimitris Michailidis 	switch (cfg.rx_filter) {
1037ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_NONE:
1038ee6373ddSDimitris Michailidis 		break;
1039ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_ALL:
1040ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_SOME:
1041ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V1_L4_EVENT:
1042ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V1_L4_SYNC:
1043ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V1_L4_DELAY_REQ:
1044ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_L4_EVENT:
1045ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_L4_SYNC:
1046ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_L4_DELAY_REQ:
1047ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_L2_EVENT:
1048ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_L2_SYNC:
1049ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_L2_DELAY_REQ:
1050ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_EVENT:
1051ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_SYNC:
1052ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_PTP_V2_DELAY_REQ:
1053ee6373ddSDimitris Michailidis 	case HWTSTAMP_FILTER_NTP_ALL:
1054ee6373ddSDimitris Michailidis 		cfg.rx_filter = HWTSTAMP_FILTER_ALL;
1055ee6373ddSDimitris Michailidis 		break;
1056ee6373ddSDimitris Michailidis 	default:
1057ee6373ddSDimitris Michailidis 		return -ERANGE;
1058ee6373ddSDimitris Michailidis 	}
1059ee6373ddSDimitris Michailidis 
1060ee6373ddSDimitris Michailidis 	fp->hwtstamp_cfg = cfg;
1061ee6373ddSDimitris Michailidis 	return copy_to_user(ifr->ifr_data, &cfg, sizeof(cfg)) ? -EFAULT : 0;
1062ee6373ddSDimitris Michailidis }
1063ee6373ddSDimitris Michailidis 
fun_ioctl(struct net_device * dev,struct ifreq * ifr,int cmd)1064ee6373ddSDimitris Michailidis static int fun_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
1065ee6373ddSDimitris Michailidis {
1066ee6373ddSDimitris Michailidis 	switch (cmd) {
1067ee6373ddSDimitris Michailidis 	case SIOCSHWTSTAMP:
1068ee6373ddSDimitris Michailidis 		return fun_hwtstamp_set(dev, ifr);
1069ee6373ddSDimitris Michailidis 	case SIOCGHWTSTAMP:
1070ee6373ddSDimitris Michailidis 		return fun_hwtstamp_get(dev, ifr);
1071ee6373ddSDimitris Michailidis 	default:
1072ee6373ddSDimitris Michailidis 		return -EOPNOTSUPP;
1073ee6373ddSDimitris Michailidis 	}
1074ee6373ddSDimitris Michailidis }
1075ee6373ddSDimitris Michailidis 
1076ee6373ddSDimitris Michailidis /* Prepare the queues for XDP. */
fun_enter_xdp(struct net_device * dev,struct bpf_prog * prog)1077ee6373ddSDimitris Michailidis static int fun_enter_xdp(struct net_device *dev, struct bpf_prog *prog)
1078ee6373ddSDimitris Michailidis {
1079ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1080ee6373ddSDimitris Michailidis 	unsigned int i, nqs = num_online_cpus();
1081ee6373ddSDimitris Michailidis 	struct funeth_txq **xdpqs;
1082ee6373ddSDimitris Michailidis 	struct funeth_rxq **rxqs;
1083ee6373ddSDimitris Michailidis 	int err;
1084ee6373ddSDimitris Michailidis 
1085ee6373ddSDimitris Michailidis 	xdpqs = alloc_xdpqs(dev, nqs, fp->sq_depth, 0, FUN_QSTATE_INIT_FULL);
1086ee6373ddSDimitris Michailidis 	if (IS_ERR(xdpqs))
1087ee6373ddSDimitris Michailidis 		return PTR_ERR(xdpqs);
1088ee6373ddSDimitris Michailidis 
1089ee6373ddSDimitris Michailidis 	rxqs = rtnl_dereference(fp->rxqs);
1090ee6373ddSDimitris Michailidis 	for (i = 0; i < dev->real_num_rx_queues; i++) {
1091ee6373ddSDimitris Michailidis 		err = fun_rxq_set_bpf(rxqs[i], prog);
1092ee6373ddSDimitris Michailidis 		if (err)
1093ee6373ddSDimitris Michailidis 			goto out;
1094ee6373ddSDimitris Michailidis 	}
1095ee6373ddSDimitris Michailidis 
1096ee6373ddSDimitris Michailidis 	fp->num_xdpqs = nqs;
1097ee6373ddSDimitris Michailidis 	rcu_assign_pointer(fp->xdpqs, xdpqs);
1098ee6373ddSDimitris Michailidis 	return 0;
1099ee6373ddSDimitris Michailidis out:
1100ee6373ddSDimitris Michailidis 	while (i--)
1101ee6373ddSDimitris Michailidis 		fun_rxq_set_bpf(rxqs[i], NULL);
1102ee6373ddSDimitris Michailidis 
1103ee6373ddSDimitris Michailidis 	free_xdpqs(xdpqs, nqs, 0, FUN_QSTATE_DESTROYED);
1104ee6373ddSDimitris Michailidis 	return err;
1105ee6373ddSDimitris Michailidis }
1106ee6373ddSDimitris Michailidis 
1107ee6373ddSDimitris Michailidis /* Set the queues for non-XDP operation. */
fun_end_xdp(struct net_device * dev)1108ee6373ddSDimitris Michailidis static void fun_end_xdp(struct net_device *dev)
1109ee6373ddSDimitris Michailidis {
1110ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1111ee6373ddSDimitris Michailidis 	struct funeth_txq **xdpqs;
1112ee6373ddSDimitris Michailidis 	struct funeth_rxq **rxqs;
1113ee6373ddSDimitris Michailidis 	unsigned int i;
1114ee6373ddSDimitris Michailidis 
1115ee6373ddSDimitris Michailidis 	xdpqs = rtnl_dereference(fp->xdpqs);
1116ee6373ddSDimitris Michailidis 	rcu_assign_pointer(fp->xdpqs, NULL);
1117ee6373ddSDimitris Michailidis 	synchronize_net();
1118ee6373ddSDimitris Michailidis 	/* at this point both Rx and Tx XDP processing has ended */
1119ee6373ddSDimitris Michailidis 
1120ee6373ddSDimitris Michailidis 	free_xdpqs(xdpqs, fp->num_xdpqs, 0, FUN_QSTATE_DESTROYED);
1121ee6373ddSDimitris Michailidis 	fp->num_xdpqs = 0;
1122ee6373ddSDimitris Michailidis 
1123ee6373ddSDimitris Michailidis 	rxqs = rtnl_dereference(fp->rxqs);
1124ee6373ddSDimitris Michailidis 	for (i = 0; i < dev->real_num_rx_queues; i++)
1125ee6373ddSDimitris Michailidis 		fun_rxq_set_bpf(rxqs[i], NULL);
1126ee6373ddSDimitris Michailidis }
1127ee6373ddSDimitris Michailidis 
1128ee6373ddSDimitris Michailidis #define XDP_MAX_MTU \
1129ee6373ddSDimitris Michailidis 	(PAGE_SIZE - FUN_XDP_HEADROOM - VLAN_ETH_HLEN - FUN_RX_TAILROOM)
1130ee6373ddSDimitris Michailidis 
fun_xdp_setup(struct net_device * dev,struct netdev_bpf * xdp)1131ee6373ddSDimitris Michailidis static int fun_xdp_setup(struct net_device *dev, struct netdev_bpf *xdp)
1132ee6373ddSDimitris Michailidis {
1133ee6373ddSDimitris Michailidis 	struct bpf_prog *old_prog, *prog = xdp->prog;
1134ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1135ee6373ddSDimitris Michailidis 	int i, err;
1136ee6373ddSDimitris Michailidis 
1137ee6373ddSDimitris Michailidis 	/* XDP uses at most one buffer */
1138ee6373ddSDimitris Michailidis 	if (prog && dev->mtu > XDP_MAX_MTU) {
1139ee6373ddSDimitris Michailidis 		netdev_err(dev, "device MTU %u too large for XDP\n", dev->mtu);
1140ee6373ddSDimitris Michailidis 		NL_SET_ERR_MSG_MOD(xdp->extack,
1141ee6373ddSDimitris Michailidis 				   "Device MTU too large for XDP");
1142ee6373ddSDimitris Michailidis 		return -EINVAL;
1143ee6373ddSDimitris Michailidis 	}
1144ee6373ddSDimitris Michailidis 
1145ee6373ddSDimitris Michailidis 	if (!netif_running(dev)) {
1146ee6373ddSDimitris Michailidis 		fp->num_xdpqs = prog ? num_online_cpus() : 0;
1147ee6373ddSDimitris Michailidis 	} else if (prog && !fp->xdp_prog) {
1148ee6373ddSDimitris Michailidis 		err = fun_enter_xdp(dev, prog);
1149ee6373ddSDimitris Michailidis 		if (err) {
1150ee6373ddSDimitris Michailidis 			NL_SET_ERR_MSG_MOD(xdp->extack,
1151ee6373ddSDimitris Michailidis 					   "Failed to set queues for XDP.");
1152ee6373ddSDimitris Michailidis 			return err;
1153ee6373ddSDimitris Michailidis 		}
1154ee6373ddSDimitris Michailidis 	} else if (!prog && fp->xdp_prog) {
1155ee6373ddSDimitris Michailidis 		fun_end_xdp(dev);
1156ee6373ddSDimitris Michailidis 	} else {
1157ee6373ddSDimitris Michailidis 		struct funeth_rxq **rxqs = rtnl_dereference(fp->rxqs);
1158ee6373ddSDimitris Michailidis 
1159ee6373ddSDimitris Michailidis 		for (i = 0; i < dev->real_num_rx_queues; i++)
1160ee6373ddSDimitris Michailidis 			WRITE_ONCE(rxqs[i]->xdp_prog, prog);
1161ee6373ddSDimitris Michailidis 	}
1162ee6373ddSDimitris Michailidis 
116366c0e13aSMarek Majtyka 	if (prog)
116466c0e13aSMarek Majtyka 		xdp_features_set_redirect_target(dev, true);
116566c0e13aSMarek Majtyka 	else
116666c0e13aSMarek Majtyka 		xdp_features_clear_redirect_target(dev);
116766c0e13aSMarek Majtyka 
1168ee6373ddSDimitris Michailidis 	dev->max_mtu = prog ? XDP_MAX_MTU : FUN_MAX_MTU;
1169ee6373ddSDimitris Michailidis 	old_prog = xchg(&fp->xdp_prog, prog);
1170ee6373ddSDimitris Michailidis 	if (old_prog)
1171ee6373ddSDimitris Michailidis 		bpf_prog_put(old_prog);
1172ee6373ddSDimitris Michailidis 
1173ee6373ddSDimitris Michailidis 	return 0;
1174ee6373ddSDimitris Michailidis }
1175ee6373ddSDimitris Michailidis 
fun_xdp(struct net_device * dev,struct netdev_bpf * xdp)1176ee6373ddSDimitris Michailidis static int fun_xdp(struct net_device *dev, struct netdev_bpf *xdp)
1177ee6373ddSDimitris Michailidis {
1178ee6373ddSDimitris Michailidis 	switch (xdp->command) {
1179ee6373ddSDimitris Michailidis 	case XDP_SETUP_PROG:
1180ee6373ddSDimitris Michailidis 		return fun_xdp_setup(dev, xdp);
1181ee6373ddSDimitris Michailidis 	default:
1182ee6373ddSDimitris Michailidis 		return -EINVAL;
1183ee6373ddSDimitris Michailidis 	}
1184ee6373ddSDimitris Michailidis }
1185ee6373ddSDimitris Michailidis 
fun_init_vports(struct fun_ethdev * ed,unsigned int n)1186ee6373ddSDimitris Michailidis static int fun_init_vports(struct fun_ethdev *ed, unsigned int n)
1187ee6373ddSDimitris Michailidis {
1188ee6373ddSDimitris Michailidis 	if (ed->num_vports)
1189ee6373ddSDimitris Michailidis 		return -EINVAL;
1190ee6373ddSDimitris Michailidis 
1191ee6373ddSDimitris Michailidis 	ed->vport_info = kvcalloc(n, sizeof(*ed->vport_info), GFP_KERNEL);
1192ee6373ddSDimitris Michailidis 	if (!ed->vport_info)
1193ee6373ddSDimitris Michailidis 		return -ENOMEM;
1194ee6373ddSDimitris Michailidis 	ed->num_vports = n;
1195ee6373ddSDimitris Michailidis 	return 0;
1196ee6373ddSDimitris Michailidis }
1197ee6373ddSDimitris Michailidis 
fun_free_vports(struct fun_ethdev * ed)1198ee6373ddSDimitris Michailidis static void fun_free_vports(struct fun_ethdev *ed)
1199ee6373ddSDimitris Michailidis {
1200ee6373ddSDimitris Michailidis 	kvfree(ed->vport_info);
1201ee6373ddSDimitris Michailidis 	ed->vport_info = NULL;
1202ee6373ddSDimitris Michailidis 	ed->num_vports = 0;
1203ee6373ddSDimitris Michailidis }
1204ee6373ddSDimitris Michailidis 
fun_get_vport(struct fun_ethdev * ed,unsigned int vport)1205ee6373ddSDimitris Michailidis static struct fun_vport_info *fun_get_vport(struct fun_ethdev *ed,
1206ee6373ddSDimitris Michailidis 					    unsigned int vport)
1207ee6373ddSDimitris Michailidis {
1208ee6373ddSDimitris Michailidis 	if (!ed->vport_info || vport >= ed->num_vports)
1209ee6373ddSDimitris Michailidis 		return NULL;
1210ee6373ddSDimitris Michailidis 
1211ee6373ddSDimitris Michailidis 	return ed->vport_info + vport;
1212ee6373ddSDimitris Michailidis }
1213ee6373ddSDimitris Michailidis 
fun_set_vf_mac(struct net_device * dev,int vf,u8 * mac)1214ee6373ddSDimitris Michailidis static int fun_set_vf_mac(struct net_device *dev, int vf, u8 *mac)
1215ee6373ddSDimitris Michailidis {
1216ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1217ee6373ddSDimitris Michailidis 	struct fun_adi_param mac_param = {};
1218ee6373ddSDimitris Michailidis 	struct fun_dev *fdev = fp->fdev;
1219ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed = to_fun_ethdev(fdev);
1220ee6373ddSDimitris Michailidis 	struct fun_vport_info *vi;
1221ee6373ddSDimitris Michailidis 	int rc = -EINVAL;
1222ee6373ddSDimitris Michailidis 
1223ee6373ddSDimitris Michailidis 	if (is_multicast_ether_addr(mac))
1224ee6373ddSDimitris Michailidis 		return -EINVAL;
1225ee6373ddSDimitris Michailidis 
1226ee6373ddSDimitris Michailidis 	mutex_lock(&ed->state_mutex);
1227ee6373ddSDimitris Michailidis 	vi = fun_get_vport(ed, vf);
1228ee6373ddSDimitris Michailidis 	if (!vi)
1229ee6373ddSDimitris Michailidis 		goto unlock;
1230ee6373ddSDimitris Michailidis 
1231ee6373ddSDimitris Michailidis 	mac_param.u.mac = FUN_ADI_MAC_INIT(ether_addr_to_u64(mac));
1232ee6373ddSDimitris Michailidis 	rc = fun_adi_write(fdev, FUN_ADMIN_ADI_ATTR_MACADDR, vf + 1,
1233ee6373ddSDimitris Michailidis 			   &mac_param);
1234ee6373ddSDimitris Michailidis 	if (!rc)
1235ee6373ddSDimitris Michailidis 		ether_addr_copy(vi->mac, mac);
1236ee6373ddSDimitris Michailidis unlock:
1237ee6373ddSDimitris Michailidis 	mutex_unlock(&ed->state_mutex);
1238ee6373ddSDimitris Michailidis 	return rc;
1239ee6373ddSDimitris Michailidis }
1240ee6373ddSDimitris Michailidis 
fun_set_vf_vlan(struct net_device * dev,int vf,u16 vlan,u8 qos,__be16 vlan_proto)1241ee6373ddSDimitris Michailidis static int fun_set_vf_vlan(struct net_device *dev, int vf, u16 vlan, u8 qos,
1242ee6373ddSDimitris Michailidis 			   __be16 vlan_proto)
1243ee6373ddSDimitris Michailidis {
1244ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1245ee6373ddSDimitris Michailidis 	struct fun_adi_param vlan_param = {};
1246ee6373ddSDimitris Michailidis 	struct fun_dev *fdev = fp->fdev;
1247ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed = to_fun_ethdev(fdev);
1248ee6373ddSDimitris Michailidis 	struct fun_vport_info *vi;
1249ee6373ddSDimitris Michailidis 	int rc = -EINVAL;
1250ee6373ddSDimitris Michailidis 
1251ee6373ddSDimitris Michailidis 	if (vlan > 4095 || qos > 7)
1252ee6373ddSDimitris Michailidis 		return -EINVAL;
1253ee6373ddSDimitris Michailidis 	if (vlan_proto && vlan_proto != htons(ETH_P_8021Q) &&
1254ee6373ddSDimitris Michailidis 	    vlan_proto != htons(ETH_P_8021AD))
1255ee6373ddSDimitris Michailidis 		return -EINVAL;
1256ee6373ddSDimitris Michailidis 
1257ee6373ddSDimitris Michailidis 	mutex_lock(&ed->state_mutex);
1258ee6373ddSDimitris Michailidis 	vi = fun_get_vport(ed, vf);
1259ee6373ddSDimitris Michailidis 	if (!vi)
1260ee6373ddSDimitris Michailidis 		goto unlock;
1261ee6373ddSDimitris Michailidis 
1262ee6373ddSDimitris Michailidis 	vlan_param.u.vlan = FUN_ADI_VLAN_INIT(be16_to_cpu(vlan_proto),
1263ee6373ddSDimitris Michailidis 					      ((u16)qos << VLAN_PRIO_SHIFT) | vlan);
1264ee6373ddSDimitris Michailidis 	rc = fun_adi_write(fdev, FUN_ADMIN_ADI_ATTR_VLAN, vf + 1, &vlan_param);
1265ee6373ddSDimitris Michailidis 	if (!rc) {
1266ee6373ddSDimitris Michailidis 		vi->vlan = vlan;
1267ee6373ddSDimitris Michailidis 		vi->qos = qos;
1268ee6373ddSDimitris Michailidis 		vi->vlan_proto = vlan_proto;
1269ee6373ddSDimitris Michailidis 	}
1270ee6373ddSDimitris Michailidis unlock:
1271ee6373ddSDimitris Michailidis 	mutex_unlock(&ed->state_mutex);
1272ee6373ddSDimitris Michailidis 	return rc;
1273ee6373ddSDimitris Michailidis }
1274ee6373ddSDimitris Michailidis 
fun_set_vf_rate(struct net_device * dev,int vf,int min_tx_rate,int max_tx_rate)1275ee6373ddSDimitris Michailidis static int fun_set_vf_rate(struct net_device *dev, int vf, int min_tx_rate,
1276ee6373ddSDimitris Michailidis 			   int max_tx_rate)
1277ee6373ddSDimitris Michailidis {
1278ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1279ee6373ddSDimitris Michailidis 	struct fun_adi_param rate_param = {};
1280ee6373ddSDimitris Michailidis 	struct fun_dev *fdev = fp->fdev;
1281ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed = to_fun_ethdev(fdev);
1282ee6373ddSDimitris Michailidis 	struct fun_vport_info *vi;
1283ee6373ddSDimitris Michailidis 	int rc = -EINVAL;
1284ee6373ddSDimitris Michailidis 
1285ee6373ddSDimitris Michailidis 	if (min_tx_rate)
1286ee6373ddSDimitris Michailidis 		return -EINVAL;
1287ee6373ddSDimitris Michailidis 
1288ee6373ddSDimitris Michailidis 	mutex_lock(&ed->state_mutex);
1289ee6373ddSDimitris Michailidis 	vi = fun_get_vport(ed, vf);
1290ee6373ddSDimitris Michailidis 	if (!vi)
1291ee6373ddSDimitris Michailidis 		goto unlock;
1292ee6373ddSDimitris Michailidis 
1293ee6373ddSDimitris Michailidis 	rate_param.u.rate = FUN_ADI_RATE_INIT(max_tx_rate);
1294ee6373ddSDimitris Michailidis 	rc = fun_adi_write(fdev, FUN_ADMIN_ADI_ATTR_RATE, vf + 1, &rate_param);
1295ee6373ddSDimitris Michailidis 	if (!rc)
1296ee6373ddSDimitris Michailidis 		vi->max_rate = max_tx_rate;
1297ee6373ddSDimitris Michailidis unlock:
1298ee6373ddSDimitris Michailidis 	mutex_unlock(&ed->state_mutex);
1299ee6373ddSDimitris Michailidis 	return rc;
1300ee6373ddSDimitris Michailidis }
1301ee6373ddSDimitris Michailidis 
fun_get_vf_config(struct net_device * dev,int vf,struct ifla_vf_info * ivi)1302ee6373ddSDimitris Michailidis static int fun_get_vf_config(struct net_device *dev, int vf,
1303ee6373ddSDimitris Michailidis 			     struct ifla_vf_info *ivi)
1304ee6373ddSDimitris Michailidis {
1305ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1306ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed = to_fun_ethdev(fp->fdev);
1307ee6373ddSDimitris Michailidis 	const struct fun_vport_info *vi;
1308ee6373ddSDimitris Michailidis 
1309ee6373ddSDimitris Michailidis 	mutex_lock(&ed->state_mutex);
1310ee6373ddSDimitris Michailidis 	vi = fun_get_vport(ed, vf);
1311ee6373ddSDimitris Michailidis 	if (!vi)
1312ee6373ddSDimitris Michailidis 		goto unlock;
1313ee6373ddSDimitris Michailidis 
1314ee6373ddSDimitris Michailidis 	memset(ivi, 0, sizeof(*ivi));
1315ee6373ddSDimitris Michailidis 	ivi->vf = vf;
1316ee6373ddSDimitris Michailidis 	ether_addr_copy(ivi->mac, vi->mac);
1317ee6373ddSDimitris Michailidis 	ivi->vlan = vi->vlan;
1318ee6373ddSDimitris Michailidis 	ivi->qos = vi->qos;
1319ee6373ddSDimitris Michailidis 	ivi->vlan_proto = vi->vlan_proto;
1320ee6373ddSDimitris Michailidis 	ivi->max_tx_rate = vi->max_rate;
1321ee6373ddSDimitris Michailidis 	ivi->spoofchk = vi->spoofchk;
1322ee6373ddSDimitris Michailidis unlock:
1323ee6373ddSDimitris Michailidis 	mutex_unlock(&ed->state_mutex);
1324ee6373ddSDimitris Michailidis 	return vi ? 0 : -EINVAL;
1325ee6373ddSDimitris Michailidis }
1326ee6373ddSDimitris Michailidis 
fun_uninit(struct net_device * dev)1327ee6373ddSDimitris Michailidis static void fun_uninit(struct net_device *dev)
1328ee6373ddSDimitris Michailidis {
1329ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1330ee6373ddSDimitris Michailidis 
1331ee6373ddSDimitris Michailidis 	fun_prune_queue_irqs(dev);
1332ee6373ddSDimitris Michailidis 	xa_destroy(&fp->irqs);
1333ee6373ddSDimitris Michailidis }
1334ee6373ddSDimitris Michailidis 
1335ee6373ddSDimitris Michailidis static const struct net_device_ops fun_netdev_ops = {
1336ee6373ddSDimitris Michailidis 	.ndo_open		= funeth_open,
1337ee6373ddSDimitris Michailidis 	.ndo_stop		= funeth_close,
1338ee6373ddSDimitris Michailidis 	.ndo_start_xmit		= fun_start_xmit,
1339ee6373ddSDimitris Michailidis 	.ndo_get_stats64	= fun_get_stats64,
1340ee6373ddSDimitris Michailidis 	.ndo_change_mtu		= fun_change_mtu,
1341ee6373ddSDimitris Michailidis 	.ndo_set_mac_address	= fun_set_macaddr,
1342ee6373ddSDimitris Michailidis 	.ndo_validate_addr	= eth_validate_addr,
1343ee6373ddSDimitris Michailidis 	.ndo_eth_ioctl		= fun_ioctl,
1344ee6373ddSDimitris Michailidis 	.ndo_uninit		= fun_uninit,
1345ee6373ddSDimitris Michailidis 	.ndo_bpf		= fun_xdp,
1346ee6373ddSDimitris Michailidis 	.ndo_xdp_xmit		= fun_xdp_xmit_frames,
1347ee6373ddSDimitris Michailidis 	.ndo_set_vf_mac		= fun_set_vf_mac,
1348ee6373ddSDimitris Michailidis 	.ndo_set_vf_vlan	= fun_set_vf_vlan,
1349ee6373ddSDimitris Michailidis 	.ndo_set_vf_rate	= fun_set_vf_rate,
1350ee6373ddSDimitris Michailidis 	.ndo_get_vf_config	= fun_get_vf_config,
1351ee6373ddSDimitris Michailidis };
1352ee6373ddSDimitris Michailidis 
1353ee6373ddSDimitris Michailidis #define GSO_ENCAP_FLAGS (NETIF_F_GSO_GRE | NETIF_F_GSO_IPXIP4 | \
1354ee6373ddSDimitris Michailidis 			 NETIF_F_GSO_IPXIP6 | NETIF_F_GSO_UDP_TUNNEL | \
1355ee6373ddSDimitris Michailidis 			 NETIF_F_GSO_UDP_TUNNEL_CSUM)
13566ce1df88SDimitris Michailidis #define TSO_FLAGS (NETIF_F_TSO | NETIF_F_TSO6 | NETIF_F_TSO_ECN | \
13576ce1df88SDimitris Michailidis 		   NETIF_F_GSO_UDP_L4)
1358ee6373ddSDimitris Michailidis #define VLAN_FEAT (NETIF_F_SG | NETIF_F_HW_CSUM | TSO_FLAGS | \
1359ee6373ddSDimitris Michailidis 		   GSO_ENCAP_FLAGS | NETIF_F_HIGHDMA)
1360ee6373ddSDimitris Michailidis 
fun_dflt_rss_indir(struct funeth_priv * fp,unsigned int nrx)1361ee6373ddSDimitris Michailidis static void fun_dflt_rss_indir(struct funeth_priv *fp, unsigned int nrx)
1362ee6373ddSDimitris Michailidis {
1363ee6373ddSDimitris Michailidis 	unsigned int i;
1364ee6373ddSDimitris Michailidis 
1365ee6373ddSDimitris Michailidis 	for (i = 0; i < fp->indir_table_nentries; i++)
1366ee6373ddSDimitris Michailidis 		fp->indir_table[i] = ethtool_rxfh_indir_default(i, nrx);
1367ee6373ddSDimitris Michailidis }
1368ee6373ddSDimitris Michailidis 
1369ee6373ddSDimitris Michailidis /* Reset the RSS indirection table to equal distribution across the current
1370ee6373ddSDimitris Michailidis  * number of Rx queues. Called at init time and whenever the number of Rx
1371ee6373ddSDimitris Michailidis  * queues changes subsequently. Note that this may also resize the indirection
1372ee6373ddSDimitris Michailidis  * table.
1373ee6373ddSDimitris Michailidis  */
fun_reset_rss_indir(struct net_device * dev,unsigned int nrx)1374ee6373ddSDimitris Michailidis static void fun_reset_rss_indir(struct net_device *dev, unsigned int nrx)
1375ee6373ddSDimitris Michailidis {
1376ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1377ee6373ddSDimitris Michailidis 
1378ee6373ddSDimitris Michailidis 	if (!fp->rss_cfg)
1379ee6373ddSDimitris Michailidis 		return;
1380ee6373ddSDimitris Michailidis 
1381ee6373ddSDimitris Michailidis 	/* Set the table size to the max possible that allows an equal number
1382ee6373ddSDimitris Michailidis 	 * of occurrences of each CQ.
1383ee6373ddSDimitris Michailidis 	 */
1384ee6373ddSDimitris Michailidis 	fp->indir_table_nentries = rounddown(FUN_ETH_RSS_MAX_INDIR_ENT, nrx);
1385ee6373ddSDimitris Michailidis 	fun_dflt_rss_indir(fp, nrx);
1386ee6373ddSDimitris Michailidis }
1387ee6373ddSDimitris Michailidis 
1388ee6373ddSDimitris Michailidis /* Update the RSS LUT to contain only queues in [0, nrx). Normally this will
1389ee6373ddSDimitris Michailidis  * update the LUT to an equal distribution among nrx queues, If @only_if_needed
1390ee6373ddSDimitris Michailidis  * is set the LUT is left unchanged if it already does not reference any queues
1391ee6373ddSDimitris Michailidis  * >= nrx.
1392ee6373ddSDimitris Michailidis  */
fun_rss_set_qnum(struct net_device * dev,unsigned int nrx,bool only_if_needed)1393ee6373ddSDimitris Michailidis static int fun_rss_set_qnum(struct net_device *dev, unsigned int nrx,
1394ee6373ddSDimitris Michailidis 			    bool only_if_needed)
1395ee6373ddSDimitris Michailidis {
1396ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1397ee6373ddSDimitris Michailidis 	u32 old_lut[FUN_ETH_RSS_MAX_INDIR_ENT];
1398ee6373ddSDimitris Michailidis 	unsigned int i, oldsz;
1399ee6373ddSDimitris Michailidis 	int err;
1400ee6373ddSDimitris Michailidis 
1401ee6373ddSDimitris Michailidis 	if (!fp->rss_cfg)
1402ee6373ddSDimitris Michailidis 		return 0;
1403ee6373ddSDimitris Michailidis 
1404ee6373ddSDimitris Michailidis 	if (only_if_needed) {
1405ee6373ddSDimitris Michailidis 		for (i = 0; i < fp->indir_table_nentries; i++)
1406ee6373ddSDimitris Michailidis 			if (fp->indir_table[i] >= nrx)
1407ee6373ddSDimitris Michailidis 				break;
1408ee6373ddSDimitris Michailidis 
1409ee6373ddSDimitris Michailidis 		if (i >= fp->indir_table_nentries)
1410ee6373ddSDimitris Michailidis 			return 0;
1411ee6373ddSDimitris Michailidis 	}
1412ee6373ddSDimitris Michailidis 
1413ee6373ddSDimitris Michailidis 	memcpy(old_lut, fp->indir_table, sizeof(old_lut));
1414ee6373ddSDimitris Michailidis 	oldsz = fp->indir_table_nentries;
1415ee6373ddSDimitris Michailidis 	fun_reset_rss_indir(dev, nrx);
1416ee6373ddSDimitris Michailidis 
1417ee6373ddSDimitris Michailidis 	err = fun_config_rss(dev, fp->hash_algo, fp->rss_key,
1418ee6373ddSDimitris Michailidis 			     fp->indir_table, FUN_ADMIN_SUBOP_MODIFY);
1419ee6373ddSDimitris Michailidis 	if (!err)
1420ee6373ddSDimitris Michailidis 		return 0;
1421ee6373ddSDimitris Michailidis 
1422ee6373ddSDimitris Michailidis 	memcpy(fp->indir_table, old_lut, sizeof(old_lut));
1423ee6373ddSDimitris Michailidis 	fp->indir_table_nentries = oldsz;
1424ee6373ddSDimitris Michailidis 	return err;
1425ee6373ddSDimitris Michailidis }
1426ee6373ddSDimitris Michailidis 
1427ee6373ddSDimitris Michailidis /* Allocate the DMA area for the RSS configuration commands to the device, and
1428ee6373ddSDimitris Michailidis  * initialize the hash, hash key, indirection table size and its entries to
1429ee6373ddSDimitris Michailidis  * their defaults. The indirection table defaults to equal distribution across
1430ee6373ddSDimitris Michailidis  * the Rx queues.
1431ee6373ddSDimitris Michailidis  */
fun_init_rss(struct net_device * dev)1432ee6373ddSDimitris Michailidis static int fun_init_rss(struct net_device *dev)
1433ee6373ddSDimitris Michailidis {
1434ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1435ee6373ddSDimitris Michailidis 	size_t size = sizeof(fp->rss_key) + sizeof(fp->indir_table);
1436ee6373ddSDimitris Michailidis 
1437ee6373ddSDimitris Michailidis 	fp->rss_hw_id = FUN_HCI_ID_INVALID;
1438ee6373ddSDimitris Michailidis 	if (!(fp->port_caps & FUN_PORT_CAP_OFFLOADS))
1439ee6373ddSDimitris Michailidis 		return 0;
1440ee6373ddSDimitris Michailidis 
1441ee6373ddSDimitris Michailidis 	fp->rss_cfg = dma_alloc_coherent(&fp->pdev->dev, size,
1442ee6373ddSDimitris Michailidis 					 &fp->rss_dma_addr, GFP_KERNEL);
1443ee6373ddSDimitris Michailidis 	if (!fp->rss_cfg)
1444ee6373ddSDimitris Michailidis 		return -ENOMEM;
1445ee6373ddSDimitris Michailidis 
1446ee6373ddSDimitris Michailidis 	fp->hash_algo = FUN_ETH_RSS_ALG_TOEPLITZ;
1447ee6373ddSDimitris Michailidis 	netdev_rss_key_fill(fp->rss_key, sizeof(fp->rss_key));
1448ee6373ddSDimitris Michailidis 	fun_reset_rss_indir(dev, dev->real_num_rx_queues);
1449ee6373ddSDimitris Michailidis 	return 0;
1450ee6373ddSDimitris Michailidis }
1451ee6373ddSDimitris Michailidis 
fun_free_rss(struct funeth_priv * fp)1452ee6373ddSDimitris Michailidis static void fun_free_rss(struct funeth_priv *fp)
1453ee6373ddSDimitris Michailidis {
1454ee6373ddSDimitris Michailidis 	if (fp->rss_cfg) {
1455ee6373ddSDimitris Michailidis 		dma_free_coherent(&fp->pdev->dev,
1456ee6373ddSDimitris Michailidis 				  sizeof(fp->rss_key) + sizeof(fp->indir_table),
1457ee6373ddSDimitris Michailidis 				  fp->rss_cfg, fp->rss_dma_addr);
1458ee6373ddSDimitris Michailidis 		fp->rss_cfg = NULL;
1459ee6373ddSDimitris Michailidis 	}
1460ee6373ddSDimitris Michailidis }
1461ee6373ddSDimitris Michailidis 
fun_set_ring_count(struct net_device * netdev,unsigned int ntx,unsigned int nrx)1462ee6373ddSDimitris Michailidis void fun_set_ring_count(struct net_device *netdev, unsigned int ntx,
1463ee6373ddSDimitris Michailidis 			unsigned int nrx)
1464ee6373ddSDimitris Michailidis {
1465ee6373ddSDimitris Michailidis 	netif_set_real_num_tx_queues(netdev, ntx);
1466ee6373ddSDimitris Michailidis 	if (nrx != netdev->real_num_rx_queues) {
1467ee6373ddSDimitris Michailidis 		netif_set_real_num_rx_queues(netdev, nrx);
1468ee6373ddSDimitris Michailidis 		fun_reset_rss_indir(netdev, nrx);
1469ee6373ddSDimitris Michailidis 	}
1470ee6373ddSDimitris Michailidis }
1471ee6373ddSDimitris Michailidis 
fun_init_stats_area(struct funeth_priv * fp)1472ee6373ddSDimitris Michailidis static int fun_init_stats_area(struct funeth_priv *fp)
1473ee6373ddSDimitris Michailidis {
1474ee6373ddSDimitris Michailidis 	unsigned int nstats;
1475ee6373ddSDimitris Michailidis 
1476ee6373ddSDimitris Michailidis 	if (!(fp->port_caps & FUN_PORT_CAP_STATS))
1477ee6373ddSDimitris Michailidis 		return 0;
1478ee6373ddSDimitris Michailidis 
1479ee6373ddSDimitris Michailidis 	nstats = PORT_MAC_RX_STATS_MAX + PORT_MAC_TX_STATS_MAX +
1480ee6373ddSDimitris Michailidis 		 PORT_MAC_FEC_STATS_MAX;
1481ee6373ddSDimitris Michailidis 
1482ee6373ddSDimitris Michailidis 	fp->stats = dma_alloc_coherent(&fp->pdev->dev, nstats * sizeof(u64),
1483ee6373ddSDimitris Michailidis 				       &fp->stats_dma_addr, GFP_KERNEL);
1484ee6373ddSDimitris Michailidis 	if (!fp->stats)
1485ee6373ddSDimitris Michailidis 		return -ENOMEM;
1486ee6373ddSDimitris Michailidis 	return 0;
1487ee6373ddSDimitris Michailidis }
1488ee6373ddSDimitris Michailidis 
fun_free_stats_area(struct funeth_priv * fp)1489ee6373ddSDimitris Michailidis static void fun_free_stats_area(struct funeth_priv *fp)
1490ee6373ddSDimitris Michailidis {
1491ee6373ddSDimitris Michailidis 	unsigned int nstats;
1492ee6373ddSDimitris Michailidis 
1493ee6373ddSDimitris Michailidis 	if (fp->stats) {
1494ee6373ddSDimitris Michailidis 		nstats = PORT_MAC_RX_STATS_MAX + PORT_MAC_TX_STATS_MAX;
1495ee6373ddSDimitris Michailidis 		dma_free_coherent(&fp->pdev->dev, nstats * sizeof(u64),
1496ee6373ddSDimitris Michailidis 				  fp->stats, fp->stats_dma_addr);
1497ee6373ddSDimitris Michailidis 		fp->stats = NULL;
1498ee6373ddSDimitris Michailidis 	}
1499ee6373ddSDimitris Michailidis }
1500ee6373ddSDimitris Michailidis 
fun_dl_port_register(struct net_device * netdev)1501ee6373ddSDimitris Michailidis static int fun_dl_port_register(struct net_device *netdev)
1502ee6373ddSDimitris Michailidis {
1503ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(netdev);
1504ee6373ddSDimitris Michailidis 	struct devlink *dl = priv_to_devlink(fp->fdev);
1505ee6373ddSDimitris Michailidis 	struct devlink_port_attrs attrs = {};
1506ee6373ddSDimitris Michailidis 	unsigned int idx;
1507ee6373ddSDimitris Michailidis 
1508ee6373ddSDimitris Michailidis 	if (fp->port_caps & FUN_PORT_CAP_VPORT) {
1509ee6373ddSDimitris Michailidis 		attrs.flavour = DEVLINK_PORT_FLAVOUR_VIRTUAL;
1510ee6373ddSDimitris Michailidis 		idx = fp->lport;
1511ee6373ddSDimitris Michailidis 	} else {
1512ee6373ddSDimitris Michailidis 		idx = netdev->dev_port;
1513ee6373ddSDimitris Michailidis 		attrs.flavour = DEVLINK_PORT_FLAVOUR_PHYSICAL;
1514ee6373ddSDimitris Michailidis 		attrs.lanes = fp->lane_attrs & 7;
1515ee6373ddSDimitris Michailidis 		if (fp->lane_attrs & FUN_PORT_LANE_SPLIT) {
1516ee6373ddSDimitris Michailidis 			attrs.split = 1;
1517ee6373ddSDimitris Michailidis 			attrs.phys.port_number = fp->lport & ~3;
1518ee6373ddSDimitris Michailidis 			attrs.phys.split_subport_number = fp->lport & 3;
1519ee6373ddSDimitris Michailidis 		} else {
1520ee6373ddSDimitris Michailidis 			attrs.phys.port_number = fp->lport;
1521ee6373ddSDimitris Michailidis 		}
1522ee6373ddSDimitris Michailidis 	}
1523ee6373ddSDimitris Michailidis 
1524ee6373ddSDimitris Michailidis 	devlink_port_attrs_set(&fp->dl_port, &attrs);
1525ee6373ddSDimitris Michailidis 
1526ee6373ddSDimitris Michailidis 	return devlink_port_register(dl, &fp->dl_port, idx);
1527ee6373ddSDimitris Michailidis }
1528ee6373ddSDimitris Michailidis 
1529ee6373ddSDimitris Michailidis /* Determine the max Tx/Rx queues for a port. */
fun_max_qs(struct fun_ethdev * ed,unsigned int * ntx,unsigned int * nrx)1530ee6373ddSDimitris Michailidis static int fun_max_qs(struct fun_ethdev *ed, unsigned int *ntx,
1531ee6373ddSDimitris Michailidis 		      unsigned int *nrx)
1532ee6373ddSDimitris Michailidis {
1533ee6373ddSDimitris Michailidis 	int neth;
1534ee6373ddSDimitris Michailidis 
1535ee6373ddSDimitris Michailidis 	if (ed->num_ports > 1 || is_kdump_kernel()) {
1536ee6373ddSDimitris Michailidis 		*ntx = 1;
1537ee6373ddSDimitris Michailidis 		*nrx = 1;
1538ee6373ddSDimitris Michailidis 		return 0;
1539ee6373ddSDimitris Michailidis 	}
1540ee6373ddSDimitris Michailidis 
1541ee6373ddSDimitris Michailidis 	neth = fun_get_res_count(&ed->fdev, FUN_ADMIN_OP_ETH);
1542ee6373ddSDimitris Michailidis 	if (neth < 0)
1543ee6373ddSDimitris Michailidis 		return neth;
1544ee6373ddSDimitris Michailidis 
1545ee6373ddSDimitris Michailidis 	/* We determine the max number of queues based on the CPU
1546ee6373ddSDimitris Michailidis 	 * cores, device interrupts and queues, RSS size, and device Tx flows.
1547ee6373ddSDimitris Michailidis 	 *
1548ee6373ddSDimitris Michailidis 	 * - At least 1 Rx and 1 Tx queues.
1549ee6373ddSDimitris Michailidis 	 * - At most 1 Rx/Tx queue per core.
1550ee6373ddSDimitris Michailidis 	 * - Each Rx/Tx queue needs 1 SQ.
1551ee6373ddSDimitris Michailidis 	 */
1552ee6373ddSDimitris Michailidis 	*ntx = min(ed->nsqs_per_port - 1, num_online_cpus());
1553ee6373ddSDimitris Michailidis 	*nrx = *ntx;
1554ee6373ddSDimitris Michailidis 	if (*ntx > neth)
1555ee6373ddSDimitris Michailidis 		*ntx = neth;
1556ee6373ddSDimitris Michailidis 	if (*nrx > FUN_ETH_RSS_MAX_INDIR_ENT)
1557ee6373ddSDimitris Michailidis 		*nrx = FUN_ETH_RSS_MAX_INDIR_ENT;
1558ee6373ddSDimitris Michailidis 	return 0;
1559ee6373ddSDimitris Michailidis }
1560ee6373ddSDimitris Michailidis 
fun_queue_defaults(struct net_device * dev,unsigned int nsqs)1561ee6373ddSDimitris Michailidis static void fun_queue_defaults(struct net_device *dev, unsigned int nsqs)
1562ee6373ddSDimitris Michailidis {
1563ee6373ddSDimitris Michailidis 	unsigned int ntx, nrx;
1564ee6373ddSDimitris Michailidis 
1565ee6373ddSDimitris Michailidis 	ntx = min(dev->num_tx_queues, FUN_DFLT_QUEUES);
1566ee6373ddSDimitris Michailidis 	nrx = min(dev->num_rx_queues, FUN_DFLT_QUEUES);
1567ee6373ddSDimitris Michailidis 	if (ntx <= nrx) {
1568ee6373ddSDimitris Michailidis 		ntx = min(ntx, nsqs / 2);
1569ee6373ddSDimitris Michailidis 		nrx = min(nrx, nsqs - ntx);
1570ee6373ddSDimitris Michailidis 	} else {
1571ee6373ddSDimitris Michailidis 		nrx = min(nrx, nsqs / 2);
1572ee6373ddSDimitris Michailidis 		ntx = min(ntx, nsqs - nrx);
1573ee6373ddSDimitris Michailidis 	}
1574ee6373ddSDimitris Michailidis 
1575ee6373ddSDimitris Michailidis 	netif_set_real_num_tx_queues(dev, ntx);
1576ee6373ddSDimitris Michailidis 	netif_set_real_num_rx_queues(dev, nrx);
1577ee6373ddSDimitris Michailidis }
1578ee6373ddSDimitris Michailidis 
1579ee6373ddSDimitris Michailidis /* Replace the existing Rx/Tx/XDP queues with equal number of queues with
1580ee6373ddSDimitris Michailidis  * different settings, e.g. depth. This is a disruptive replacement that
1581ee6373ddSDimitris Michailidis  * temporarily shuts down the data path and should be limited to changes that
1582ee6373ddSDimitris Michailidis  * can't be applied to live queues. The old queues are always discarded.
1583ee6373ddSDimitris Michailidis  */
fun_replace_queues(struct net_device * dev,struct fun_qset * newqs,struct netlink_ext_ack * extack)1584ee6373ddSDimitris Michailidis int fun_replace_queues(struct net_device *dev, struct fun_qset *newqs,
1585ee6373ddSDimitris Michailidis 		       struct netlink_ext_ack *extack)
1586ee6373ddSDimitris Michailidis {
1587ee6373ddSDimitris Michailidis 	struct fun_qset oldqs = { .state = FUN_QSTATE_DESTROYED };
1588ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1589ee6373ddSDimitris Michailidis 	int err;
1590ee6373ddSDimitris Michailidis 
1591ee6373ddSDimitris Michailidis 	newqs->nrxqs = dev->real_num_rx_queues;
1592ee6373ddSDimitris Michailidis 	newqs->ntxqs = dev->real_num_tx_queues;
1593ee6373ddSDimitris Michailidis 	newqs->nxdpqs = fp->num_xdpqs;
1594ee6373ddSDimitris Michailidis 	newqs->state = FUN_QSTATE_INIT_SW;
1595ee6373ddSDimitris Michailidis 	err = fun_alloc_rings(dev, newqs);
1596ee6373ddSDimitris Michailidis 	if (err) {
1597ee6373ddSDimitris Michailidis 		NL_SET_ERR_MSG_MOD(extack,
1598ee6373ddSDimitris Michailidis 				   "Unable to allocate memory for new queues, keeping current settings");
1599ee6373ddSDimitris Michailidis 		return err;
1600ee6373ddSDimitris Michailidis 	}
1601ee6373ddSDimitris Michailidis 
1602ee6373ddSDimitris Michailidis 	fun_down(dev, &oldqs);
1603ee6373ddSDimitris Michailidis 
1604ee6373ddSDimitris Michailidis 	err = fun_up(dev, newqs);
1605ee6373ddSDimitris Michailidis 	if (!err)
1606ee6373ddSDimitris Michailidis 		return 0;
1607ee6373ddSDimitris Michailidis 
1608ee6373ddSDimitris Michailidis 	/* The new queues couldn't be installed. We do not retry the old queues
1609ee6373ddSDimitris Michailidis 	 * as they are the same to the device as the new queues and would
1610ee6373ddSDimitris Michailidis 	 * similarly fail.
1611ee6373ddSDimitris Michailidis 	 */
1612ee6373ddSDimitris Michailidis 	newqs->state = FUN_QSTATE_DESTROYED;
1613ee6373ddSDimitris Michailidis 	fun_free_rings(dev, newqs);
1614ee6373ddSDimitris Michailidis 	NL_SET_ERR_MSG_MOD(extack, "Unable to restore the data path with the new queues.");
1615ee6373ddSDimitris Michailidis 	return err;
1616ee6373ddSDimitris Michailidis }
1617ee6373ddSDimitris Michailidis 
1618ee6373ddSDimitris Michailidis /* Change the number of Rx/Tx queues of a device while it is up. This is done
1619ee6373ddSDimitris Michailidis  * by incrementally adding/removing queues to meet the new requirements while
1620ee6373ddSDimitris Michailidis  * handling ongoing traffic.
1621ee6373ddSDimitris Michailidis  */
fun_change_num_queues(struct net_device * dev,unsigned int ntx,unsigned int nrx)1622ee6373ddSDimitris Michailidis int fun_change_num_queues(struct net_device *dev, unsigned int ntx,
1623ee6373ddSDimitris Michailidis 			  unsigned int nrx)
1624ee6373ddSDimitris Michailidis {
1625ee6373ddSDimitris Michailidis 	unsigned int keep_tx = min(dev->real_num_tx_queues, ntx);
1626ee6373ddSDimitris Michailidis 	unsigned int keep_rx = min(dev->real_num_rx_queues, nrx);
1627ee6373ddSDimitris Michailidis 	struct funeth_priv *fp = netdev_priv(dev);
1628ee6373ddSDimitris Michailidis 	struct fun_qset oldqs = {
1629ee6373ddSDimitris Michailidis 		.rxqs = rtnl_dereference(fp->rxqs),
1630ee6373ddSDimitris Michailidis 		.txqs = fp->txqs,
1631ee6373ddSDimitris Michailidis 		.nrxqs = dev->real_num_rx_queues,
1632ee6373ddSDimitris Michailidis 		.ntxqs = dev->real_num_tx_queues,
1633ee6373ddSDimitris Michailidis 		.rxq_start = keep_rx,
1634ee6373ddSDimitris Michailidis 		.txq_start = keep_tx,
1635ee6373ddSDimitris Michailidis 		.state = FUN_QSTATE_DESTROYED
1636ee6373ddSDimitris Michailidis 	};
1637ee6373ddSDimitris Michailidis 	struct fun_qset newqs = {
1638ee6373ddSDimitris Michailidis 		.nrxqs = nrx,
1639ee6373ddSDimitris Michailidis 		.ntxqs = ntx,
1640ee6373ddSDimitris Michailidis 		.rxq_start = keep_rx,
1641ee6373ddSDimitris Michailidis 		.txq_start = keep_tx,
1642ee6373ddSDimitris Michailidis 		.cq_depth = fp->cq_depth,
1643ee6373ddSDimitris Michailidis 		.rq_depth = fp->rq_depth,
1644ee6373ddSDimitris Michailidis 		.sq_depth = fp->sq_depth,
1645ee6373ddSDimitris Michailidis 		.state = FUN_QSTATE_INIT_FULL
1646ee6373ddSDimitris Michailidis 	};
1647ee6373ddSDimitris Michailidis 	int i, err;
1648ee6373ddSDimitris Michailidis 
1649ee6373ddSDimitris Michailidis 	err = fun_alloc_rings(dev, &newqs);
1650ee6373ddSDimitris Michailidis 	if (err)
1651ee6373ddSDimitris Michailidis 		goto free_irqs;
1652ee6373ddSDimitris Michailidis 
1653ee6373ddSDimitris Michailidis 	err = fun_enable_irqs(dev); /* of any newly added queues */
1654ee6373ddSDimitris Michailidis 	if (err)
1655ee6373ddSDimitris Michailidis 		goto free_rings;
1656ee6373ddSDimitris Michailidis 
1657ee6373ddSDimitris Michailidis 	/* copy the queues we are keeping to the new set */
1658ee6373ddSDimitris Michailidis 	memcpy(newqs.rxqs, oldqs.rxqs, keep_rx * sizeof(*oldqs.rxqs));
1659ee6373ddSDimitris Michailidis 	memcpy(newqs.txqs, fp->txqs, keep_tx * sizeof(*fp->txqs));
1660ee6373ddSDimitris Michailidis 
1661ee6373ddSDimitris Michailidis 	if (nrx < dev->real_num_rx_queues) {
1662ee6373ddSDimitris Michailidis 		err = fun_rss_set_qnum(dev, nrx, true);
1663ee6373ddSDimitris Michailidis 		if (err)
1664ee6373ddSDimitris Michailidis 			goto disable_tx_irqs;
1665ee6373ddSDimitris Michailidis 
1666ee6373ddSDimitris Michailidis 		for (i = nrx; i < dev->real_num_rx_queues; i++)
1667ee6373ddSDimitris Michailidis 			fun_disable_one_irq(container_of(oldqs.rxqs[i]->napi,
1668ee6373ddSDimitris Michailidis 							 struct fun_irq, napi));
1669ee6373ddSDimitris Michailidis 
1670ee6373ddSDimitris Michailidis 		netif_set_real_num_rx_queues(dev, nrx);
1671ee6373ddSDimitris Michailidis 	}
1672ee6373ddSDimitris Michailidis 
1673ee6373ddSDimitris Michailidis 	if (ntx < dev->real_num_tx_queues)
1674ee6373ddSDimitris Michailidis 		netif_set_real_num_tx_queues(dev, ntx);
1675ee6373ddSDimitris Michailidis 
1676ee6373ddSDimitris Michailidis 	rcu_assign_pointer(fp->rxqs, newqs.rxqs);
1677ee6373ddSDimitris Michailidis 	fp->txqs = newqs.txqs;
1678ee6373ddSDimitris Michailidis 	synchronize_net();
1679ee6373ddSDimitris Michailidis 
1680ee6373ddSDimitris Michailidis 	if (ntx > dev->real_num_tx_queues)
1681ee6373ddSDimitris Michailidis 		netif_set_real_num_tx_queues(dev, ntx);
1682ee6373ddSDimitris Michailidis 
1683ee6373ddSDimitris Michailidis 	if (nrx > dev->real_num_rx_queues) {
1684ee6373ddSDimitris Michailidis 		netif_set_real_num_rx_queues(dev, nrx);
1685ee6373ddSDimitris Michailidis 		fun_rss_set_qnum(dev, nrx, false);
1686ee6373ddSDimitris Michailidis 	}
1687ee6373ddSDimitris Michailidis 
1688ee6373ddSDimitris Michailidis 	/* disable interrupts of any excess Tx queues */
1689ee6373ddSDimitris Michailidis 	for (i = keep_tx; i < oldqs.ntxqs; i++)
1690ee6373ddSDimitris Michailidis 		fun_disable_one_irq(oldqs.txqs[i]->irq);
1691ee6373ddSDimitris Michailidis 
1692ee6373ddSDimitris Michailidis 	fun_free_rings(dev, &oldqs);
1693ee6373ddSDimitris Michailidis 	fun_prune_queue_irqs(dev);
1694ee6373ddSDimitris Michailidis 	return 0;
1695ee6373ddSDimitris Michailidis 
1696ee6373ddSDimitris Michailidis disable_tx_irqs:
1697ee6373ddSDimitris Michailidis 	for (i = oldqs.ntxqs; i < ntx; i++)
1698ee6373ddSDimitris Michailidis 		fun_disable_one_irq(newqs.txqs[i]->irq);
1699ee6373ddSDimitris Michailidis free_rings:
1700ee6373ddSDimitris Michailidis 	newqs.state = FUN_QSTATE_DESTROYED;
1701ee6373ddSDimitris Michailidis 	fun_free_rings(dev, &newqs);
1702ee6373ddSDimitris Michailidis free_irqs:
1703ee6373ddSDimitris Michailidis 	fun_prune_queue_irqs(dev);
1704ee6373ddSDimitris Michailidis 	return err;
1705ee6373ddSDimitris Michailidis }
1706ee6373ddSDimitris Michailidis 
fun_create_netdev(struct fun_ethdev * ed,unsigned int portid)1707ee6373ddSDimitris Michailidis static int fun_create_netdev(struct fun_ethdev *ed, unsigned int portid)
1708ee6373ddSDimitris Michailidis {
1709ee6373ddSDimitris Michailidis 	struct fun_dev *fdev = &ed->fdev;
1710ee6373ddSDimitris Michailidis 	struct net_device *netdev;
1711ee6373ddSDimitris Michailidis 	struct funeth_priv *fp;
1712ee6373ddSDimitris Michailidis 	unsigned int ntx, nrx;
1713ee6373ddSDimitris Michailidis 	int rc;
1714ee6373ddSDimitris Michailidis 
1715ee6373ddSDimitris Michailidis 	rc = fun_max_qs(ed, &ntx, &nrx);
1716ee6373ddSDimitris Michailidis 	if (rc)
1717ee6373ddSDimitris Michailidis 		return rc;
1718ee6373ddSDimitris Michailidis 
1719ee6373ddSDimitris Michailidis 	netdev = alloc_etherdev_mqs(sizeof(*fp), ntx, nrx);
1720ee6373ddSDimitris Michailidis 	if (!netdev) {
1721ee6373ddSDimitris Michailidis 		rc = -ENOMEM;
1722ee6373ddSDimitris Michailidis 		goto done;
1723ee6373ddSDimitris Michailidis 	}
1724ee6373ddSDimitris Michailidis 
1725ee6373ddSDimitris Michailidis 	netdev->dev_port = portid;
1726ee6373ddSDimitris Michailidis 	fun_queue_defaults(netdev, ed->nsqs_per_port);
1727ee6373ddSDimitris Michailidis 
1728ee6373ddSDimitris Michailidis 	fp = netdev_priv(netdev);
1729ee6373ddSDimitris Michailidis 	fp->fdev = fdev;
1730ee6373ddSDimitris Michailidis 	fp->pdev = to_pci_dev(fdev->dev);
1731ee6373ddSDimitris Michailidis 	fp->netdev = netdev;
1732ee6373ddSDimitris Michailidis 	xa_init(&fp->irqs);
1733ee6373ddSDimitris Michailidis 	fp->rx_irq_ofst = ntx;
1734ee6373ddSDimitris Michailidis 	seqcount_init(&fp->link_seq);
1735ee6373ddSDimitris Michailidis 
1736ee6373ddSDimitris Michailidis 	fp->lport = INVALID_LPORT;
1737ee6373ddSDimitris Michailidis 	rc = fun_port_create(netdev);
1738ee6373ddSDimitris Michailidis 	if (rc)
1739ee6373ddSDimitris Michailidis 		goto free_netdev;
1740ee6373ddSDimitris Michailidis 
1741ee6373ddSDimitris Michailidis 	/* bind port to admin CQ for async events */
1742ee6373ddSDimitris Michailidis 	rc = fun_bind(fdev, FUN_ADMIN_BIND_TYPE_PORT, portid,
1743ee6373ddSDimitris Michailidis 		      FUN_ADMIN_BIND_TYPE_EPCQ, 0);
1744ee6373ddSDimitris Michailidis 	if (rc)
1745ee6373ddSDimitris Michailidis 		goto destroy_port;
1746ee6373ddSDimitris Michailidis 
1747ee6373ddSDimitris Michailidis 	rc = fun_get_port_attributes(netdev);
1748ee6373ddSDimitris Michailidis 	if (rc)
1749ee6373ddSDimitris Michailidis 		goto destroy_port;
1750ee6373ddSDimitris Michailidis 
1751ee6373ddSDimitris Michailidis 	rc = fun_init_rss(netdev);
1752ee6373ddSDimitris Michailidis 	if (rc)
1753ee6373ddSDimitris Michailidis 		goto destroy_port;
1754ee6373ddSDimitris Michailidis 
1755ee6373ddSDimitris Michailidis 	rc = fun_init_stats_area(fp);
1756ee6373ddSDimitris Michailidis 	if (rc)
1757ee6373ddSDimitris Michailidis 		goto free_rss;
1758ee6373ddSDimitris Michailidis 
1759ee6373ddSDimitris Michailidis 	SET_NETDEV_DEV(netdev, fdev->dev);
1760ac73d4bfSJiri Pirko 	SET_NETDEV_DEVLINK_PORT(netdev, &fp->dl_port);
1761ee6373ddSDimitris Michailidis 	netdev->netdev_ops = &fun_netdev_ops;
1762ee6373ddSDimitris Michailidis 
1763ee6373ddSDimitris Michailidis 	netdev->hw_features = NETIF_F_SG | NETIF_F_RXHASH | NETIF_F_RXCSUM;
1764ee6373ddSDimitris Michailidis 	if (fp->port_caps & FUN_PORT_CAP_OFFLOADS)
1765ee6373ddSDimitris Michailidis 		netdev->hw_features |= NETIF_F_HW_CSUM | TSO_FLAGS;
1766ee6373ddSDimitris Michailidis 	if (fp->port_caps & FUN_PORT_CAP_ENCAP_OFFLOADS)
1767ee6373ddSDimitris Michailidis 		netdev->hw_features |= GSO_ENCAP_FLAGS;
1768ee6373ddSDimitris Michailidis 
1769ee6373ddSDimitris Michailidis 	netdev->features |= netdev->hw_features | NETIF_F_HIGHDMA;
1770ee6373ddSDimitris Michailidis 	netdev->vlan_features = netdev->features & VLAN_FEAT;
1771ee6373ddSDimitris Michailidis 	netdev->mpls_features = netdev->vlan_features;
1772ee6373ddSDimitris Michailidis 	netdev->hw_enc_features = netdev->hw_features;
177366c0e13aSMarek Majtyka 	netdev->xdp_features = NETDEV_XDP_ACT_BASIC | NETDEV_XDP_ACT_REDIRECT;
1774ee6373ddSDimitris Michailidis 
1775ee6373ddSDimitris Michailidis 	netdev->min_mtu = ETH_MIN_MTU;
1776ee6373ddSDimitris Michailidis 	netdev->max_mtu = FUN_MAX_MTU;
1777ee6373ddSDimitris Michailidis 
1778ee6373ddSDimitris Michailidis 	fun_set_ethtool_ops(netdev);
1779ee6373ddSDimitris Michailidis 
1780ee6373ddSDimitris Michailidis 	/* configurable parameters */
1781ee6373ddSDimitris Michailidis 	fp->sq_depth = min(SQ_DEPTH, fdev->q_depth);
1782ee6373ddSDimitris Michailidis 	fp->cq_depth = min(CQ_DEPTH, fdev->q_depth);
1783ee6373ddSDimitris Michailidis 	fp->rq_depth = min_t(unsigned int, RQ_DEPTH, fdev->q_depth);
1784ee6373ddSDimitris Michailidis 	fp->rx_coal_usec  = CQ_INTCOAL_USEC;
1785ee6373ddSDimitris Michailidis 	fp->rx_coal_count = CQ_INTCOAL_NPKT;
1786ee6373ddSDimitris Michailidis 	fp->tx_coal_usec  = SQ_INTCOAL_USEC;
1787ee6373ddSDimitris Michailidis 	fp->tx_coal_count = SQ_INTCOAL_NPKT;
1788ee6373ddSDimitris Michailidis 	fp->cq_irq_db = FUN_IRQ_CQ_DB(fp->rx_coal_usec, fp->rx_coal_count);
1789ee6373ddSDimitris Michailidis 
1790ee6373ddSDimitris Michailidis 	rc = fun_dl_port_register(netdev);
1791ee6373ddSDimitris Michailidis 	if (rc)
1792ee6373ddSDimitris Michailidis 		goto free_stats;
1793ee6373ddSDimitris Michailidis 
1794ee6373ddSDimitris Michailidis 	fp->ktls_id = FUN_HCI_ID_INVALID;
1795ee6373ddSDimitris Michailidis 	fun_ktls_init(netdev);            /* optional, failure OK */
1796ee6373ddSDimitris Michailidis 
1797ee6373ddSDimitris Michailidis 	netif_carrier_off(netdev);
1798ee6373ddSDimitris Michailidis 	ed->netdevs[portid] = netdev;
1799ee6373ddSDimitris Michailidis 	rc = register_netdev(netdev);
1800ee6373ddSDimitris Michailidis 	if (rc)
1801ee6373ddSDimitris Michailidis 		goto unreg_devlink;
1802ee6373ddSDimitris Michailidis 	return 0;
1803ee6373ddSDimitris Michailidis 
1804ee6373ddSDimitris Michailidis unreg_devlink:
1805ee6373ddSDimitris Michailidis 	ed->netdevs[portid] = NULL;
1806ee6373ddSDimitris Michailidis 	fun_ktls_cleanup(fp);
1807ee6373ddSDimitris Michailidis 	devlink_port_unregister(&fp->dl_port);
1808ee6373ddSDimitris Michailidis free_stats:
1809ee6373ddSDimitris Michailidis 	fun_free_stats_area(fp);
1810ee6373ddSDimitris Michailidis free_rss:
1811ee6373ddSDimitris Michailidis 	fun_free_rss(fp);
1812ee6373ddSDimitris Michailidis destroy_port:
1813ee6373ddSDimitris Michailidis 	fun_port_destroy(netdev);
1814ee6373ddSDimitris Michailidis free_netdev:
1815ee6373ddSDimitris Michailidis 	free_netdev(netdev);
1816ee6373ddSDimitris Michailidis done:
1817ee6373ddSDimitris Michailidis 	dev_err(fdev->dev, "couldn't allocate port %u, error %d", portid, rc);
1818ee6373ddSDimitris Michailidis 	return rc;
1819ee6373ddSDimitris Michailidis }
1820ee6373ddSDimitris Michailidis 
fun_destroy_netdev(struct net_device * netdev)1821ee6373ddSDimitris Michailidis static void fun_destroy_netdev(struct net_device *netdev)
1822ee6373ddSDimitris Michailidis {
1823ee6373ddSDimitris Michailidis 	struct funeth_priv *fp;
1824ee6373ddSDimitris Michailidis 
1825ee6373ddSDimitris Michailidis 	fp = netdev_priv(netdev);
1826ee6373ddSDimitris Michailidis 	unregister_netdev(netdev);
1827dfe60949SJiri Pirko 	devlink_port_unregister(&fp->dl_port);
1828ee6373ddSDimitris Michailidis 	fun_ktls_cleanup(fp);
1829ee6373ddSDimitris Michailidis 	fun_free_stats_area(fp);
1830ee6373ddSDimitris Michailidis 	fun_free_rss(fp);
1831ee6373ddSDimitris Michailidis 	fun_port_destroy(netdev);
1832ee6373ddSDimitris Michailidis 	free_netdev(netdev);
1833ee6373ddSDimitris Michailidis }
1834ee6373ddSDimitris Michailidis 
fun_create_ports(struct fun_ethdev * ed,unsigned int nports)1835ee6373ddSDimitris Michailidis static int fun_create_ports(struct fun_ethdev *ed, unsigned int nports)
1836ee6373ddSDimitris Michailidis {
1837ee6373ddSDimitris Michailidis 	struct fun_dev *fd = &ed->fdev;
1838ee6373ddSDimitris Michailidis 	int i, rc;
1839ee6373ddSDimitris Michailidis 
1840ee6373ddSDimitris Michailidis 	/* The admin queue takes 1 IRQ and 2 SQs. */
1841ee6373ddSDimitris Michailidis 	ed->nsqs_per_port = min(fd->num_irqs - 1,
1842ee6373ddSDimitris Michailidis 				fd->kern_end_qid - 2) / nports;
1843ee6373ddSDimitris Michailidis 	if (ed->nsqs_per_port < 2) {
1844ee6373ddSDimitris Michailidis 		dev_err(fd->dev, "Too few SQs for %u ports", nports);
1845ee6373ddSDimitris Michailidis 		return -EINVAL;
1846ee6373ddSDimitris Michailidis 	}
1847ee6373ddSDimitris Michailidis 
1848ee6373ddSDimitris Michailidis 	ed->netdevs = kcalloc(nports, sizeof(*ed->netdevs), GFP_KERNEL);
1849ee6373ddSDimitris Michailidis 	if (!ed->netdevs)
1850ee6373ddSDimitris Michailidis 		return -ENOMEM;
1851ee6373ddSDimitris Michailidis 
1852ee6373ddSDimitris Michailidis 	ed->num_ports = nports;
1853ee6373ddSDimitris Michailidis 	for (i = 0; i < nports; i++) {
1854ee6373ddSDimitris Michailidis 		rc = fun_create_netdev(ed, i);
1855ee6373ddSDimitris Michailidis 		if (rc)
1856ee6373ddSDimitris Michailidis 			goto free_netdevs;
1857ee6373ddSDimitris Michailidis 	}
1858ee6373ddSDimitris Michailidis 
1859ee6373ddSDimitris Michailidis 	return 0;
1860ee6373ddSDimitris Michailidis 
1861ee6373ddSDimitris Michailidis free_netdevs:
1862ee6373ddSDimitris Michailidis 	while (i)
1863ee6373ddSDimitris Michailidis 		fun_destroy_netdev(ed->netdevs[--i]);
1864ee6373ddSDimitris Michailidis 	kfree(ed->netdevs);
1865ee6373ddSDimitris Michailidis 	ed->netdevs = NULL;
1866ee6373ddSDimitris Michailidis 	ed->num_ports = 0;
1867ee6373ddSDimitris Michailidis 	return rc;
1868ee6373ddSDimitris Michailidis }
1869ee6373ddSDimitris Michailidis 
fun_destroy_ports(struct fun_ethdev * ed)1870ee6373ddSDimitris Michailidis static void fun_destroy_ports(struct fun_ethdev *ed)
1871ee6373ddSDimitris Michailidis {
1872ee6373ddSDimitris Michailidis 	unsigned int i;
1873ee6373ddSDimitris Michailidis 
1874ee6373ddSDimitris Michailidis 	for (i = 0; i < ed->num_ports; i++)
1875ee6373ddSDimitris Michailidis 		fun_destroy_netdev(ed->netdevs[i]);
1876ee6373ddSDimitris Michailidis 
1877ee6373ddSDimitris Michailidis 	kfree(ed->netdevs);
1878ee6373ddSDimitris Michailidis 	ed->netdevs = NULL;
1879ee6373ddSDimitris Michailidis 	ed->num_ports = 0;
1880ee6373ddSDimitris Michailidis }
1881ee6373ddSDimitris Michailidis 
fun_update_link_state(const struct fun_ethdev * ed,const struct fun_admin_port_notif * notif)1882ee6373ddSDimitris Michailidis static void fun_update_link_state(const struct fun_ethdev *ed,
1883ee6373ddSDimitris Michailidis 				  const struct fun_admin_port_notif *notif)
1884ee6373ddSDimitris Michailidis {
1885ee6373ddSDimitris Michailidis 	unsigned int port_idx = be16_to_cpu(notif->id);
1886ee6373ddSDimitris Michailidis 	struct net_device *netdev;
1887ee6373ddSDimitris Michailidis 	struct funeth_priv *fp;
1888ee6373ddSDimitris Michailidis 
1889ee6373ddSDimitris Michailidis 	if (port_idx >= ed->num_ports)
1890ee6373ddSDimitris Michailidis 		return;
1891ee6373ddSDimitris Michailidis 
1892ee6373ddSDimitris Michailidis 	netdev = ed->netdevs[port_idx];
1893ee6373ddSDimitris Michailidis 	fp = netdev_priv(netdev);
1894ee6373ddSDimitris Michailidis 
1895ee6373ddSDimitris Michailidis 	write_seqcount_begin(&fp->link_seq);
1896ee6373ddSDimitris Michailidis 	fp->link_speed = be32_to_cpu(notif->speed) * 10;  /* 10 Mbps->Mbps */
1897ee6373ddSDimitris Michailidis 	fp->active_fc = notif->flow_ctrl;
1898ee6373ddSDimitris Michailidis 	fp->active_fec = notif->fec;
1899ee6373ddSDimitris Michailidis 	fp->xcvr_type = notif->xcvr_type;
1900ee6373ddSDimitris Michailidis 	fp->link_down_reason = notif->link_down_reason;
1901ee6373ddSDimitris Michailidis 	fp->lp_advertising = be64_to_cpu(notif->lp_advertising);
1902ee6373ddSDimitris Michailidis 
1903ee6373ddSDimitris Michailidis 	if ((notif->link_state | notif->missed_events) & FUN_PORT_FLAG_MAC_DOWN)
1904ee6373ddSDimitris Michailidis 		netif_carrier_off(netdev);
1905ee6373ddSDimitris Michailidis 	if (notif->link_state & FUN_PORT_FLAG_MAC_UP)
1906ee6373ddSDimitris Michailidis 		netif_carrier_on(netdev);
1907ee6373ddSDimitris Michailidis 
1908ee6373ddSDimitris Michailidis 	write_seqcount_end(&fp->link_seq);
1909ee6373ddSDimitris Michailidis 	fun_report_link(netdev);
1910ee6373ddSDimitris Michailidis }
1911ee6373ddSDimitris Michailidis 
1912ee6373ddSDimitris Michailidis /* handler for async events delivered through the admin CQ */
fun_event_cb(struct fun_dev * fdev,void * entry)1913ee6373ddSDimitris Michailidis static void fun_event_cb(struct fun_dev *fdev, void *entry)
1914ee6373ddSDimitris Michailidis {
1915ee6373ddSDimitris Michailidis 	u8 op = ((struct fun_admin_rsp_common *)entry)->op;
1916ee6373ddSDimitris Michailidis 
1917ee6373ddSDimitris Michailidis 	if (op == FUN_ADMIN_OP_PORT) {
1918ee6373ddSDimitris Michailidis 		const struct fun_admin_port_notif *rsp = entry;
1919ee6373ddSDimitris Michailidis 
1920ee6373ddSDimitris Michailidis 		if (rsp->subop == FUN_ADMIN_SUBOP_NOTIFY) {
1921ee6373ddSDimitris Michailidis 			fun_update_link_state(to_fun_ethdev(fdev), rsp);
1922ee6373ddSDimitris Michailidis 		} else if (rsp->subop == FUN_ADMIN_SUBOP_RES_COUNT) {
1923ee6373ddSDimitris Michailidis 			const struct fun_admin_res_count_rsp *r = entry;
1924ee6373ddSDimitris Michailidis 
1925ee6373ddSDimitris Michailidis 			if (r->count.data)
1926ee6373ddSDimitris Michailidis 				set_bit(FUN_SERV_RES_CHANGE, &fdev->service_flags);
1927ee6373ddSDimitris Michailidis 			else
1928ee6373ddSDimitris Michailidis 				set_bit(FUN_SERV_DEL_PORTS, &fdev->service_flags);
1929ee6373ddSDimitris Michailidis 			fun_serv_sched(fdev);
1930ee6373ddSDimitris Michailidis 		} else {
1931ee6373ddSDimitris Michailidis 			dev_info(fdev->dev, "adminq event unexpected op %u subop %u",
1932ee6373ddSDimitris Michailidis 				 op, rsp->subop);
1933ee6373ddSDimitris Michailidis 		}
1934ee6373ddSDimitris Michailidis 	} else {
1935ee6373ddSDimitris Michailidis 		dev_info(fdev->dev, "adminq event unexpected op %u", op);
1936ee6373ddSDimitris Michailidis 	}
1937ee6373ddSDimitris Michailidis }
1938ee6373ddSDimitris Michailidis 
1939ee6373ddSDimitris Michailidis /* handler for pending work managed by the service task */
fun_service_cb(struct fun_dev * fdev)1940ee6373ddSDimitris Michailidis static void fun_service_cb(struct fun_dev *fdev)
1941ee6373ddSDimitris Michailidis {
1942ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed = to_fun_ethdev(fdev);
1943ee6373ddSDimitris Michailidis 	int rc;
1944ee6373ddSDimitris Michailidis 
1945ee6373ddSDimitris Michailidis 	if (test_and_clear_bit(FUN_SERV_DEL_PORTS, &fdev->service_flags))
1946ee6373ddSDimitris Michailidis 		fun_destroy_ports(ed);
1947ee6373ddSDimitris Michailidis 
1948ee6373ddSDimitris Michailidis 	if (!test_and_clear_bit(FUN_SERV_RES_CHANGE, &fdev->service_flags))
1949ee6373ddSDimitris Michailidis 		return;
1950ee6373ddSDimitris Michailidis 
1951ee6373ddSDimitris Michailidis 	rc = fun_get_res_count(fdev, FUN_ADMIN_OP_PORT);
1952ee6373ddSDimitris Michailidis 	if (rc < 0 || rc == ed->num_ports)
1953ee6373ddSDimitris Michailidis 		return;
1954ee6373ddSDimitris Michailidis 
1955ee6373ddSDimitris Michailidis 	if (ed->num_ports)
1956ee6373ddSDimitris Michailidis 		fun_destroy_ports(ed);
1957ee6373ddSDimitris Michailidis 	if (rc)
1958ee6373ddSDimitris Michailidis 		fun_create_ports(ed, rc);
1959ee6373ddSDimitris Michailidis }
1960ee6373ddSDimitris Michailidis 
funeth_sriov_configure(struct pci_dev * pdev,int nvfs)1961ee6373ddSDimitris Michailidis static int funeth_sriov_configure(struct pci_dev *pdev, int nvfs)
1962ee6373ddSDimitris Michailidis {
1963ee6373ddSDimitris Michailidis 	struct fun_dev *fdev = pci_get_drvdata(pdev);
1964ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed = to_fun_ethdev(fdev);
1965ee6373ddSDimitris Michailidis 	int rc;
1966ee6373ddSDimitris Michailidis 
1967ee6373ddSDimitris Michailidis 	if (nvfs == 0) {
1968ee6373ddSDimitris Michailidis 		if (pci_vfs_assigned(pdev)) {
1969ee6373ddSDimitris Michailidis 			dev_warn(&pdev->dev,
1970ee6373ddSDimitris Michailidis 				 "Cannot disable SR-IOV while VFs are assigned\n");
1971ee6373ddSDimitris Michailidis 			return -EPERM;
1972ee6373ddSDimitris Michailidis 		}
1973ee6373ddSDimitris Michailidis 
1974ee6373ddSDimitris Michailidis 		mutex_lock(&ed->state_mutex);
1975ee6373ddSDimitris Michailidis 		fun_free_vports(ed);
1976ee6373ddSDimitris Michailidis 		mutex_unlock(&ed->state_mutex);
1977ee6373ddSDimitris Michailidis 		pci_disable_sriov(pdev);
1978ee6373ddSDimitris Michailidis 		return 0;
1979ee6373ddSDimitris Michailidis 	}
1980ee6373ddSDimitris Michailidis 
1981ee6373ddSDimitris Michailidis 	rc = pci_enable_sriov(pdev, nvfs);
1982ee6373ddSDimitris Michailidis 	if (rc)
1983ee6373ddSDimitris Michailidis 		return rc;
1984ee6373ddSDimitris Michailidis 
1985ee6373ddSDimitris Michailidis 	mutex_lock(&ed->state_mutex);
1986ee6373ddSDimitris Michailidis 	rc = fun_init_vports(ed, nvfs);
1987ee6373ddSDimitris Michailidis 	mutex_unlock(&ed->state_mutex);
1988ee6373ddSDimitris Michailidis 	if (rc) {
1989ee6373ddSDimitris Michailidis 		pci_disable_sriov(pdev);
1990ee6373ddSDimitris Michailidis 		return rc;
1991ee6373ddSDimitris Michailidis 	}
1992ee6373ddSDimitris Michailidis 
1993ee6373ddSDimitris Michailidis 	return nvfs;
1994ee6373ddSDimitris Michailidis }
1995ee6373ddSDimitris Michailidis 
funeth_probe(struct pci_dev * pdev,const struct pci_device_id * id)1996ee6373ddSDimitris Michailidis static int funeth_probe(struct pci_dev *pdev, const struct pci_device_id *id)
1997ee6373ddSDimitris Michailidis {
1998ee6373ddSDimitris Michailidis 	struct fun_dev_params aqreq = {
1999ee6373ddSDimitris Michailidis 		.cqe_size_log2 = ilog2(ADMIN_CQE_SIZE),
2000ee6373ddSDimitris Michailidis 		.sqe_size_log2 = ilog2(ADMIN_SQE_SIZE),
2001ee6373ddSDimitris Michailidis 		.cq_depth      = ADMIN_CQ_DEPTH,
2002ee6373ddSDimitris Michailidis 		.sq_depth      = ADMIN_SQ_DEPTH,
2003ee6373ddSDimitris Michailidis 		.rq_depth      = ADMIN_RQ_DEPTH,
2004ee6373ddSDimitris Michailidis 		.min_msix      = 2,              /* 1 Rx + 1 Tx */
2005ee6373ddSDimitris Michailidis 		.event_cb      = fun_event_cb,
2006ee6373ddSDimitris Michailidis 		.serv_cb       = fun_service_cb,
2007ee6373ddSDimitris Michailidis 	};
2008ee6373ddSDimitris Michailidis 	struct devlink *devlink;
2009ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed;
2010ee6373ddSDimitris Michailidis 	struct fun_dev *fdev;
2011ee6373ddSDimitris Michailidis 	int rc;
2012ee6373ddSDimitris Michailidis 
2013ee6373ddSDimitris Michailidis 	devlink = fun_devlink_alloc(&pdev->dev);
2014ee6373ddSDimitris Michailidis 	if (!devlink) {
2015ee6373ddSDimitris Michailidis 		dev_err(&pdev->dev, "devlink alloc failed\n");
2016ee6373ddSDimitris Michailidis 		return -ENOMEM;
2017ee6373ddSDimitris Michailidis 	}
2018ee6373ddSDimitris Michailidis 
2019ee6373ddSDimitris Michailidis 	ed = devlink_priv(devlink);
2020ee6373ddSDimitris Michailidis 	mutex_init(&ed->state_mutex);
2021ee6373ddSDimitris Michailidis 
2022ee6373ddSDimitris Michailidis 	fdev = &ed->fdev;
2023ee6373ddSDimitris Michailidis 	rc = fun_dev_enable(fdev, pdev, &aqreq, KBUILD_MODNAME);
2024ee6373ddSDimitris Michailidis 	if (rc)
2025ee6373ddSDimitris Michailidis 		goto free_devlink;
2026ee6373ddSDimitris Michailidis 
2027ee6373ddSDimitris Michailidis 	rc = fun_get_res_count(fdev, FUN_ADMIN_OP_PORT);
2028ee6373ddSDimitris Michailidis 	if (rc > 0)
2029ee6373ddSDimitris Michailidis 		rc = fun_create_ports(ed, rc);
2030ee6373ddSDimitris Michailidis 	if (rc < 0)
2031ee6373ddSDimitris Michailidis 		goto disable_dev;
2032ee6373ddSDimitris Michailidis 
2033ee6373ddSDimitris Michailidis 	fun_serv_restart(fdev);
2034ee6373ddSDimitris Michailidis 	fun_devlink_register(devlink);
2035ee6373ddSDimitris Michailidis 	return 0;
2036ee6373ddSDimitris Michailidis 
2037ee6373ddSDimitris Michailidis disable_dev:
2038ee6373ddSDimitris Michailidis 	fun_dev_disable(fdev);
2039ee6373ddSDimitris Michailidis free_devlink:
2040ee6373ddSDimitris Michailidis 	mutex_destroy(&ed->state_mutex);
2041ee6373ddSDimitris Michailidis 	fun_devlink_free(devlink);
2042ee6373ddSDimitris Michailidis 	return rc;
2043ee6373ddSDimitris Michailidis }
2044ee6373ddSDimitris Michailidis 
funeth_remove(struct pci_dev * pdev)2045ee6373ddSDimitris Michailidis static void funeth_remove(struct pci_dev *pdev)
2046ee6373ddSDimitris Michailidis {
2047ee6373ddSDimitris Michailidis 	struct fun_dev *fdev = pci_get_drvdata(pdev);
2048ee6373ddSDimitris Michailidis 	struct devlink *devlink;
2049ee6373ddSDimitris Michailidis 	struct fun_ethdev *ed;
2050ee6373ddSDimitris Michailidis 
2051ee6373ddSDimitris Michailidis 	ed = to_fun_ethdev(fdev);
2052ee6373ddSDimitris Michailidis 	devlink = priv_to_devlink(ed);
2053ee6373ddSDimitris Michailidis 	fun_devlink_unregister(devlink);
2054ee6373ddSDimitris Michailidis 
2055ee6373ddSDimitris Michailidis #ifdef CONFIG_PCI_IOV
2056ee6373ddSDimitris Michailidis 	funeth_sriov_configure(pdev, 0);
2057ee6373ddSDimitris Michailidis #endif
2058ee6373ddSDimitris Michailidis 
2059ee6373ddSDimitris Michailidis 	fun_serv_stop(fdev);
2060ee6373ddSDimitris Michailidis 	fun_destroy_ports(ed);
2061ee6373ddSDimitris Michailidis 	fun_dev_disable(fdev);
2062ee6373ddSDimitris Michailidis 	mutex_destroy(&ed->state_mutex);
2063ee6373ddSDimitris Michailidis 
2064ee6373ddSDimitris Michailidis 	fun_devlink_free(devlink);
2065ee6373ddSDimitris Michailidis }
2066ee6373ddSDimitris Michailidis 
2067ee6373ddSDimitris Michailidis static struct pci_driver funeth_driver = {
2068ee6373ddSDimitris Michailidis 	.name		 = KBUILD_MODNAME,
2069ee6373ddSDimitris Michailidis 	.id_table	 = funeth_id_table,
2070ee6373ddSDimitris Michailidis 	.probe		 = funeth_probe,
2071ee6373ddSDimitris Michailidis 	.remove		 = funeth_remove,
2072ee6373ddSDimitris Michailidis 	.shutdown	 = funeth_remove,
2073ee6373ddSDimitris Michailidis 	.sriov_configure = funeth_sriov_configure,
2074ee6373ddSDimitris Michailidis };
2075ee6373ddSDimitris Michailidis 
2076ee6373ddSDimitris Michailidis module_pci_driver(funeth_driver);
2077ee6373ddSDimitris Michailidis 
2078ee6373ddSDimitris Michailidis MODULE_AUTHOR("Dimitris Michailidis <dmichail@fungible.com>");
2079ee6373ddSDimitris Michailidis MODULE_DESCRIPTION("Fungible Ethernet Network Driver");
2080ee6373ddSDimitris Michailidis MODULE_LICENSE("Dual BSD/GPL");
2081ee6373ddSDimitris Michailidis MODULE_DEVICE_TABLE(pci, funeth_id_table);
2082