xref: /openbmc/linux/drivers/infiniband/hw/hfi1/ipoib_main.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
1438d7ddaSGary Leshner // SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
2438d7ddaSGary Leshner /*
3438d7ddaSGary Leshner  * Copyright(c) 2020 Intel Corporation.
4438d7ddaSGary Leshner  *
5438d7ddaSGary Leshner  */
6438d7ddaSGary Leshner 
7438d7ddaSGary Leshner /*
8438d7ddaSGary Leshner  * This file contains HFI1 support for ipoib functionality
9438d7ddaSGary Leshner  */
10438d7ddaSGary Leshner 
11438d7ddaSGary Leshner #include "ipoib.h"
12438d7ddaSGary Leshner #include "hfi.h"
13438d7ddaSGary Leshner 
qpn_from_mac(const u8 * mac_arr)14fd92213eSJakub Kicinski static u32 qpn_from_mac(const u8 *mac_arr)
15438d7ddaSGary Leshner {
16438d7ddaSGary Leshner 	return (u32)mac_arr[1] << 16 | mac_arr[2] << 8 | mac_arr[3];
17438d7ddaSGary Leshner }
18438d7ddaSGary Leshner 
hfi1_ipoib_dev_init(struct net_device * dev)19438d7ddaSGary Leshner static int hfi1_ipoib_dev_init(struct net_device *dev)
20438d7ddaSGary Leshner {
21438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
22370caa5bSGrzegorz Andrejczuk 	int ret;
23438d7ddaSGary Leshner 
24aa0616a9SHeiner Kallweit 	dev->tstats = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats);
25*e5cce44aSMike Marciniszyn 	if (!dev->tstats)
26*e5cce44aSMike Marciniszyn 		return -ENOMEM;
27438d7ddaSGary Leshner 
28370caa5bSGrzegorz Andrejczuk 	ret = priv->netdev_ops->ndo_init(dev);
29370caa5bSGrzegorz Andrejczuk 	if (ret)
30*e5cce44aSMike Marciniszyn 		goto out_ret;
31370caa5bSGrzegorz Andrejczuk 
32370caa5bSGrzegorz Andrejczuk 	ret = hfi1_netdev_add_data(priv->dd,
33370caa5bSGrzegorz Andrejczuk 				   qpn_from_mac(priv->netdev->dev_addr),
34370caa5bSGrzegorz Andrejczuk 				   dev);
35370caa5bSGrzegorz Andrejczuk 	if (ret < 0) {
36370caa5bSGrzegorz Andrejczuk 		priv->netdev_ops->ndo_uninit(dev);
37*e5cce44aSMike Marciniszyn 		goto out_ret;
38370caa5bSGrzegorz Andrejczuk 	}
39370caa5bSGrzegorz Andrejczuk 
40370caa5bSGrzegorz Andrejczuk 	return 0;
41*e5cce44aSMike Marciniszyn out_ret:
42*e5cce44aSMike Marciniszyn 	free_percpu(dev->tstats);
43*e5cce44aSMike Marciniszyn 	dev->tstats = NULL;
44*e5cce44aSMike Marciniszyn 	return ret;
45438d7ddaSGary Leshner }
46438d7ddaSGary Leshner 
hfi1_ipoib_dev_uninit(struct net_device * dev)47438d7ddaSGary Leshner static void hfi1_ipoib_dev_uninit(struct net_device *dev)
48438d7ddaSGary Leshner {
49438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
50438d7ddaSGary Leshner 
51*e5cce44aSMike Marciniszyn 	free_percpu(dev->tstats);
52*e5cce44aSMike Marciniszyn 	dev->tstats = NULL;
53*e5cce44aSMike Marciniszyn 
54370caa5bSGrzegorz Andrejczuk 	hfi1_netdev_remove_data(priv->dd, qpn_from_mac(priv->netdev->dev_addr));
55370caa5bSGrzegorz Andrejczuk 
56438d7ddaSGary Leshner 	priv->netdev_ops->ndo_uninit(dev);
57438d7ddaSGary Leshner }
58438d7ddaSGary Leshner 
hfi1_ipoib_dev_open(struct net_device * dev)59438d7ddaSGary Leshner static int hfi1_ipoib_dev_open(struct net_device *dev)
60438d7ddaSGary Leshner {
61438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
62438d7ddaSGary Leshner 	int ret;
63438d7ddaSGary Leshner 
64438d7ddaSGary Leshner 	ret = priv->netdev_ops->ndo_open(dev);
65438d7ddaSGary Leshner 	if (!ret) {
66438d7ddaSGary Leshner 		struct hfi1_ibport *ibp = to_iport(priv->device,
67438d7ddaSGary Leshner 						   priv->port_num);
68438d7ddaSGary Leshner 		struct rvt_qp *qp;
69438d7ddaSGary Leshner 		u32 qpn = qpn_from_mac(priv->netdev->dev_addr);
70438d7ddaSGary Leshner 
71438d7ddaSGary Leshner 		rcu_read_lock();
72438d7ddaSGary Leshner 		qp = rvt_lookup_qpn(ib_to_rvt(priv->device), &ibp->rvp, qpn);
73438d7ddaSGary Leshner 		if (!qp) {
74438d7ddaSGary Leshner 			rcu_read_unlock();
75438d7ddaSGary Leshner 			priv->netdev_ops->ndo_stop(dev);
76438d7ddaSGary Leshner 			return -EINVAL;
77438d7ddaSGary Leshner 		}
78438d7ddaSGary Leshner 		rvt_get_qp(qp);
79438d7ddaSGary Leshner 		priv->qp = qp;
80438d7ddaSGary Leshner 		rcu_read_unlock();
81438d7ddaSGary Leshner 
82370caa5bSGrzegorz Andrejczuk 		hfi1_netdev_enable_queues(priv->dd);
83438d7ddaSGary Leshner 		hfi1_ipoib_napi_tx_enable(dev);
84438d7ddaSGary Leshner 	}
85438d7ddaSGary Leshner 
86438d7ddaSGary Leshner 	return ret;
87438d7ddaSGary Leshner }
88438d7ddaSGary Leshner 
hfi1_ipoib_dev_stop(struct net_device * dev)89438d7ddaSGary Leshner static int hfi1_ipoib_dev_stop(struct net_device *dev)
90438d7ddaSGary Leshner {
91438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
92438d7ddaSGary Leshner 
93438d7ddaSGary Leshner 	if (!priv->qp)
94438d7ddaSGary Leshner 		return 0;
95438d7ddaSGary Leshner 
96438d7ddaSGary Leshner 	hfi1_ipoib_napi_tx_disable(dev);
97370caa5bSGrzegorz Andrejczuk 	hfi1_netdev_disable_queues(priv->dd);
98438d7ddaSGary Leshner 
99438d7ddaSGary Leshner 	rvt_put_qp(priv->qp);
100438d7ddaSGary Leshner 	priv->qp = NULL;
101438d7ddaSGary Leshner 
102438d7ddaSGary Leshner 	return priv->netdev_ops->ndo_stop(dev);
103438d7ddaSGary Leshner }
104438d7ddaSGary Leshner 
105438d7ddaSGary Leshner static const struct net_device_ops hfi1_ipoib_netdev_ops = {
106438d7ddaSGary Leshner 	.ndo_init         = hfi1_ipoib_dev_init,
107438d7ddaSGary Leshner 	.ndo_uninit       = hfi1_ipoib_dev_uninit,
108438d7ddaSGary Leshner 	.ndo_open         = hfi1_ipoib_dev_open,
109438d7ddaSGary Leshner 	.ndo_stop         = hfi1_ipoib_dev_stop,
110aa0616a9SHeiner Kallweit 	.ndo_get_stats64  = dev_get_tstats64,
111438d7ddaSGary Leshner };
112438d7ddaSGary Leshner 
hfi1_ipoib_mcast_attach(struct net_device * dev,struct ib_device * device,union ib_gid * mgid,u16 mlid,int set_qkey,u32 qkey)113438d7ddaSGary Leshner static int hfi1_ipoib_mcast_attach(struct net_device *dev,
114438d7ddaSGary Leshner 				   struct ib_device *device,
115438d7ddaSGary Leshner 				   union ib_gid *mgid,
116438d7ddaSGary Leshner 				   u16 mlid,
117438d7ddaSGary Leshner 				   int set_qkey,
118438d7ddaSGary Leshner 				   u32 qkey)
119438d7ddaSGary Leshner {
120438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
121438d7ddaSGary Leshner 	u32 qpn = (u32)qpn_from_mac(priv->netdev->dev_addr);
122438d7ddaSGary Leshner 	struct hfi1_ibport *ibp = to_iport(priv->device, priv->port_num);
123438d7ddaSGary Leshner 	struct rvt_qp *qp;
124438d7ddaSGary Leshner 	int ret = -EINVAL;
125438d7ddaSGary Leshner 
126438d7ddaSGary Leshner 	rcu_read_lock();
127438d7ddaSGary Leshner 
128438d7ddaSGary Leshner 	qp = rvt_lookup_qpn(ib_to_rvt(priv->device), &ibp->rvp, qpn);
129438d7ddaSGary Leshner 	if (qp) {
130438d7ddaSGary Leshner 		rvt_get_qp(qp);
131438d7ddaSGary Leshner 		rcu_read_unlock();
132438d7ddaSGary Leshner 		if (set_qkey)
133438d7ddaSGary Leshner 			priv->qkey = qkey;
134438d7ddaSGary Leshner 
135438d7ddaSGary Leshner 		/* attach QP to multicast group */
136438d7ddaSGary Leshner 		ret = ib_attach_mcast(&qp->ibqp, mgid, mlid);
137438d7ddaSGary Leshner 		rvt_put_qp(qp);
138438d7ddaSGary Leshner 	} else {
139438d7ddaSGary Leshner 		rcu_read_unlock();
140438d7ddaSGary Leshner 	}
141438d7ddaSGary Leshner 
142438d7ddaSGary Leshner 	return ret;
143438d7ddaSGary Leshner }
144438d7ddaSGary Leshner 
hfi1_ipoib_mcast_detach(struct net_device * dev,struct ib_device * device,union ib_gid * mgid,u16 mlid)145438d7ddaSGary Leshner static int hfi1_ipoib_mcast_detach(struct net_device *dev,
146438d7ddaSGary Leshner 				   struct ib_device *device,
147438d7ddaSGary Leshner 				   union ib_gid *mgid,
148438d7ddaSGary Leshner 				   u16 mlid)
149438d7ddaSGary Leshner {
150438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
151438d7ddaSGary Leshner 	u32 qpn = (u32)qpn_from_mac(priv->netdev->dev_addr);
152438d7ddaSGary Leshner 	struct hfi1_ibport *ibp = to_iport(priv->device, priv->port_num);
153438d7ddaSGary Leshner 	struct rvt_qp *qp;
154438d7ddaSGary Leshner 	int ret = -EINVAL;
155438d7ddaSGary Leshner 
156438d7ddaSGary Leshner 	rcu_read_lock();
157438d7ddaSGary Leshner 
158438d7ddaSGary Leshner 	qp = rvt_lookup_qpn(ib_to_rvt(priv->device), &ibp->rvp, qpn);
159438d7ddaSGary Leshner 	if (qp) {
160438d7ddaSGary Leshner 		rvt_get_qp(qp);
161438d7ddaSGary Leshner 		rcu_read_unlock();
162438d7ddaSGary Leshner 		ret = ib_detach_mcast(&qp->ibqp, mgid, mlid);
163438d7ddaSGary Leshner 		rvt_put_qp(qp);
164438d7ddaSGary Leshner 	} else {
165438d7ddaSGary Leshner 		rcu_read_unlock();
166438d7ddaSGary Leshner 	}
167438d7ddaSGary Leshner 	return ret;
168438d7ddaSGary Leshner }
169438d7ddaSGary Leshner 
hfi1_ipoib_netdev_dtor(struct net_device * dev)170438d7ddaSGary Leshner static void hfi1_ipoib_netdev_dtor(struct net_device *dev)
171438d7ddaSGary Leshner {
172438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
173438d7ddaSGary Leshner 
174438d7ddaSGary Leshner 	hfi1_ipoib_txreq_deinit(priv);
175370caa5bSGrzegorz Andrejczuk 	hfi1_ipoib_rxq_deinit(priv->netdev);
176438d7ddaSGary Leshner 
177aa0616a9SHeiner Kallweit 	free_percpu(dev->tstats);
178*e5cce44aSMike Marciniszyn 	dev->tstats = NULL;
179438d7ddaSGary Leshner }
180438d7ddaSGary Leshner 
hfi1_ipoib_set_id(struct net_device * dev,int id)181438d7ddaSGary Leshner static void hfi1_ipoib_set_id(struct net_device *dev, int id)
182438d7ddaSGary Leshner {
183438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv = hfi1_ipoib_priv(dev);
184438d7ddaSGary Leshner 
185438d7ddaSGary Leshner 	priv->pkey_index = (u16)id;
186438d7ddaSGary Leshner 	ib_query_pkey(priv->device,
187438d7ddaSGary Leshner 		      priv->port_num,
188438d7ddaSGary Leshner 		      priv->pkey_index,
189438d7ddaSGary Leshner 		      &priv->pkey);
190438d7ddaSGary Leshner }
191438d7ddaSGary Leshner 
hfi1_ipoib_setup_rn(struct ib_device * device,u32 port_num,struct net_device * netdev,void * param)192438d7ddaSGary Leshner static int hfi1_ipoib_setup_rn(struct ib_device *device,
1931fb7f897SMark Bloch 			       u32 port_num,
194438d7ddaSGary Leshner 			       struct net_device *netdev,
195438d7ddaSGary Leshner 			       void *param)
196438d7ddaSGary Leshner {
197438d7ddaSGary Leshner 	struct hfi1_devdata *dd = dd_from_ibdev(device);
198438d7ddaSGary Leshner 	struct rdma_netdev *rn = netdev_priv(netdev);
199438d7ddaSGary Leshner 	struct hfi1_ipoib_dev_priv *priv;
200438d7ddaSGary Leshner 	int rc;
201438d7ddaSGary Leshner 
202438d7ddaSGary Leshner 	rn->send = hfi1_ipoib_send;
203042a00f9SMike Marciniszyn 	rn->tx_timeout = hfi1_ipoib_tx_timeout;
204438d7ddaSGary Leshner 	rn->attach_mcast = hfi1_ipoib_mcast_attach;
205438d7ddaSGary Leshner 	rn->detach_mcast = hfi1_ipoib_mcast_detach;
206438d7ddaSGary Leshner 	rn->set_id = hfi1_ipoib_set_id;
207438d7ddaSGary Leshner 	rn->hca = device;
208438d7ddaSGary Leshner 	rn->port_num = port_num;
209438d7ddaSGary Leshner 	rn->mtu = netdev->mtu;
210438d7ddaSGary Leshner 
211438d7ddaSGary Leshner 	priv = hfi1_ipoib_priv(netdev);
212438d7ddaSGary Leshner 	priv->dd = dd;
213438d7ddaSGary Leshner 	priv->netdev = netdev;
214438d7ddaSGary Leshner 	priv->device = device;
215438d7ddaSGary Leshner 	priv->port_num = port_num;
216438d7ddaSGary Leshner 	priv->netdev_ops = netdev->netdev_ops;
217438d7ddaSGary Leshner 
218438d7ddaSGary Leshner 	ib_query_pkey(device, port_num, priv->pkey_index, &priv->pkey);
219438d7ddaSGary Leshner 
220438d7ddaSGary Leshner 	rc = hfi1_ipoib_txreq_init(priv);
221438d7ddaSGary Leshner 	if (rc) {
222438d7ddaSGary Leshner 		dd_dev_err(dd, "IPoIB netdev TX init - failed(%d)\n", rc);
223438d7ddaSGary Leshner 		return rc;
224438d7ddaSGary Leshner 	}
225438d7ddaSGary Leshner 
226370caa5bSGrzegorz Andrejczuk 	rc = hfi1_ipoib_rxq_init(netdev);
227370caa5bSGrzegorz Andrejczuk 	if (rc) {
228370caa5bSGrzegorz Andrejczuk 		dd_dev_err(dd, "IPoIB netdev RX init - failed(%d)\n", rc);
2295f8f55b9SMike Marciniszyn 		hfi1_ipoib_txreq_deinit(priv);
230370caa5bSGrzegorz Andrejczuk 		return rc;
231370caa5bSGrzegorz Andrejczuk 	}
232370caa5bSGrzegorz Andrejczuk 
2335f8f55b9SMike Marciniszyn 	netdev->netdev_ops = &hfi1_ipoib_netdev_ops;
2345f8f55b9SMike Marciniszyn 
235438d7ddaSGary Leshner 	netdev->priv_destructor = hfi1_ipoib_netdev_dtor;
236438d7ddaSGary Leshner 	netdev->needs_free_netdev = true;
237438d7ddaSGary Leshner 
238438d7ddaSGary Leshner 	return 0;
239438d7ddaSGary Leshner }
240438d7ddaSGary Leshner 
hfi1_ipoib_rn_get_params(struct ib_device * device,u32 port_num,enum rdma_netdev_t type,struct rdma_netdev_alloc_params * params)241438d7ddaSGary Leshner int hfi1_ipoib_rn_get_params(struct ib_device *device,
2421fb7f897SMark Bloch 			     u32 port_num,
243438d7ddaSGary Leshner 			     enum rdma_netdev_t type,
244438d7ddaSGary Leshner 			     struct rdma_netdev_alloc_params *params)
245438d7ddaSGary Leshner {
246438d7ddaSGary Leshner 	struct hfi1_devdata *dd = dd_from_ibdev(device);
247438d7ddaSGary Leshner 
248438d7ddaSGary Leshner 	if (type != RDMA_NETDEV_IPOIB)
249438d7ddaSGary Leshner 		return -EOPNOTSUPP;
250438d7ddaSGary Leshner 
251370caa5bSGrzegorz Andrejczuk 	if (!HFI1_CAP_IS_KSET(AIP) || !dd->num_netdev_contexts)
252438d7ddaSGary Leshner 		return -EOPNOTSUPP;
253438d7ddaSGary Leshner 
254438d7ddaSGary Leshner 	if (!port_num || port_num > dd->num_pports)
255438d7ddaSGary Leshner 		return -EINVAL;
256438d7ddaSGary Leshner 
257438d7ddaSGary Leshner 	params->sizeof_priv = sizeof(struct hfi1_ipoib_rdma_netdev);
258438d7ddaSGary Leshner 	params->txqs = dd->num_sdma;
259370caa5bSGrzegorz Andrejczuk 	params->rxqs = dd->num_netdev_contexts;
260438d7ddaSGary Leshner 	params->param = NULL;
261438d7ddaSGary Leshner 	params->initialize_rdma_netdev = hfi1_ipoib_setup_rn;
262438d7ddaSGary Leshner 
263438d7ddaSGary Leshner 	return 0;
264438d7ddaSGary Leshner }
265