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