blob: 44b1104eb168e50b7d598a2c5f756c5681d1e416 [file] [log] [blame]
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001/*
2 * Copyright (c) 2005 Topspin Communications. All rights reserved.
Roland Dreier33b9b3e2006-01-30 14:29:21 -08003 * Copyright (c) 2005, 2006 Cisco Systems. All rights reserved.
Roland Dreier2a1d9b72005-08-10 23:03:10 -07004 * Copyright (c) 2005 Mellanox Technologies. All rights reserved.
5 * Copyright (c) 2005 Voltaire, Inc. All rights reserved.
Roland Dreier67cdb402005-10-14 15:26:04 -07006 * Copyright (c) 2005 PathScale, Inc. All rights reserved.
Roland Dreierbc38a6a2005-07-07 17:57:13 -07007 *
8 * This software is available to you under a choice of one of two
9 * licenses. You may choose to be licensed under the terms of the GNU
10 * General Public License (GPL) Version 2, available from the file
11 * COPYING in the main directory of this source tree, or the
12 * OpenIB.org BSD license below:
13 *
14 * Redistribution and use in source and binary forms, with or
15 * without modification, are permitted provided that the following
16 * conditions are met:
17 *
18 * - Redistributions of source code must retain the above
19 * copyright notice, this list of conditions and the following
20 * disclaimer.
21 *
22 * - Redistributions in binary form must reproduce the above
23 * copyright notice, this list of conditions and the following
24 * disclaimer in the documentation and/or other materials
25 * provided with the distribution.
26 *
27 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
28 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
29 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
30 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
31 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
32 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
33 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
34 * SOFTWARE.
Roland Dreierbc38a6a2005-07-07 17:57:13 -070035 */
36
37#include <linux/module.h>
38#include <linux/init.h>
39#include <linux/device.h>
40#include <linux/err.h>
41#include <linux/fs.h>
42#include <linux/poll.h>
Alexey Dobriyana99bbaf2009-10-04 16:11:37 +040043#include <linux/sched.h>
Roland Dreierbc38a6a2005-07-07 17:57:13 -070044#include <linux/file.h>
Roland Dreier70a30e12005-10-28 15:38:26 -070045#include <linux/cdev.h>
Roland Dreiera265e552010-02-24 16:51:20 -080046#include <linux/anon_inodes.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090047#include <linux/slab.h>
Roland Dreierbc38a6a2005-07-07 17:57:13 -070048
49#include <asm/uaccess.h>
50
Jason Gunthorpee6bd18f2016-04-10 19:13:13 -060051#include <rdma/ib.h>
52
Roland Dreierbc38a6a2005-07-07 17:57:13 -070053#include "uverbs.h"
54
55MODULE_AUTHOR("Roland Dreier");
56MODULE_DESCRIPTION("InfiniBand userspace verbs access");
57MODULE_LICENSE("Dual BSD/GPL");
58
Roland Dreierbc38a6a2005-07-07 17:57:13 -070059enum {
60 IB_UVERBS_MAJOR = 231,
61 IB_UVERBS_BASE_MINOR = 192,
62 IB_UVERBS_MAX_DEVICES = 32
63};
64
65#define IB_UVERBS_BASE_DEV MKDEV(IB_UVERBS_MAJOR, IB_UVERBS_BASE_MINOR)
66
Roland Dreier70a30e12005-10-28 15:38:26 -070067static struct class *uverbs_class;
68
Roland Dreier9ead1902006-06-17 20:44:49 -070069DEFINE_SPINLOCK(ib_uverbs_idr_lock);
Roland Dreierbc38a6a2005-07-07 17:57:13 -070070DEFINE_IDR(ib_uverbs_pd_idr);
71DEFINE_IDR(ib_uverbs_mr_idr);
72DEFINE_IDR(ib_uverbs_mw_idr);
73DEFINE_IDR(ib_uverbs_ah_idr);
74DEFINE_IDR(ib_uverbs_cq_idr);
75DEFINE_IDR(ib_uverbs_qp_idr);
Roland Dreierf520ba52005-08-18 12:24:13 -070076DEFINE_IDR(ib_uverbs_srq_idr);
Sean Hefty53d0bd12011-05-24 08:33:46 -070077DEFINE_IDR(ib_uverbs_xrcd_idr);
Hadar Hen Zion436f2ad2013-08-14 13:58:30 +030078DEFINE_IDR(ib_uverbs_rule_idr);
Yishai Hadasf213c052016-05-23 15:20:49 +030079DEFINE_IDR(ib_uverbs_wq_idr);
Yishai Hadasde019a92016-05-23 15:20:52 +030080DEFINE_IDR(ib_uverbs_rwq_ind_tbl_idr);
Roland Dreierbc38a6a2005-07-07 17:57:13 -070081
Roland Dreier6276e082009-09-05 20:24:23 -070082static DEFINE_SPINLOCK(map_lock);
Roland Dreierbc38a6a2005-07-07 17:57:13 -070083static DECLARE_BITMAP(dev_map, IB_UVERBS_MAX_DEVICES);
84
85static ssize_t (*uverbs_cmd_table[])(struct ib_uverbs_file *file,
Yishai Hadas057aec02015-08-13 18:32:04 +030086 struct ib_device *ib_dev,
Roland Dreierbc38a6a2005-07-07 17:57:13 -070087 const char __user *buf, int in_len,
88 int out_len) = {
Alexander Chiang9afed762010-02-02 19:08:19 +000089 [IB_USER_VERBS_CMD_GET_CONTEXT] = ib_uverbs_get_context,
90 [IB_USER_VERBS_CMD_QUERY_DEVICE] = ib_uverbs_query_device,
91 [IB_USER_VERBS_CMD_QUERY_PORT] = ib_uverbs_query_port,
92 [IB_USER_VERBS_CMD_ALLOC_PD] = ib_uverbs_alloc_pd,
93 [IB_USER_VERBS_CMD_DEALLOC_PD] = ib_uverbs_dealloc_pd,
94 [IB_USER_VERBS_CMD_REG_MR] = ib_uverbs_reg_mr,
Matan Barak7e6edb92014-07-31 11:01:28 +030095 [IB_USER_VERBS_CMD_REREG_MR] = ib_uverbs_rereg_mr,
Alexander Chiang9afed762010-02-02 19:08:19 +000096 [IB_USER_VERBS_CMD_DEREG_MR] = ib_uverbs_dereg_mr,
Shani Michaeli6b52a122013-02-06 16:19:13 +000097 [IB_USER_VERBS_CMD_ALLOC_MW] = ib_uverbs_alloc_mw,
98 [IB_USER_VERBS_CMD_DEALLOC_MW] = ib_uverbs_dealloc_mw,
Roland Dreier6b73597e2005-09-26 13:53:25 -070099 [IB_USER_VERBS_CMD_CREATE_COMP_CHANNEL] = ib_uverbs_create_comp_channel,
Alexander Chiang9afed762010-02-02 19:08:19 +0000100 [IB_USER_VERBS_CMD_CREATE_CQ] = ib_uverbs_create_cq,
101 [IB_USER_VERBS_CMD_RESIZE_CQ] = ib_uverbs_resize_cq,
102 [IB_USER_VERBS_CMD_POLL_CQ] = ib_uverbs_poll_cq,
103 [IB_USER_VERBS_CMD_REQ_NOTIFY_CQ] = ib_uverbs_req_notify_cq,
104 [IB_USER_VERBS_CMD_DESTROY_CQ] = ib_uverbs_destroy_cq,
105 [IB_USER_VERBS_CMD_CREATE_QP] = ib_uverbs_create_qp,
106 [IB_USER_VERBS_CMD_QUERY_QP] = ib_uverbs_query_qp,
107 [IB_USER_VERBS_CMD_MODIFY_QP] = ib_uverbs_modify_qp,
108 [IB_USER_VERBS_CMD_DESTROY_QP] = ib_uverbs_destroy_qp,
109 [IB_USER_VERBS_CMD_POST_SEND] = ib_uverbs_post_send,
110 [IB_USER_VERBS_CMD_POST_RECV] = ib_uverbs_post_recv,
111 [IB_USER_VERBS_CMD_POST_SRQ_RECV] = ib_uverbs_post_srq_recv,
112 [IB_USER_VERBS_CMD_CREATE_AH] = ib_uverbs_create_ah,
113 [IB_USER_VERBS_CMD_DESTROY_AH] = ib_uverbs_destroy_ah,
114 [IB_USER_VERBS_CMD_ATTACH_MCAST] = ib_uverbs_attach_mcast,
115 [IB_USER_VERBS_CMD_DETACH_MCAST] = ib_uverbs_detach_mcast,
116 [IB_USER_VERBS_CMD_CREATE_SRQ] = ib_uverbs_create_srq,
117 [IB_USER_VERBS_CMD_MODIFY_SRQ] = ib_uverbs_modify_srq,
118 [IB_USER_VERBS_CMD_QUERY_SRQ] = ib_uverbs_query_srq,
119 [IB_USER_VERBS_CMD_DESTROY_SRQ] = ib_uverbs_destroy_srq,
Sean Hefty53d0bd12011-05-24 08:33:46 -0700120 [IB_USER_VERBS_CMD_OPEN_XRCD] = ib_uverbs_open_xrcd,
121 [IB_USER_VERBS_CMD_CLOSE_XRCD] = ib_uverbs_close_xrcd,
Sean Hefty42849b22011-08-11 13:57:43 -0700122 [IB_USER_VERBS_CMD_CREATE_XSRQ] = ib_uverbs_create_xsrq,
Hadar Hen Zion436f2ad2013-08-14 13:58:30 +0300123 [IB_USER_VERBS_CMD_OPEN_QP] = ib_uverbs_open_qp,
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700124};
125
Yann Droneaudf21519b2013-11-06 23:21:49 +0100126static int (*uverbs_ex_cmd_table[])(struct ib_uverbs_file *file,
Yishai Hadas057aec02015-08-13 18:32:04 +0300127 struct ib_device *ib_dev,
Yann Droneaudf21519b2013-11-06 23:21:49 +0100128 struct ib_udata *ucore,
129 struct ib_udata *uhw) = {
130 [IB_USER_VERBS_EX_CMD_CREATE_FLOW] = ib_uverbs_ex_create_flow,
Eli Cohen5a77abf2014-12-11 17:04:15 +0200131 [IB_USER_VERBS_EX_CMD_DESTROY_FLOW] = ib_uverbs_ex_destroy_flow,
Eli Cohen02d1aa72015-02-08 13:28:50 +0200132 [IB_USER_VERBS_EX_CMD_QUERY_DEVICE] = ib_uverbs_ex_query_device,
Matan Barak565197d2015-06-11 16:35:23 +0300133 [IB_USER_VERBS_EX_CMD_CREATE_CQ] = ib_uverbs_ex_create_cq,
Eran Ben Elisha6d8a7492015-10-21 17:00:42 +0300134 [IB_USER_VERBS_EX_CMD_CREATE_QP] = ib_uverbs_ex_create_qp,
Yishai Hadasf213c052016-05-23 15:20:49 +0300135 [IB_USER_VERBS_EX_CMD_CREATE_WQ] = ib_uverbs_ex_create_wq,
136 [IB_USER_VERBS_EX_CMD_MODIFY_WQ] = ib_uverbs_ex_modify_wq,
137 [IB_USER_VERBS_EX_CMD_DESTROY_WQ] = ib_uverbs_ex_destroy_wq,
Yishai Hadasde019a92016-05-23 15:20:52 +0300138 [IB_USER_VERBS_EX_CMD_CREATE_RWQ_IND_TBL] = ib_uverbs_ex_create_rwq_ind_table,
139 [IB_USER_VERBS_EX_CMD_DESTROY_RWQ_IND_TBL] = ib_uverbs_ex_destroy_rwq_ind_table,
Yann Droneaudf21519b2013-11-06 23:21:49 +0100140};
Yann Droneaudf21519b2013-11-06 23:21:49 +0100141
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700142static void ib_uverbs_add_one(struct ib_device *device);
Haggai Eran7c1eb452015-07-30 17:50:14 +0300143static void ib_uverbs_remove_one(struct ib_device *device, void *client_data);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700144
Christoph Hellwigfeb7c1e2015-12-23 19:12:48 +0100145int uverbs_dealloc_mw(struct ib_mw *mw)
146{
147 struct ib_pd *pd = mw->pd;
148 int ret;
149
150 ret = mw->device->dealloc_mw(mw);
151 if (!ret)
152 atomic_dec(&pd->usecnt);
153 return ret;
154}
155
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300156static void ib_uverbs_release_dev(struct kobject *kobj)
Roland Dreier70a30e12005-10-28 15:38:26 -0700157{
158 struct ib_uverbs_device *dev =
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300159 container_of(kobj, struct ib_uverbs_device, kobj);
Roland Dreier70a30e12005-10-28 15:38:26 -0700160
Yishai Hadas036b1062015-08-13 18:32:05 +0300161 cleanup_srcu_struct(&dev->disassociate_srcu);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300162 kfree(dev);
Roland Dreier70a30e12005-10-28 15:38:26 -0700163}
164
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300165static struct kobj_type ib_uverbs_dev_ktype = {
166 .release = ib_uverbs_release_dev,
167};
168
Roland Dreier04d29b02007-10-09 19:59:15 -0700169static void ib_uverbs_release_event_file(struct kref *ref)
170{
171 struct ib_uverbs_event_file *file =
172 container_of(ref, struct ib_uverbs_event_file, ref);
173
174 kfree(file);
175}
176
Roland Dreier70a30e12005-10-28 15:38:26 -0700177void ib_uverbs_release_ucq(struct ib_uverbs_file *file,
178 struct ib_uverbs_event_file *ev_file,
179 struct ib_ucq_object *uobj)
180{
181 struct ib_uverbs_event *evt, *tmp;
182
183 if (ev_file) {
184 spin_lock_irq(&ev_file->lock);
185 list_for_each_entry_safe(evt, tmp, &uobj->comp_list, obj_list) {
186 list_del(&evt->list);
187 kfree(evt);
188 }
189 spin_unlock_irq(&ev_file->lock);
190
191 kref_put(&ev_file->ref, ib_uverbs_release_event_file);
192 }
193
194 spin_lock_irq(&file->async_file->lock);
195 list_for_each_entry_safe(evt, tmp, &uobj->async_list, obj_list) {
196 list_del(&evt->list);
197 kfree(evt);
198 }
199 spin_unlock_irq(&file->async_file->lock);
200}
201
202void ib_uverbs_release_uevent(struct ib_uverbs_file *file,
203 struct ib_uevent_object *uobj)
204{
205 struct ib_uverbs_event *evt, *tmp;
206
207 spin_lock_irq(&file->async_file->lock);
208 list_for_each_entry_safe(evt, tmp, &uobj->event_list, obj_list) {
209 list_del(&evt->list);
210 kfree(evt);
211 }
212 spin_unlock_irq(&file->async_file->lock);
213}
214
Jack Morgensteinf4e40152005-11-29 16:57:01 -0800215static void ib_uverbs_detach_umcast(struct ib_qp *qp,
216 struct ib_uqp_object *uobj)
217{
218 struct ib_uverbs_mcast_entry *mcast, *tmp;
219
220 list_for_each_entry_safe(mcast, tmp, &uobj->mcast_list, list) {
221 ib_detach_mcast(qp, &mcast->gid, mcast->lid);
222 list_del(&mcast->list);
223 kfree(mcast);
224 }
225}
226
Roland Dreier70a30e12005-10-28 15:38:26 -0700227static int ib_uverbs_cleanup_ucontext(struct ib_uverbs_file *file,
228 struct ib_ucontext *context)
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700229{
230 struct ib_uobject *uobj, *tmp;
231
Roland Dreierf7c6a7b2007-03-04 16:15:11 -0800232 context->closing = 1;
233
Roland Dreier67cdb402005-10-14 15:26:04 -0700234 list_for_each_entry_safe(uobj, tmp, &context->ah_list, list) {
Roland Dreier9ead1902006-06-17 20:44:49 -0700235 struct ib_ah *ah = uobj->object;
236
237 idr_remove_uobj(&ib_uverbs_ah_idr, uobj);
Roland Dreier67cdb402005-10-14 15:26:04 -0700238 ib_destroy_ah(ah);
Roland Dreier67cdb402005-10-14 15:26:04 -0700239 kfree(uobj);
240 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700241
Shani Michaeli6b52a122013-02-06 16:19:13 +0000242 /* Remove MWs before QPs, in order to support type 2A MWs. */
243 list_for_each_entry_safe(uobj, tmp, &context->mw_list, list) {
244 struct ib_mw *mw = uobj->object;
245
246 idr_remove_uobj(&ib_uverbs_mw_idr, uobj);
Christoph Hellwigfeb7c1e2015-12-23 19:12:48 +0100247 uverbs_dealloc_mw(mw);
Shani Michaeli6b52a122013-02-06 16:19:13 +0000248 kfree(uobj);
249 }
250
Hadar Hen Zion436f2ad2013-08-14 13:58:30 +0300251 list_for_each_entry_safe(uobj, tmp, &context->rule_list, list) {
252 struct ib_flow *flow_id = uobj->object;
253
254 idr_remove_uobj(&ib_uverbs_rule_idr, uobj);
255 ib_destroy_flow(flow_id);
256 kfree(uobj);
257 }
258
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700259 list_for_each_entry_safe(uobj, tmp, &context->qp_list, list) {
Roland Dreier9ead1902006-06-17 20:44:49 -0700260 struct ib_qp *qp = uobj->object;
Jack Morgensteinf4e40152005-11-29 16:57:01 -0800261 struct ib_uqp_object *uqp =
262 container_of(uobj, struct ib_uqp_object, uevent.uobject);
Roland Dreier9ead1902006-06-17 20:44:49 -0700263
264 idr_remove_uobj(&ib_uverbs_qp_idr, uobj);
Tariq Toukan5b810a22016-10-27 16:36:26 +0300265 if (qp == qp->real_qp)
Sean Heftyb93f3c12011-05-27 00:00:12 -0700266 ib_uverbs_detach_umcast(qp, uqp);
Tariq Toukan5b810a22016-10-27 16:36:26 +0300267 ib_destroy_qp(qp);
Jack Morgensteinf4e40152005-11-29 16:57:01 -0800268 ib_uverbs_release_uevent(file, &uqp->uevent);
269 kfree(uqp);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700270 }
271
Yishai Hadasde019a92016-05-23 15:20:52 +0300272 list_for_each_entry_safe(uobj, tmp, &context->rwq_ind_tbl_list, list) {
273 struct ib_rwq_ind_table *rwq_ind_tbl = uobj->object;
274 struct ib_wq **ind_tbl = rwq_ind_tbl->ind_tbl;
275
276 idr_remove_uobj(&ib_uverbs_rwq_ind_tbl_idr, uobj);
277 ib_destroy_rwq_ind_table(rwq_ind_tbl);
278 kfree(ind_tbl);
279 kfree(uobj);
280 }
281
Yishai Hadasf213c052016-05-23 15:20:49 +0300282 list_for_each_entry_safe(uobj, tmp, &context->wq_list, list) {
283 struct ib_wq *wq = uobj->object;
284 struct ib_uwq_object *uwq =
285 container_of(uobj, struct ib_uwq_object, uevent.uobject);
286
287 idr_remove_uobj(&ib_uverbs_wq_idr, uobj);
288 ib_destroy_wq(wq);
289 ib_uverbs_release_uevent(file, &uwq->uevent);
290 kfree(uwq);
291 }
292
Sébastien Duguéa233c4b2015-04-09 11:13:41 +0200293 list_for_each_entry_safe(uobj, tmp, &context->srq_list, list) {
294 struct ib_srq *srq = uobj->object;
295 struct ib_uevent_object *uevent =
296 container_of(uobj, struct ib_uevent_object, uobject);
297
298 idr_remove_uobj(&ib_uverbs_srq_idr, uobj);
299 ib_destroy_srq(srq);
300 ib_uverbs_release_uevent(file, uevent);
301 kfree(uevent);
302 }
303
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700304 list_for_each_entry_safe(uobj, tmp, &context->cq_list, list) {
Roland Dreier9ead1902006-06-17 20:44:49 -0700305 struct ib_cq *cq = uobj->object;
Roland Dreier70a30e12005-10-28 15:38:26 -0700306 struct ib_uverbs_event_file *ev_file = cq->cq_context;
307 struct ib_ucq_object *ucq =
308 container_of(uobj, struct ib_ucq_object, uobject);
Roland Dreier9ead1902006-06-17 20:44:49 -0700309
310 idr_remove_uobj(&ib_uverbs_cq_idr, uobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700311 ib_destroy_cq(cq);
Roland Dreier70a30e12005-10-28 15:38:26 -0700312 ib_uverbs_release_ucq(file, ev_file, ucq);
313 kfree(ucq);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700314 }
315
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700316 list_for_each_entry_safe(uobj, tmp, &context->mr_list, list) {
Roland Dreier9ead1902006-06-17 20:44:49 -0700317 struct ib_mr *mr = uobj->object;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700318
Roland Dreier9ead1902006-06-17 20:44:49 -0700319 idr_remove_uobj(&ib_uverbs_mr_idr, uobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700320 ib_dereg_mr(mr);
Roland Dreierf7c6a7b2007-03-04 16:15:11 -0800321 kfree(uobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700322 }
323
Sean Hefty53d0bd12011-05-24 08:33:46 -0700324 mutex_lock(&file->device->xrcd_tree_mutex);
325 list_for_each_entry_safe(uobj, tmp, &context->xrcd_list, list) {
326 struct ib_xrcd *xrcd = uobj->object;
327 struct ib_uxrcd_object *uxrcd =
328 container_of(uobj, struct ib_uxrcd_object, uobject);
329
330 idr_remove_uobj(&ib_uverbs_xrcd_idr, uobj);
331 ib_uverbs_dealloc_xrcd(file->device, xrcd);
332 kfree(uxrcd);
333 }
334 mutex_unlock(&file->device->xrcd_tree_mutex);
335
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700336 list_for_each_entry_safe(uobj, tmp, &context->pd_list, list) {
Roland Dreier9ead1902006-06-17 20:44:49 -0700337 struct ib_pd *pd = uobj->object;
338
339 idr_remove_uobj(&ib_uverbs_pd_idr, uobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700340 ib_dealloc_pd(pd);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700341 kfree(uobj);
342 }
343
Shachar Raindel8ada2c12014-12-11 17:04:17 +0200344 put_pid(context->tgid);
345
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700346 return context->device->dealloc_ucontext(context);
347}
348
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300349static void ib_uverbs_comp_dev(struct ib_uverbs_device *dev)
350{
351 complete(&dev->comp);
352}
353
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700354static void ib_uverbs_release_file(struct kref *ref)
355{
356 struct ib_uverbs_file *file =
357 container_of(ref, struct ib_uverbs_file, ref);
Yishai Hadas036b1062015-08-13 18:32:05 +0300358 struct ib_device *ib_dev;
359 int srcu_key;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700360
Yishai Hadas036b1062015-08-13 18:32:05 +0300361 srcu_key = srcu_read_lock(&file->device->disassociate_srcu);
362 ib_dev = srcu_dereference(file->device->ib_dev,
363 &file->device->disassociate_srcu);
364 if (ib_dev && !ib_dev->disassociate_ucontext)
365 module_put(ib_dev->owner);
366 srcu_read_unlock(&file->device->disassociate_srcu, srcu_key);
367
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300368 if (atomic_dec_and_test(&file->device->refcount))
369 ib_uverbs_comp_dev(file->device);
Roland Dreier70a30e12005-10-28 15:38:26 -0700370
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700371 kfree(file);
372}
373
374static ssize_t ib_uverbs_event_read(struct file *filp, char __user *buf,
375 size_t count, loff_t *pos)
376{
377 struct ib_uverbs_event_file *file = filp->private_data;
Roland Dreier63aaf642005-09-09 15:55:08 -0700378 struct ib_uverbs_event *event;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700379 int eventsz;
380 int ret = 0;
381
382 spin_lock_irq(&file->lock);
383
Roland Dreier6b73597e2005-09-26 13:53:25 -0700384 while (list_empty(&file->event_list)) {
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700385 spin_unlock_irq(&file->lock);
386
387 if (filp->f_flags & O_NONBLOCK)
388 return -EAGAIN;
389
390 if (wait_event_interruptible(file->poll_wait,
Yishai Hadas036b1062015-08-13 18:32:05 +0300391 (!list_empty(&file->event_list) ||
392 /* The barriers built into wait_event_interruptible()
393 * and wake_up() guarentee this will see the null set
394 * without using RCU
395 */
396 !file->uverbs_file->device->ib_dev)))
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700397 return -ERESTARTSYS;
398
Yishai Hadas036b1062015-08-13 18:32:05 +0300399 /* If device was disassociated and no event exists set an error */
400 if (list_empty(&file->event_list) &&
401 !file->uverbs_file->device->ib_dev)
402 return -EIO;
403
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700404 spin_lock_irq(&file->lock);
405 }
406
Roland Dreier63aaf642005-09-09 15:55:08 -0700407 event = list_entry(file->event_list.next, struct ib_uverbs_event, list);
408
409 if (file->is_async)
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700410 eventsz = sizeof (struct ib_uverbs_async_event_desc);
Roland Dreier63aaf642005-09-09 15:55:08 -0700411 else
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700412 eventsz = sizeof (struct ib_uverbs_comp_event_desc);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700413
414 if (eventsz > count) {
415 ret = -EINVAL;
416 event = NULL;
Roland Dreier63aaf642005-09-09 15:55:08 -0700417 } else {
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700418 list_del(file->event_list.next);
Roland Dreier63aaf642005-09-09 15:55:08 -0700419 if (event->counter) {
420 ++(*event->counter);
421 list_del(&event->obj_list);
422 }
423 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700424
425 spin_unlock_irq(&file->lock);
426
427 if (event) {
428 if (copy_to_user(buf, event, eventsz))
429 ret = -EFAULT;
430 else
431 ret = eventsz;
432 }
433
434 kfree(event);
435
436 return ret;
437}
438
439static unsigned int ib_uverbs_event_poll(struct file *filp,
440 struct poll_table_struct *wait)
441{
442 unsigned int pollflags = 0;
443 struct ib_uverbs_event_file *file = filp->private_data;
444
445 poll_wait(filp, &file->poll_wait, wait);
446
447 spin_lock_irq(&file->lock);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700448 if (!list_empty(&file->event_list))
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700449 pollflags = POLLIN | POLLRDNORM;
450 spin_unlock_irq(&file->lock);
451
452 return pollflags;
453}
454
Gleb Natapovabdf1192005-07-27 14:40:00 -0700455static int ib_uverbs_event_fasync(int fd, struct file *filp, int on)
456{
457 struct ib_uverbs_event_file *file = filp->private_data;
458
459 return fasync_helper(fd, filp, on, &file->async_queue);
460}
461
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700462static int ib_uverbs_event_close(struct inode *inode, struct file *filp)
463{
464 struct ib_uverbs_event_file *file = filp->private_data;
Roland Dreier6b73597e2005-09-26 13:53:25 -0700465 struct ib_uverbs_event *entry, *tmp;
Yishai Hadas036b1062015-08-13 18:32:05 +0300466 int closed_already = 0;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700467
Yishai Hadas036b1062015-08-13 18:32:05 +0300468 mutex_lock(&file->uverbs_file->device->lists_mutex);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700469 spin_lock_irq(&file->lock);
Yishai Hadas036b1062015-08-13 18:32:05 +0300470 closed_already = file->is_closed;
Roland Dreier1ae5c182008-04-16 21:01:08 -0700471 file->is_closed = 1;
Roland Dreier6b73597e2005-09-26 13:53:25 -0700472 list_for_each_entry_safe(entry, tmp, &file->event_list, list) {
473 if (entry->counter)
474 list_del(&entry->obj_list);
475 kfree(entry);
476 }
477 spin_unlock_irq(&file->lock);
Yishai Hadas036b1062015-08-13 18:32:05 +0300478 if (!closed_already) {
479 list_del(&file->list);
480 if (file->is_async)
481 ib_unregister_event_handler(&file->uverbs_file->
482 event_handler);
483 }
484 mutex_unlock(&file->uverbs_file->device->lists_mutex);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700485
Yishai Hadas03c40442015-08-13 18:32:02 +0300486 kref_put(&file->uverbs_file->ref, ib_uverbs_release_file);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700487 kref_put(&file->ref, ib_uverbs_release_event_file);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700488
489 return 0;
490}
491
Arjan van de Ven2b8693c2007-02-12 00:55:32 -0800492static const struct file_operations uverbs_event_fops = {
Roland Dreier6b73597e2005-09-26 13:53:25 -0700493 .owner = THIS_MODULE,
Alexander Chiang9afed762010-02-02 19:08:19 +0000494 .read = ib_uverbs_event_read,
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700495 .poll = ib_uverbs_event_poll,
Gleb Natapovabdf1192005-07-27 14:40:00 -0700496 .release = ib_uverbs_event_close,
Roland Dreierbc1db9a2010-04-09 17:13:50 -0700497 .fasync = ib_uverbs_event_fasync,
498 .llseek = no_llseek,
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700499};
500
501void ib_uverbs_comp_handler(struct ib_cq *cq, void *cq_context)
502{
Roland Dreier6b73597e2005-09-26 13:53:25 -0700503 struct ib_uverbs_event_file *file = cq_context;
504 struct ib_ucq_object *uobj;
505 struct ib_uverbs_event *entry;
506 unsigned long flags;
507
508 if (!file)
509 return;
510
511 spin_lock_irqsave(&file->lock, flags);
Roland Dreier1ae5c182008-04-16 21:01:08 -0700512 if (file->is_closed) {
Roland Dreier6b73597e2005-09-26 13:53:25 -0700513 spin_unlock_irqrestore(&file->lock, flags);
514 return;
515 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700516
517 entry = kmalloc(sizeof *entry, GFP_ATOMIC);
Roland Dreier305a7e82005-10-11 15:39:38 -0700518 if (!entry) {
519 spin_unlock_irqrestore(&file->lock, flags);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700520 return;
Roland Dreier305a7e82005-10-11 15:39:38 -0700521 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700522
Roland Dreier63aaf642005-09-09 15:55:08 -0700523 uobj = container_of(cq->uobject, struct ib_ucq_object, uobject);
524
525 entry->desc.comp.cq_handle = cq->uobject->user_handle;
526 entry->counter = &uobj->comp_events_reported;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700527
Roland Dreier6b73597e2005-09-26 13:53:25 -0700528 list_add_tail(&entry->list, &file->event_list);
Roland Dreier63aaf642005-09-09 15:55:08 -0700529 list_add_tail(&entry->obj_list, &uobj->comp_list);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700530 spin_unlock_irqrestore(&file->lock, flags);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700531
Roland Dreier6b73597e2005-09-26 13:53:25 -0700532 wake_up_interruptible(&file->poll_wait);
533 kill_fasync(&file->async_queue, SIGIO, POLL_IN);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700534}
535
536static void ib_uverbs_async_handler(struct ib_uverbs_file *file,
Roland Dreier63aaf642005-09-09 15:55:08 -0700537 __u64 element, __u64 event,
538 struct list_head *obj_list,
539 u32 *counter)
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700540{
Roland Dreier63aaf642005-09-09 15:55:08 -0700541 struct ib_uverbs_event *entry;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700542 unsigned long flags;
543
Roland Dreier6b73597e2005-09-26 13:53:25 -0700544 spin_lock_irqsave(&file->async_file->lock, flags);
Jack Morgensteinfb77bce2008-06-18 15:36:38 -0700545 if (file->async_file->is_closed) {
Roland Dreier6b73597e2005-09-26 13:53:25 -0700546 spin_unlock_irqrestore(&file->async_file->lock, flags);
547 return;
548 }
549
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700550 entry = kmalloc(sizeof *entry, GFP_ATOMIC);
Roland Dreier305a7e82005-10-11 15:39:38 -0700551 if (!entry) {
552 spin_unlock_irqrestore(&file->async_file->lock, flags);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700553 return;
Roland Dreier305a7e82005-10-11 15:39:38 -0700554 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700555
Roland Dreier63aaf642005-09-09 15:55:08 -0700556 entry->desc.async.element = element;
557 entry->desc.async.event_type = event;
Eli Cohen377b5132014-09-14 16:47:52 +0300558 entry->desc.async.reserved = 0;
Roland Dreier63aaf642005-09-09 15:55:08 -0700559 entry->counter = counter;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700560
Roland Dreier6b73597e2005-09-26 13:53:25 -0700561 list_add_tail(&entry->list, &file->async_file->event_list);
Roland Dreier63aaf642005-09-09 15:55:08 -0700562 if (obj_list)
563 list_add_tail(&entry->obj_list, obj_list);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700564 spin_unlock_irqrestore(&file->async_file->lock, flags);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700565
Roland Dreier6b73597e2005-09-26 13:53:25 -0700566 wake_up_interruptible(&file->async_file->poll_wait);
567 kill_fasync(&file->async_file->async_queue, SIGIO, POLL_IN);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700568}
569
570void ib_uverbs_cq_event_handler(struct ib_event *event, void *context_ptr)
571{
Roland Dreier7162a3e2005-10-30 09:50:04 -0800572 struct ib_ucq_object *uobj = container_of(event->element.cq->uobject,
573 struct ib_ucq_object, uobject);
Roland Dreier63aaf642005-09-09 15:55:08 -0700574
Roland Dreier7162a3e2005-10-30 09:50:04 -0800575 ib_uverbs_async_handler(uobj->uverbs_file, uobj->uobject.user_handle,
Roland Dreier63aaf642005-09-09 15:55:08 -0700576 event->event, &uobj->async_list,
577 &uobj->async_events_reported);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700578}
579
580void ib_uverbs_qp_event_handler(struct ib_event *event, void *context_ptr)
581{
Roland Dreier63aaf642005-09-09 15:55:08 -0700582 struct ib_uevent_object *uobj;
583
Jack Morgensteina040f952014-09-23 12:38:26 +0300584 /* for XRC target qp's, check that qp is live */
585 if (!event->element.qp->uobject || !event->element.qp->uobject->live)
586 return;
587
Roland Dreier63aaf642005-09-09 15:55:08 -0700588 uobj = container_of(event->element.qp->uobject,
589 struct ib_uevent_object, uobject);
590
591 ib_uverbs_async_handler(context_ptr, uobj->uobject.user_handle,
592 event->event, &uobj->event_list,
593 &uobj->events_reported);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700594}
595
Yishai Hadasf213c052016-05-23 15:20:49 +0300596void ib_uverbs_wq_event_handler(struct ib_event *event, void *context_ptr)
597{
598 struct ib_uevent_object *uobj = container_of(event->element.wq->uobject,
599 struct ib_uevent_object, uobject);
600
601 ib_uverbs_async_handler(context_ptr, uobj->uobject.user_handle,
602 event->event, &uobj->event_list,
603 &uobj->events_reported);
604}
605
Roland Dreierf520ba52005-08-18 12:24:13 -0700606void ib_uverbs_srq_event_handler(struct ib_event *event, void *context_ptr)
607{
Roland Dreier63aaf642005-09-09 15:55:08 -0700608 struct ib_uevent_object *uobj;
609
610 uobj = container_of(event->element.srq->uobject,
611 struct ib_uevent_object, uobject);
612
613 ib_uverbs_async_handler(context_ptr, uobj->uobject.user_handle,
614 event->event, &uobj->event_list,
615 &uobj->events_reported);
Roland Dreierf520ba52005-08-18 12:24:13 -0700616}
617
Roland Dreier6b73597e2005-09-26 13:53:25 -0700618void ib_uverbs_event_handler(struct ib_event_handler *handler,
619 struct ib_event *event)
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700620{
621 struct ib_uverbs_file *file =
622 container_of(handler, struct ib_uverbs_file, event_handler);
623
Roland Dreier63aaf642005-09-09 15:55:08 -0700624 ib_uverbs_async_handler(file, event->element.port_num, event->event,
625 NULL, NULL);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700626}
627
Yishai Hadas03c40442015-08-13 18:32:02 +0300628void ib_uverbs_free_async_event_file(struct ib_uverbs_file *file)
629{
630 kref_put(&file->async_file->ref, ib_uverbs_release_event_file);
631 file->async_file = NULL;
632}
633
Roland Dreier6b73597e2005-09-26 13:53:25 -0700634struct file *ib_uverbs_alloc_event_file(struct ib_uverbs_file *uverbs_file,
Yishai Hadas057aec02015-08-13 18:32:04 +0300635 struct ib_device *ib_dev,
Al Virob1e4594b2010-01-18 01:38:00 -0500636 int is_async)
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700637{
Roland Dreier6b73597e2005-09-26 13:53:25 -0700638 struct ib_uverbs_event_file *ev_file;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700639 struct file *filp;
Yishai Hadas03c40442015-08-13 18:32:02 +0300640 int ret;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700641
Yishai Hadas03c40442015-08-13 18:32:02 +0300642 ev_file = kzalloc(sizeof(*ev_file), GFP_KERNEL);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700643 if (!ev_file)
644 return ERR_PTR(-ENOMEM);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700645
Roland Dreier6b73597e2005-09-26 13:53:25 -0700646 kref_init(&ev_file->ref);
647 spin_lock_init(&ev_file->lock);
648 INIT_LIST_HEAD(&ev_file->event_list);
649 init_waitqueue_head(&ev_file->poll_wait);
650 ev_file->uverbs_file = uverbs_file;
Yishai Hadas03c40442015-08-13 18:32:02 +0300651 kref_get(&ev_file->uverbs_file->ref);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700652 ev_file->async_queue = NULL;
Roland Dreier1ae5c182008-04-16 21:01:08 -0700653 ev_file->is_closed = 0;
Roland Dreier6b73597e2005-09-26 13:53:25 -0700654
Al Virob1e4594b2010-01-18 01:38:00 -0500655 filp = anon_inode_getfile("[infinibandevent]", &uverbs_event_fops,
Roland Dreiera265e552010-02-24 16:51:20 -0800656 ev_file, O_RDONLY);
Al Virob1e4594b2010-01-18 01:38:00 -0500657 if (IS_ERR(filp))
Yishai Hadas03c40442015-08-13 18:32:02 +0300658 goto err_put_refs;
Roland Dreiera7dab9e2008-04-16 21:01:08 -0700659
Yishai Hadas036b1062015-08-13 18:32:05 +0300660 mutex_lock(&uverbs_file->device->lists_mutex);
661 list_add_tail(&ev_file->list,
662 &uverbs_file->device->uverbs_events_file_list);
663 mutex_unlock(&uverbs_file->device->lists_mutex);
664
Yishai Hadas03c40442015-08-13 18:32:02 +0300665 if (is_async) {
666 WARN_ON(uverbs_file->async_file);
667 uverbs_file->async_file = ev_file;
668 kref_get(&uverbs_file->async_file->ref);
669 INIT_IB_EVENT_HANDLER(&uverbs_file->event_handler,
Yishai Hadas057aec02015-08-13 18:32:04 +0300670 ib_dev,
Yishai Hadas03c40442015-08-13 18:32:02 +0300671 ib_uverbs_event_handler);
672 ret = ib_register_event_handler(&uverbs_file->event_handler);
673 if (ret)
674 goto err_put_file;
675
676 /* At that point async file stuff was fully set */
677 ev_file->is_async = 1;
678 }
679
680 return filp;
681
682err_put_file:
683 fput(filp);
684 kref_put(&uverbs_file->async_file->ref, ib_uverbs_release_event_file);
685 uverbs_file->async_file = NULL;
686 return ERR_PTR(ret);
687
688err_put_refs:
689 kref_put(&ev_file->uverbs_file->ref, ib_uverbs_release_file);
690 kref_put(&ev_file->ref, ib_uverbs_release_event_file);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700691 return filp;
Roland Dreier6b73597e2005-09-26 13:53:25 -0700692}
693
694/*
695 * Look up a completion event file by FD. If lookup is successful,
696 * takes a ref to the event file struct that it returns; if
697 * unsuccessful, returns NULL.
698 */
699struct ib_uverbs_event_file *ib_uverbs_lookup_comp_file(int fd)
700{
701 struct ib_uverbs_event_file *ev_file = NULL;
Al Viro2903ff02012-08-28 12:52:22 -0400702 struct fd f = fdget(fd);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700703
Al Viro2903ff02012-08-28 12:52:22 -0400704 if (!f.file)
Roland Dreier6b73597e2005-09-26 13:53:25 -0700705 return NULL;
706
Al Viro2903ff02012-08-28 12:52:22 -0400707 if (f.file->f_op != &uverbs_event_fops)
Roland Dreier6b73597e2005-09-26 13:53:25 -0700708 goto out;
709
Al Viro2903ff02012-08-28 12:52:22 -0400710 ev_file = f.file->private_data;
Roland Dreier6b73597e2005-09-26 13:53:25 -0700711 if (ev_file->is_async) {
712 ev_file = NULL;
713 goto out;
714 }
715
716 kref_get(&ev_file->ref);
717
718out:
Al Viro2903ff02012-08-28 12:52:22 -0400719 fdput(f);
Roland Dreier6b73597e2005-09-26 13:53:25 -0700720 return ev_file;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700721}
722
Eli Cohen2dbd5182016-02-14 17:07:48 +0200723static int verify_command_mask(struct ib_device *ib_dev, __u32 command)
724{
725 u64 mask;
726
727 if (command <= IB_USER_VERBS_CMD_OPEN_QP)
728 mask = ib_dev->uverbs_cmd_mask;
729 else
730 mask = ib_dev->uverbs_ex_cmd_mask;
731
732 if (mask & ((u64)1 << command))
733 return 0;
734
735 return -1;
736}
737
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700738static ssize_t ib_uverbs_write(struct file *filp, const char __user *buf,
739 size_t count, loff_t *pos)
740{
741 struct ib_uverbs_file *file = filp->private_data;
Yishai Hadas036b1062015-08-13 18:32:05 +0300742 struct ib_device *ib_dev;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700743 struct ib_uverbs_cmd_hdr hdr;
Eli Cohen74a0b0a2016-02-14 17:07:47 +0200744 __u32 command;
Yann Droneaudf21519b2013-11-06 23:21:49 +0100745 __u32 flags;
Yishai Hadas036b1062015-08-13 18:32:05 +0300746 int srcu_key;
747 ssize_t ret;
Yishai Hadas057aec02015-08-13 18:32:04 +0300748
Jason Gunthorpee6bd18f2016-04-10 19:13:13 -0600749 if (WARN_ON_ONCE(!ib_safe_file_access(filp)))
750 return -EACCES;
751
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700752 if (count < sizeof hdr)
753 return -EINVAL;
754
755 if (copy_from_user(&hdr, buf, sizeof hdr))
756 return -EFAULT;
757
Yishai Hadas036b1062015-08-13 18:32:05 +0300758 srcu_key = srcu_read_lock(&file->device->disassociate_srcu);
759 ib_dev = srcu_dereference(file->device->ib_dev,
760 &file->device->disassociate_srcu);
761 if (!ib_dev) {
762 ret = -EIO;
763 goto out;
764 }
765
Eli Cohen74a0b0a2016-02-14 17:07:47 +0200766 if (hdr.command & ~(__u32)(IB_USER_VERBS_CMD_FLAGS_MASK |
767 IB_USER_VERBS_CMD_COMMAND_MASK)) {
768 ret = -EINVAL;
769 goto out;
770 }
771
772 command = hdr.command & IB_USER_VERBS_CMD_COMMAND_MASK;
Eli Cohen2dbd5182016-02-14 17:07:48 +0200773 if (verify_command_mask(ib_dev, command)) {
774 ret = -EOPNOTSUPP;
775 goto out;
776 }
Eli Cohen74a0b0a2016-02-14 17:07:47 +0200777
Eli Coheneaebc7d2016-02-14 17:07:49 +0200778 if (!file->ucontext &&
779 command != IB_USER_VERBS_CMD_GET_CONTEXT) {
780 ret = -EINVAL;
781 goto out;
782 }
783
Yann Droneaudf21519b2013-11-06 23:21:49 +0100784 flags = (hdr.command &
785 IB_USER_VERBS_CMD_FLAGS_MASK) >> IB_USER_VERBS_CMD_FLAGS_SHIFT;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700786
Yann Droneaudf21519b2013-11-06 23:21:49 +0100787 if (!flags) {
Yann Droneaudf21519b2013-11-06 23:21:49 +0100788 if (command >= ARRAY_SIZE(uverbs_cmd_table) ||
Yishai Hadas036b1062015-08-13 18:32:05 +0300789 !uverbs_cmd_table[command]) {
790 ret = -EINVAL;
791 goto out;
792 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100793
Yishai Hadas036b1062015-08-13 18:32:05 +0300794 if (hdr.in_words * 4 != count) {
795 ret = -EINVAL;
796 goto out;
797 }
Igor Ivanov400dbc92013-08-14 13:58:29 +0300798
Yishai Hadas036b1062015-08-13 18:32:05 +0300799 ret = uverbs_cmd_table[command](file, ib_dev,
Yann Droneaudf21519b2013-11-06 23:21:49 +0100800 buf + sizeof(hdr),
801 hdr.in_words * 4,
802 hdr.out_words * 4);
803
Yann Droneaudf21519b2013-11-06 23:21:49 +0100804 } else if (flags == IB_USER_VERBS_CMD_FLAG_EXTENDED) {
Yann Droneaudf21519b2013-11-06 23:21:49 +0100805 struct ib_uverbs_ex_cmd_hdr ex_hdr;
806 struct ib_udata ucore;
807 struct ib_udata uhw;
Yann Droneaudf21519b2013-11-06 23:21:49 +0100808 size_t written_count = count;
809
Yann Droneaudf21519b2013-11-06 23:21:49 +0100810 if (command >= ARRAY_SIZE(uverbs_ex_cmd_table) ||
Yishai Hadas036b1062015-08-13 18:32:05 +0300811 !uverbs_ex_cmd_table[command]) {
812 ret = -ENOSYS;
813 goto out;
814 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100815
Yishai Hadas036b1062015-08-13 18:32:05 +0300816 if (!file->ucontext) {
817 ret = -EINVAL;
818 goto out;
819 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100820
Yishai Hadas036b1062015-08-13 18:32:05 +0300821 if (count < (sizeof(hdr) + sizeof(ex_hdr))) {
822 ret = -EINVAL;
823 goto out;
824 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100825
Yishai Hadas036b1062015-08-13 18:32:05 +0300826 if (copy_from_user(&ex_hdr, buf + sizeof(hdr), sizeof(ex_hdr))) {
827 ret = -EFAULT;
828 goto out;
829 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100830
831 count -= sizeof(hdr) + sizeof(ex_hdr);
832 buf += sizeof(hdr) + sizeof(ex_hdr);
833
Yishai Hadas036b1062015-08-13 18:32:05 +0300834 if ((hdr.in_words + ex_hdr.provider_in_words) * 8 != count) {
835 ret = -EINVAL;
836 goto out;
837 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100838
Yishai Hadas036b1062015-08-13 18:32:05 +0300839 if (ex_hdr.cmd_hdr_reserved) {
840 ret = -EINVAL;
841 goto out;
842 }
Yann Droneaud7efb1b12013-12-11 23:01:47 +0100843
Yann Droneaudf21519b2013-11-06 23:21:49 +0100844 if (ex_hdr.response) {
Yishai Hadas036b1062015-08-13 18:32:05 +0300845 if (!hdr.out_words && !ex_hdr.provider_out_words) {
846 ret = -EINVAL;
847 goto out;
848 }
Yann Droneaud6cc3df82013-12-11 23:01:51 +0100849
850 if (!access_ok(VERIFY_WRITE,
851 (void __user *) (unsigned long) ex_hdr.response,
Yishai Hadas036b1062015-08-13 18:32:05 +0300852 (hdr.out_words + ex_hdr.provider_out_words) * 8)) {
853 ret = -EFAULT;
854 goto out;
855 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100856 } else {
Yishai Hadas036b1062015-08-13 18:32:05 +0300857 if (hdr.out_words || ex_hdr.provider_out_words) {
858 ret = -EINVAL;
859 goto out;
860 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100861 }
862
Roland Dreiera96e4e22013-12-19 08:37:03 -0800863 INIT_UDATA_BUF_OR_NULL(&ucore, buf, (unsigned long) ex_hdr.response,
864 hdr.in_words * 8, hdr.out_words * 8);
Yann Droneaudf21519b2013-11-06 23:21:49 +0100865
Roland Dreiera96e4e22013-12-19 08:37:03 -0800866 INIT_UDATA_BUF_OR_NULL(&uhw,
867 buf + ucore.inlen,
868 (unsigned long) ex_hdr.response + ucore.outlen,
869 ex_hdr.provider_in_words * 8,
870 ex_hdr.provider_out_words * 8);
Yann Droneaudf21519b2013-11-06 23:21:49 +0100871
Yishai Hadas036b1062015-08-13 18:32:05 +0300872 ret = uverbs_ex_cmd_table[command](file,
Yishai Hadas057aec02015-08-13 18:32:04 +0300873 ib_dev,
Yann Droneaudf21519b2013-11-06 23:21:49 +0100874 &ucore,
875 &uhw);
Yishai Hadas036b1062015-08-13 18:32:05 +0300876 if (!ret)
877 ret = written_count;
878 } else {
879 ret = -ENOSYS;
Igor Ivanov400dbc92013-08-14 13:58:29 +0300880 }
Yann Droneaudf21519b2013-11-06 23:21:49 +0100881
Yishai Hadas036b1062015-08-13 18:32:05 +0300882out:
883 srcu_read_unlock(&file->device->disassociate_srcu, srcu_key);
884 return ret;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700885}
886
887static int ib_uverbs_mmap(struct file *filp, struct vm_area_struct *vma)
888{
889 struct ib_uverbs_file *file = filp->private_data;
Yishai Hadas036b1062015-08-13 18:32:05 +0300890 struct ib_device *ib_dev;
891 int ret = 0;
892 int srcu_key;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700893
Yishai Hadas036b1062015-08-13 18:32:05 +0300894 srcu_key = srcu_read_lock(&file->device->disassociate_srcu);
895 ib_dev = srcu_dereference(file->device->ib_dev,
896 &file->device->disassociate_srcu);
897 if (!ib_dev) {
898 ret = -EIO;
899 goto out;
900 }
901
902 if (!file->ucontext)
903 ret = -ENODEV;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700904 else
Yishai Hadas036b1062015-08-13 18:32:05 +0300905 ret = ib_dev->mmap(file->ucontext, vma);
906out:
907 srcu_read_unlock(&file->device->disassociate_srcu, srcu_key);
908 return ret;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700909}
910
Roland Dreier5b2d2812008-06-27 14:43:20 -0700911/*
912 * ib_uverbs_open() does not need the BKL:
913 *
Alexander Chiang2a72f212010-02-02 19:07:54 +0000914 * - the ib_uverbs_device structures are properly reference counted and
Roland Dreier5b2d2812008-06-27 14:43:20 -0700915 * everything else is purely local to the file being created, so
916 * races against other open calls are not a problem;
917 * - there is no ioctl method to race against;
Alexander Chiang2a72f212010-02-02 19:07:54 +0000918 * - the open method will either immediately run -ENXIO, or all
919 * required initialization will be done.
Roland Dreier5b2d2812008-06-27 14:43:20 -0700920 */
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700921static int ib_uverbs_open(struct inode *inode, struct file *filp)
922{
Roland Dreier70a30e12005-10-28 15:38:26 -0700923 struct ib_uverbs_device *dev;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700924 struct ib_uverbs_file *file;
Yishai Hadas036b1062015-08-13 18:32:05 +0300925 struct ib_device *ib_dev;
Roland Dreier70a30e12005-10-28 15:38:26 -0700926 int ret;
Yishai Hadas036b1062015-08-13 18:32:05 +0300927 int module_dependent;
928 int srcu_key;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700929
Alexander Chiang2a72f212010-02-02 19:07:54 +0000930 dev = container_of(inode->i_cdev, struct ib_uverbs_device, cdev);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300931 if (!atomic_inc_not_zero(&dev->refcount))
Roland Dreier70a30e12005-10-28 15:38:26 -0700932 return -ENXIO;
933
Yishai Hadas036b1062015-08-13 18:32:05 +0300934 srcu_key = srcu_read_lock(&dev->disassociate_srcu);
935 mutex_lock(&dev->lists_mutex);
936 ib_dev = srcu_dereference(dev->ib_dev,
937 &dev->disassociate_srcu);
938 if (!ib_dev) {
939 ret = -EIO;
Roland Dreier70a30e12005-10-28 15:38:26 -0700940 goto err;
941 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700942
Yishai Hadas036b1062015-08-13 18:32:05 +0300943 /* In case IB device supports disassociate ucontext, there is no hard
944 * dependency between uverbs device and its low level device.
945 */
946 module_dependent = !(ib_dev->disassociate_ucontext);
947
948 if (module_dependent) {
949 if (!try_module_get(ib_dev->owner)) {
950 ret = -ENODEV;
951 goto err;
952 }
953 }
954
955 file = kzalloc(sizeof(*file), GFP_KERNEL);
Roland Dreier63c47c22005-09-26 13:01:03 -0700956 if (!file) {
Roland Dreier70a30e12005-10-28 15:38:26 -0700957 ret = -ENOMEM;
Yishai Hadas036b1062015-08-13 18:32:05 +0300958 if (module_dependent)
959 goto err_module;
960
961 goto err;
Roland Dreier63c47c22005-09-26 13:01:03 -0700962 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700963
Roland Dreier70a30e12005-10-28 15:38:26 -0700964 file->device = dev;
965 file->ucontext = NULL;
966 file->async_file = NULL;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700967 kref_init(&file->ref);
Ingo Molnar95ed6442006-01-13 14:51:39 -0800968 mutex_init(&file->mutex);
Jason Gunthorped1e09f302016-07-03 15:28:18 +0300969 mutex_init(&file->cleanup_mutex);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700970
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700971 filp->private_data = file;
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300972 kobject_get(&dev->kobj);
Yishai Hadas036b1062015-08-13 18:32:05 +0300973 list_add_tail(&file->list, &dev->uverbs_file_list);
974 mutex_unlock(&dev->lists_mutex);
975 srcu_read_unlock(&dev->disassociate_srcu, srcu_key);
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700976
Roland Dreierbc1db9a2010-04-09 17:13:50 -0700977 return nonseekable_open(inode, filp);
Roland Dreier70a30e12005-10-28 15:38:26 -0700978
979err_module:
Yishai Hadas036b1062015-08-13 18:32:05 +0300980 module_put(ib_dev->owner);
Roland Dreier70a30e12005-10-28 15:38:26 -0700981
982err:
Yishai Hadas036b1062015-08-13 18:32:05 +0300983 mutex_unlock(&dev->lists_mutex);
984 srcu_read_unlock(&dev->disassociate_srcu, srcu_key);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300985 if (atomic_dec_and_test(&dev->refcount))
986 ib_uverbs_comp_dev(dev);
987
Roland Dreier70a30e12005-10-28 15:38:26 -0700988 return ret;
Roland Dreierbc38a6a2005-07-07 17:57:13 -0700989}
990
991static int ib_uverbs_close(struct inode *inode, struct file *filp)
992{
993 struct ib_uverbs_file *file = filp->private_data;
Yishai Hadas35d4a0b2015-08-13 18:32:03 +0300994 struct ib_uverbs_device *dev = file->device;
Jason Gunthorped1e09f302016-07-03 15:28:18 +0300995
996 mutex_lock(&file->cleanup_mutex);
997 if (file->ucontext) {
998 ib_uverbs_cleanup_ucontext(file, file->ucontext);
999 file->ucontext = NULL;
1000 }
1001 mutex_unlock(&file->cleanup_mutex);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001002
Yishai Hadas036b1062015-08-13 18:32:05 +03001003 mutex_lock(&file->device->lists_mutex);
Yishai Hadas036b1062015-08-13 18:32:05 +03001004 if (!file->is_closed) {
1005 list_del(&file->list);
1006 file->is_closed = 1;
1007 }
1008 mutex_unlock(&file->device->lists_mutex);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001009
Roland Dreier70a30e12005-10-28 15:38:26 -07001010 if (file->async_file)
1011 kref_put(&file->async_file->ref, ib_uverbs_release_event_file);
1012
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001013 kref_put(&file->ref, ib_uverbs_release_file);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001014 kobject_put(&dev->kobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001015
1016 return 0;
1017}
1018
Arjan van de Ven2b8693c2007-02-12 00:55:32 -08001019static const struct file_operations uverbs_fops = {
Alexander Chiang9afed762010-02-02 19:08:19 +00001020 .owner = THIS_MODULE,
1021 .write = ib_uverbs_write,
1022 .open = ib_uverbs_open,
Roland Dreierbc1db9a2010-04-09 17:13:50 -07001023 .release = ib_uverbs_close,
1024 .llseek = no_llseek,
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001025};
1026
Arjan van de Ven2b8693c2007-02-12 00:55:32 -08001027static const struct file_operations uverbs_mmap_fops = {
Alexander Chiang9afed762010-02-02 19:08:19 +00001028 .owner = THIS_MODULE,
1029 .write = ib_uverbs_write,
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001030 .mmap = ib_uverbs_mmap,
Alexander Chiang9afed762010-02-02 19:08:19 +00001031 .open = ib_uverbs_open,
Roland Dreierbc1db9a2010-04-09 17:13:50 -07001032 .release = ib_uverbs_close,
1033 .llseek = no_llseek,
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001034};
1035
1036static struct ib_client uverbs_client = {
1037 .name = "uverbs",
1038 .add = ib_uverbs_add_one,
1039 .remove = ib_uverbs_remove_one
1040};
1041
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001042static ssize_t show_ibdev(struct device *device, struct device_attribute *attr,
1043 char *buf)
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001044{
Yishai Hadas036b1062015-08-13 18:32:05 +03001045 int ret = -ENODEV;
1046 int srcu_key;
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001047 struct ib_uverbs_device *dev = dev_get_drvdata(device);
Yishai Hadas036b1062015-08-13 18:32:05 +03001048 struct ib_device *ib_dev;
Roland Dreier70a30e12005-10-28 15:38:26 -07001049
1050 if (!dev)
1051 return -ENODEV;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001052
Yishai Hadas036b1062015-08-13 18:32:05 +03001053 srcu_key = srcu_read_lock(&dev->disassociate_srcu);
1054 ib_dev = srcu_dereference(dev->ib_dev, &dev->disassociate_srcu);
1055 if (ib_dev)
1056 ret = sprintf(buf, "%s\n", ib_dev->name);
1057 srcu_read_unlock(&dev->disassociate_srcu, srcu_key);
1058
1059 return ret;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001060}
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001061static DEVICE_ATTR(ibdev, S_IRUGO, show_ibdev, NULL);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001062
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001063static ssize_t show_dev_abi_version(struct device *device,
1064 struct device_attribute *attr, char *buf)
Roland Dreier274c0892005-09-29 14:17:48 -07001065{
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001066 struct ib_uverbs_device *dev = dev_get_drvdata(device);
Yishai Hadas036b1062015-08-13 18:32:05 +03001067 int ret = -ENODEV;
1068 int srcu_key;
1069 struct ib_device *ib_dev;
Roland Dreier70a30e12005-10-28 15:38:26 -07001070
1071 if (!dev)
1072 return -ENODEV;
Yishai Hadas036b1062015-08-13 18:32:05 +03001073 srcu_key = srcu_read_lock(&dev->disassociate_srcu);
1074 ib_dev = srcu_dereference(dev->ib_dev, &dev->disassociate_srcu);
1075 if (ib_dev)
1076 ret = sprintf(buf, "%d\n", ib_dev->uverbs_abi_ver);
1077 srcu_read_unlock(&dev->disassociate_srcu, srcu_key);
Roland Dreier274c0892005-09-29 14:17:48 -07001078
Yishai Hadas036b1062015-08-13 18:32:05 +03001079 return ret;
Roland Dreier274c0892005-09-29 14:17:48 -07001080}
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001081static DEVICE_ATTR(abi_version, S_IRUGO, show_dev_abi_version, NULL);
Roland Dreier274c0892005-09-29 14:17:48 -07001082
Andi Kleen0933e2d2010-01-05 12:48:09 +01001083static CLASS_ATTR_STRING(abi_version, S_IRUGO,
1084 __stringify(IB_USER_VERBS_ABI_VERSION));
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001085
Alexander Chiang6d6a0e72010-02-02 19:08:09 +00001086static dev_t overflow_maj;
1087static DECLARE_BITMAP(overflow_map, IB_UVERBS_MAX_DEVICES);
1088
1089/*
1090 * If we have more than IB_UVERBS_MAX_DEVICES, dynamically overflow by
1091 * requesting a new major number and doubling the number of max devices we
1092 * support. It's stupid, but simple.
1093 */
1094static int find_overflow_devnum(void)
1095{
1096 int ret;
1097
1098 if (!overflow_maj) {
1099 ret = alloc_chrdev_region(&overflow_maj, 0, IB_UVERBS_MAX_DEVICES,
1100 "infiniband_verbs");
1101 if (ret) {
Parav Panditaba25a3e2016-03-02 00:50:29 +05301102 pr_err("user_verbs: couldn't register dynamic device number\n");
Alexander Chiang6d6a0e72010-02-02 19:08:09 +00001103 return ret;
1104 }
1105 }
1106
1107 ret = find_first_zero_bit(overflow_map, IB_UVERBS_MAX_DEVICES);
1108 if (ret >= IB_UVERBS_MAX_DEVICES)
1109 return -1;
1110
1111 return ret;
1112}
1113
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001114static void ib_uverbs_add_one(struct ib_device *device)
1115{
Alexander Chiang38707982010-02-02 19:07:59 +00001116 int devnum;
Alexander Chiangddbd6882010-02-02 19:08:04 +00001117 dev_t base;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001118 struct ib_uverbs_device *uverbs_dev;
Yishai Hadas036b1062015-08-13 18:32:05 +03001119 int ret;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001120
1121 if (!device->alloc_ucontext)
1122 return;
1123
Roland Dreierde6eb662005-11-02 07:23:14 -08001124 uverbs_dev = kzalloc(sizeof *uverbs_dev, GFP_KERNEL);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001125 if (!uverbs_dev)
1126 return;
1127
Yishai Hadas036b1062015-08-13 18:32:05 +03001128 ret = init_srcu_struct(&uverbs_dev->disassociate_srcu);
1129 if (ret) {
1130 kfree(uverbs_dev);
1131 return;
1132 }
1133
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001134 atomic_set(&uverbs_dev->refcount, 1);
Jack Morgensteinfd60ae42006-08-03 10:56:42 -07001135 init_completion(&uverbs_dev->comp);
Sean Hefty53d0bd12011-05-24 08:33:46 -07001136 uverbs_dev->xrcd_tree = RB_ROOT;
1137 mutex_init(&uverbs_dev->xrcd_tree_mutex);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001138 kobject_init(&uverbs_dev->kobj, &ib_uverbs_dev_ktype);
Yishai Hadas036b1062015-08-13 18:32:05 +03001139 mutex_init(&uverbs_dev->lists_mutex);
1140 INIT_LIST_HEAD(&uverbs_dev->uverbs_file_list);
1141 INIT_LIST_HEAD(&uverbs_dev->uverbs_events_file_list);
Roland Dreier70a30e12005-10-28 15:38:26 -07001142
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001143 spin_lock(&map_lock);
Alexander Chiang38707982010-02-02 19:07:59 +00001144 devnum = find_first_zero_bit(dev_map, IB_UVERBS_MAX_DEVICES);
1145 if (devnum >= IB_UVERBS_MAX_DEVICES) {
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001146 spin_unlock(&map_lock);
Alexander Chiang6d6a0e72010-02-02 19:08:09 +00001147 devnum = find_overflow_devnum();
1148 if (devnum < 0)
1149 goto err;
1150
1151 spin_lock(&map_lock);
1152 uverbs_dev->devnum = devnum + IB_UVERBS_MAX_DEVICES;
1153 base = devnum + overflow_maj;
1154 set_bit(devnum, overflow_map);
1155 } else {
1156 uverbs_dev->devnum = devnum;
1157 base = devnum + IB_UVERBS_BASE_DEV;
1158 set_bit(devnum, dev_map);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001159 }
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001160 spin_unlock(&map_lock);
1161
Yishai Hadas036b1062015-08-13 18:32:05 +03001162 rcu_assign_pointer(uverbs_dev->ib_dev, device);
Michael S. Tsirkinf4fd0b22007-05-03 13:48:47 +03001163 uverbs_dev->num_comp_vectors = device->num_comp_vectors;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001164
Alexander Chiang055422d2010-02-02 19:07:49 +00001165 cdev_init(&uverbs_dev->cdev, NULL);
1166 uverbs_dev->cdev.owner = THIS_MODULE;
1167 uverbs_dev->cdev.ops = device->mmap ? &uverbs_mmap_fops : &uverbs_fops;
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001168 uverbs_dev->cdev.kobj.parent = &uverbs_dev->kobj;
Alexander Chiang055422d2010-02-02 19:07:49 +00001169 kobject_set_name(&uverbs_dev->cdev.kobj, "uverbs%d", uverbs_dev->devnum);
Alexander Chiangddbd6882010-02-02 19:08:04 +00001170 if (cdev_add(&uverbs_dev->cdev, base, 1))
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001171 goto err_cdev;
1172
Greg Kroah-Hartman91bd4182008-07-21 20:03:34 -07001173 uverbs_dev->dev = device_create(uverbs_class, device->dma_device,
Alexander Chiang055422d2010-02-02 19:07:49 +00001174 uverbs_dev->cdev.dev, uverbs_dev,
Greg Kroah-Hartman91bd4182008-07-21 20:03:34 -07001175 "uverbs%d", uverbs_dev->devnum);
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001176 if (IS_ERR(uverbs_dev->dev))
Roland Dreier70a30e12005-10-28 15:38:26 -07001177 goto err_cdev;
1178
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001179 if (device_create_file(uverbs_dev->dev, &dev_attr_ibdev))
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001180 goto err_class;
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001181 if (device_create_file(uverbs_dev->dev, &dev_attr_abi_version))
Roland Dreier274c0892005-09-29 14:17:48 -07001182 goto err_class;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001183
1184 ib_set_client_data(device, &uverbs_client, uverbs_dev);
1185
1186 return;
1187
1188err_class:
Alexander Chiang055422d2010-02-02 19:07:49 +00001189 device_destroy(uverbs_class, uverbs_dev->cdev.dev);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001190
1191err_cdev:
Alexander Chiang055422d2010-02-02 19:07:49 +00001192 cdev_del(&uverbs_dev->cdev);
Alexander Chiang6d6a0e72010-02-02 19:08:09 +00001193 if (uverbs_dev->devnum < IB_UVERBS_MAX_DEVICES)
1194 clear_bit(devnum, dev_map);
1195 else
1196 clear_bit(devnum, overflow_map);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001197
1198err:
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001199 if (atomic_dec_and_test(&uverbs_dev->refcount))
1200 ib_uverbs_comp_dev(uverbs_dev);
Jack Morgensteinfd60ae42006-08-03 10:56:42 -07001201 wait_for_completion(&uverbs_dev->comp);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001202 kobject_put(&uverbs_dev->kobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001203 return;
1204}
1205
Yishai Hadas036b1062015-08-13 18:32:05 +03001206static void ib_uverbs_free_hw_resources(struct ib_uverbs_device *uverbs_dev,
1207 struct ib_device *ib_dev)
1208{
1209 struct ib_uverbs_file *file;
1210 struct ib_uverbs_event_file *event_file;
1211 struct ib_event event;
1212
1213 /* Pending running commands to terminate */
1214 synchronize_srcu(&uverbs_dev->disassociate_srcu);
1215 event.event = IB_EVENT_DEVICE_FATAL;
1216 event.element.port_num = 0;
1217 event.device = ib_dev;
1218
1219 mutex_lock(&uverbs_dev->lists_mutex);
1220 while (!list_empty(&uverbs_dev->uverbs_file_list)) {
1221 struct ib_ucontext *ucontext;
Yishai Hadas036b1062015-08-13 18:32:05 +03001222 file = list_first_entry(&uverbs_dev->uverbs_file_list,
1223 struct ib_uverbs_file, list);
1224 file->is_closed = 1;
Yishai Hadas036b1062015-08-13 18:32:05 +03001225 list_del(&file->list);
Yishai Hadas036b1062015-08-13 18:32:05 +03001226 kref_get(&file->ref);
1227 mutex_unlock(&uverbs_dev->lists_mutex);
Jason Gunthorped1e09f302016-07-03 15:28:18 +03001228
Yishai Hadas036b1062015-08-13 18:32:05 +03001229 ib_uverbs_event_handler(&file->event_handler, &event);
Jason Gunthorped1e09f302016-07-03 15:28:18 +03001230
1231 mutex_lock(&file->cleanup_mutex);
1232 ucontext = file->ucontext;
1233 file->ucontext = NULL;
1234 mutex_unlock(&file->cleanup_mutex);
1235
1236 /* At this point ib_uverbs_close cannot be running
1237 * ib_uverbs_cleanup_ucontext
1238 */
Yishai Hadas036b1062015-08-13 18:32:05 +03001239 if (ucontext) {
Jason Gunthorped1e09f302016-07-03 15:28:18 +03001240 /* We must release the mutex before going ahead and
1241 * calling disassociate_ucontext. disassociate_ucontext
1242 * might end up indirectly calling uverbs_close,
1243 * for example due to freeing the resources
1244 * (e.g mmput).
1245 */
Yishai Hadas036b1062015-08-13 18:32:05 +03001246 ib_dev->disassociate_ucontext(ucontext);
1247 ib_uverbs_cleanup_ucontext(file, ucontext);
1248 }
1249
1250 mutex_lock(&uverbs_dev->lists_mutex);
1251 kref_put(&file->ref, ib_uverbs_release_file);
1252 }
1253
1254 while (!list_empty(&uverbs_dev->uverbs_events_file_list)) {
1255 event_file = list_first_entry(&uverbs_dev->
1256 uverbs_events_file_list,
1257 struct ib_uverbs_event_file,
1258 list);
1259 spin_lock_irq(&event_file->lock);
1260 event_file->is_closed = 1;
1261 spin_unlock_irq(&event_file->lock);
1262
1263 list_del(&event_file->list);
1264 if (event_file->is_async) {
1265 ib_unregister_event_handler(&event_file->uverbs_file->
1266 event_handler);
1267 event_file->uverbs_file->event_handler.device = NULL;
1268 }
1269
1270 wake_up_interruptible(&event_file->poll_wait);
1271 kill_fasync(&event_file->async_queue, SIGIO, POLL_IN);
1272 }
1273 mutex_unlock(&uverbs_dev->lists_mutex);
1274}
1275
Haggai Eran7c1eb452015-07-30 17:50:14 +03001276static void ib_uverbs_remove_one(struct ib_device *device, void *client_data)
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001277{
Haggai Eran7c1eb452015-07-30 17:50:14 +03001278 struct ib_uverbs_device *uverbs_dev = client_data;
Yishai Hadas036b1062015-08-13 18:32:05 +03001279 int wait_clients = 1;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001280
1281 if (!uverbs_dev)
1282 return;
1283
Tony Jonesf4e91eb2008-02-22 00:13:36 +01001284 dev_set_drvdata(uverbs_dev->dev, NULL);
Alexander Chiang055422d2010-02-02 19:07:49 +00001285 device_destroy(uverbs_class, uverbs_dev->cdev.dev);
1286 cdev_del(&uverbs_dev->cdev);
Roland Dreier70a30e12005-10-28 15:38:26 -07001287
Alexander Chiang6d6a0e72010-02-02 19:08:09 +00001288 if (uverbs_dev->devnum < IB_UVERBS_MAX_DEVICES)
1289 clear_bit(uverbs_dev->devnum, dev_map);
1290 else
1291 clear_bit(uverbs_dev->devnum - IB_UVERBS_MAX_DEVICES, overflow_map);
Jack Morgensteinfd60ae42006-08-03 10:56:42 -07001292
Yishai Hadas036b1062015-08-13 18:32:05 +03001293 if (device->disassociate_ucontext) {
1294 /* We disassociate HW resources and immediately return.
1295 * Userspace will see a EIO errno for all future access.
1296 * Upon returning, ib_device may be freed internally and is not
1297 * valid any more.
1298 * uverbs_device is still available until all clients close
1299 * their files, then the uverbs device ref count will be zero
1300 * and its resources will be freed.
1301 * Note: At this point no more files can be opened since the
1302 * cdev was deleted, however active clients can still issue
1303 * commands and close their open files.
1304 */
1305 rcu_assign_pointer(uverbs_dev->ib_dev, NULL);
1306 ib_uverbs_free_hw_resources(uverbs_dev, device);
1307 wait_clients = 0;
1308 }
1309
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001310 if (atomic_dec_and_test(&uverbs_dev->refcount))
1311 ib_uverbs_comp_dev(uverbs_dev);
Yishai Hadas036b1062015-08-13 18:32:05 +03001312 if (wait_clients)
1313 wait_for_completion(&uverbs_dev->comp);
Yishai Hadas35d4a0b2015-08-13 18:32:03 +03001314 kobject_put(&uverbs_dev->kobj);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001315}
1316
Al Viro2c9ede52011-07-23 20:24:48 -04001317static char *uverbs_devnode(struct device *dev, umode_t *mode)
Roland Dreier71c29bd2011-05-23 11:10:05 -07001318{
Goldwyn Rodriguesb2bc47822011-07-04 09:26:57 -07001319 if (mode)
1320 *mode = 0666;
Roland Dreier71c29bd2011-05-23 11:10:05 -07001321 return kasprintf(GFP_KERNEL, "infiniband/%s", dev_name(dev));
1322}
1323
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001324static int __init ib_uverbs_init(void)
1325{
1326 int ret;
1327
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001328 ret = register_chrdev_region(IB_UVERBS_BASE_DEV, IB_UVERBS_MAX_DEVICES,
1329 "infiniband_verbs");
1330 if (ret) {
Parav Panditaba25a3e2016-03-02 00:50:29 +05301331 pr_err("user_verbs: couldn't register device number\n");
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001332 goto out;
1333 }
1334
Roland Dreier70a30e12005-10-28 15:38:26 -07001335 uverbs_class = class_create(THIS_MODULE, "infiniband_verbs");
1336 if (IS_ERR(uverbs_class)) {
1337 ret = PTR_ERR(uverbs_class);
Parav Panditaba25a3e2016-03-02 00:50:29 +05301338 pr_err("user_verbs: couldn't create class infiniband_verbs\n");
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001339 goto out_chrdev;
1340 }
1341
Roland Dreier71c29bd2011-05-23 11:10:05 -07001342 uverbs_class->devnode = uverbs_devnode;
1343
Andi Kleen0933e2d2010-01-05 12:48:09 +01001344 ret = class_create_file(uverbs_class, &class_attr_abi_version.attr);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001345 if (ret) {
Parav Panditaba25a3e2016-03-02 00:50:29 +05301346 pr_err("user_verbs: couldn't create abi_version attribute\n");
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001347 goto out_class;
1348 }
1349
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001350 ret = ib_register_client(&uverbs_client);
1351 if (ret) {
Parav Panditaba25a3e2016-03-02 00:50:29 +05301352 pr_err("user_verbs: couldn't register client\n");
Roland Dreiera265e552010-02-24 16:51:20 -08001353 goto out_class;
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001354 }
1355
1356 return 0;
1357
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001358out_class:
Roland Dreier70a30e12005-10-28 15:38:26 -07001359 class_destroy(uverbs_class);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001360
1361out_chrdev:
1362 unregister_chrdev_region(IB_UVERBS_BASE_DEV, IB_UVERBS_MAX_DEVICES);
1363
1364out:
1365 return ret;
1366}
1367
1368static void __exit ib_uverbs_cleanup(void)
1369{
1370 ib_unregister_client(&uverbs_client);
Roland Dreier70a30e12005-10-28 15:38:26 -07001371 class_destroy(uverbs_class);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001372 unregister_chrdev_region(IB_UVERBS_BASE_DEV, IB_UVERBS_MAX_DEVICES);
Alexander Chiang6d6a0e72010-02-02 19:08:09 +00001373 if (overflow_maj)
1374 unregister_chrdev_region(overflow_maj, IB_UVERBS_MAX_DEVICES);
Roland Dreier5d7edb32005-10-24 10:53:25 -07001375 idr_destroy(&ib_uverbs_pd_idr);
1376 idr_destroy(&ib_uverbs_mr_idr);
1377 idr_destroy(&ib_uverbs_mw_idr);
1378 idr_destroy(&ib_uverbs_ah_idr);
1379 idr_destroy(&ib_uverbs_cq_idr);
1380 idr_destroy(&ib_uverbs_qp_idr);
1381 idr_destroy(&ib_uverbs_srq_idr);
Roland Dreierbc38a6a2005-07-07 17:57:13 -07001382}
1383
1384module_init(ib_uverbs_init);
1385module_exit(ib_uverbs_cleanup);