user_mad.c 30.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2
/*
 * Copyright (c) 2004 Topspin Communications.  All rights reserved.
R
Roland Dreier 已提交
3
 * Copyright (c) 2005 Voltaire, Inc. All rights reserved.
4
 * Copyright (c) 2005 Sun Microsystems, Inc. All rights reserved.
5
 * Copyright (c) 2008 Cisco. All rights reserved.
L
Linus Torvalds 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
 *
 * This software is available to you under a choice of one of two
 * licenses.  You may choose to be licensed under the terms of the GNU
 * General Public License (GPL) Version 2, available from the file
 * COPYING in the main directory of this source tree, or the
 * OpenIB.org BSD license below:
 *
 *     Redistribution and use in source and binary forms, with or
 *     without modification, are permitted provided that the following
 *     conditions are met:
 *
 *      - Redistributions of source code must retain the above
 *        copyright notice, this list of conditions and the following
 *        disclaimer.
 *
 *      - Redistributions in binary form must reproduce the above
 *        copyright notice, this list of conditions and the following
 *        disclaimer in the documentation and/or other materials
 *        provided with the distribution.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */

#include <linux/module.h>
#include <linux/init.h>
#include <linux/device.h>
#include <linux/err.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <linux/dma-mapping.h>
#include <linux/poll.h>
44
#include <linux/mutex.h>
L
Linus Torvalds 已提交
45
#include <linux/kref.h>
46
#include <linux/compat.h>
47
#include <linux/sched.h>
48
#include <linux/semaphore.h>
L
Linus Torvalds 已提交
49 50 51

#include <asm/uaccess.h>

52 53
#include <rdma/ib_mad.h>
#include <rdma/ib_user_mad.h>
L
Linus Torvalds 已提交
54 55 56 57 58 59 60 61 62 63 64 65 66

MODULE_AUTHOR("Roland Dreier");
MODULE_DESCRIPTION("InfiniBand userspace MAD packet access");
MODULE_LICENSE("Dual BSD/GPL");

enum {
	IB_UMAD_MAX_PORTS  = 64,
	IB_UMAD_MAX_AGENTS = 32,

	IB_UMAD_MAJOR      = 231,
	IB_UMAD_MINOR_BASE = 0
};

67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
/*
 * Our lifetime rules for these structs are the following: each time a
 * device special file is opened, we look up the corresponding struct
 * ib_umad_port by minor in the umad_port[] table while holding the
 * port_lock.  If this lookup succeeds, we take a reference on the
 * ib_umad_port's struct ib_umad_device while still holding the
 * port_lock; if the lookup fails, we fail the open().  We drop these
 * references in the corresponding close().
 *
 * In addition to references coming from open character devices, there
 * is one more reference to each ib_umad_device representing the
 * module's reference taken when allocating the ib_umad_device in
 * ib_umad_add_one().
 *
 * When destroying an ib_umad_device, we clear all of its
 * ib_umad_ports from umad_port[] while holding port_lock before
 * dropping the module's reference to the ib_umad_device.  This is
 * always safe because any open() calls will either succeed and obtain
 * a reference before we clear the umad_port[] entries, or fail after
 * we clear the umad_port[] entries.
 */

L
Linus Torvalds 已提交
89
struct ib_umad_port {
90 91
	struct cdev           *cdev;
	struct device	      *dev;
L
Linus Torvalds 已提交
92

93 94
	struct cdev           *sm_cdev;
	struct device	      *sm_dev;
L
Linus Torvalds 已提交
95 96
	struct semaphore       sm_sem;

97
	struct mutex	       file_mutex;
98 99
	struct list_head       file_list;

L
Linus Torvalds 已提交
100 101
	struct ib_device      *ib_dev;
	struct ib_umad_device *umad_dev;
102
	int                    dev_num;
L
Linus Torvalds 已提交
103 104 105 106 107 108 109 110 111 112
	u8                     port_num;
};

struct ib_umad_device {
	int                  start_port, end_port;
	struct kref          ref;
	struct ib_umad_port  port[0];
};

struct ib_umad_file {
113
	struct mutex		mutex;
114 115
	struct ib_umad_port    *port;
	struct list_head	recv_list;
116
	struct list_head	send_list;
117
	struct list_head	port_list;
118
	spinlock_t		send_lock;
119 120 121
	wait_queue_head_t	recv_wait;
	struct ib_mad_agent    *agent[IB_UMAD_MAX_AGENTS];
	int			agents_dead;
R
Roland Dreier 已提交
122 123
	u8			use_pkey_index;
	u8			already_used;
L
Linus Torvalds 已提交
124 125 126
};

struct ib_umad_packet {
127
	struct ib_mad_send_buf *msg;
128
	struct ib_mad_recv_wc  *recv_wc;
L
Linus Torvalds 已提交
129
	struct list_head   list;
130 131
	int		   length;
	struct ib_user_mad mad;
L
Linus Torvalds 已提交
132 133
};

134 135
static struct class *umad_class;

L
Linus Torvalds 已提交
136
static const dev_t base_dev = MKDEV(IB_UMAD_MAJOR, IB_UMAD_MINOR_BASE);
137 138 139

static DEFINE_SPINLOCK(port_lock);
static struct ib_umad_port *umad_port[IB_UMAD_MAX_PORTS];
140
static DECLARE_BITMAP(dev_map, IB_UMAD_MAX_PORTS);
L
Linus Torvalds 已提交
141 142 143 144

static void ib_umad_add_one(struct ib_device *device);
static void ib_umad_remove_one(struct ib_device *device);

145 146 147 148 149 150 151 152
static void ib_umad_release_dev(struct kref *ref)
{
	struct ib_umad_device *dev =
		container_of(ref, struct ib_umad_device, ref);

	kfree(dev);
}

R
Roland Dreier 已提交
153 154 155 156 157 158
static int hdr_size(struct ib_umad_file *file)
{
	return file->use_pkey_index ? sizeof (struct ib_user_mad_hdr) :
		sizeof (struct ib_user_mad_hdr_old);
}

159
/* caller must hold file->mutex */
160 161 162 163 164
static struct ib_mad_agent *__get_agent(struct ib_umad_file *file, int id)
{
	return file->agents_dead ? NULL : file->agent[id];
}

L
Linus Torvalds 已提交
165 166 167 168 169 170
static int queue_packet(struct ib_umad_file *file,
			struct ib_mad_agent *agent,
			struct ib_umad_packet *packet)
{
	int ret = 1;

171
	mutex_lock(&file->mutex);
172

173 174 175
	for (packet->mad.hdr.id = 0;
	     packet->mad.hdr.id < IB_UMAD_MAX_AGENTS;
	     packet->mad.hdr.id++)
176
		if (agent == __get_agent(file, packet->mad.hdr.id)) {
L
Linus Torvalds 已提交
177 178 179 180 181 182
			list_add_tail(&packet->list, &file->recv_list);
			wake_up_interruptible(&file->recv_wait);
			ret = 0;
			break;
		}

183
	mutex_unlock(&file->mutex);
L
Linus Torvalds 已提交
184 185 186 187

	return ret;
}

188 189
static void dequeue_send(struct ib_umad_file *file,
			 struct ib_umad_packet *packet)
190
{
191 192 193
	spin_lock_irq(&file->send_lock);
	list_del(&packet->list);
	spin_unlock_irq(&file->send_lock);
194
}
195

L
Linus Torvalds 已提交
196 197 198 199
static void send_handler(struct ib_mad_agent *agent,
			 struct ib_mad_send_wc *send_wc)
{
	struct ib_umad_file *file = agent->context;
200
	struct ib_umad_packet *packet = send_wc->send_buf->context[0];
L
Linus Torvalds 已提交
201

202
	dequeue_send(file, packet);
203
	ib_destroy_ah(packet->msg->ah);
204
	ib_free_send_mad(packet->msg);
L
Linus Torvalds 已提交
205 206

	if (send_wc->status == IB_WC_RESP_TIMEOUT_ERR) {
207 208 209 210
		packet->length = IB_MGMT_MAD_HDR;
		packet->mad.hdr.status = ETIMEDOUT;
		if (!queue_packet(file, agent, packet))
			return;
211
	}
L
Linus Torvalds 已提交
212 213 214 215 216 217 218 219 220 221
	kfree(packet);
}

static void recv_handler(struct ib_mad_agent *agent,
			 struct ib_mad_recv_wc *mad_recv_wc)
{
	struct ib_umad_file *file = agent->context;
	struct ib_umad_packet *packet;

	if (mad_recv_wc->wc->status != IB_WC_SUCCESS)
222
		goto err1;
L
Linus Torvalds 已提交
223

224
	packet = kzalloc(sizeof *packet, GFP_KERNEL);
L
Linus Torvalds 已提交
225
	if (!packet)
226
		goto err1;
L
Linus Torvalds 已提交
227

228 229
	packet->length = mad_recv_wc->mad_len;
	packet->recv_wc = mad_recv_wc;
L
Linus Torvalds 已提交
230

R
Roland Dreier 已提交
231 232 233 234 235 236 237
	packet->mad.hdr.status	   = 0;
	packet->mad.hdr.length	   = hdr_size(file) + mad_recv_wc->mad_len;
	packet->mad.hdr.qpn	   = cpu_to_be32(mad_recv_wc->wc->src_qp);
	packet->mad.hdr.lid	   = cpu_to_be16(mad_recv_wc->wc->slid);
	packet->mad.hdr.sl	   = mad_recv_wc->wc->sl;
	packet->mad.hdr.path_bits  = mad_recv_wc->wc->dlid_path_bits;
	packet->mad.hdr.pkey_index = mad_recv_wc->wc->pkey_index;
238 239
	packet->mad.hdr.grh_present = !!(mad_recv_wc->wc->wc_flags & IB_WC_GRH);
	if (packet->mad.hdr.grh_present) {
240 241 242 243 244 245 246 247 248 249 250
		struct ib_ah_attr ah_attr;

		ib_init_ah_from_wc(agent->device, agent->port_num,
				   mad_recv_wc->wc, mad_recv_wc->recv_buf.grh,
				   &ah_attr);

		packet->mad.hdr.gid_index = ah_attr.grh.sgid_index;
		packet->mad.hdr.hop_limit = ah_attr.grh.hop_limit;
		packet->mad.hdr.traffic_class = ah_attr.grh.traffic_class;
		memcpy(packet->mad.hdr.gid, &ah_attr.grh.dgid, 16);
		packet->mad.hdr.flow_label = cpu_to_be32(ah_attr.grh.flow_label);
L
Linus Torvalds 已提交
251 252 253
	}

	if (queue_packet(file, agent, packet))
254 255
		goto err2;
	return;
L
Linus Torvalds 已提交
256

257 258 259
err2:
	kfree(packet);
err1:
L
Linus Torvalds 已提交
260 261 262
	ib_free_recv_mad(mad_recv_wc);
}

R
Roland Dreier 已提交
263 264
static ssize_t copy_recv_mad(struct ib_umad_file *file, char __user *buf,
			     struct ib_umad_packet *packet, size_t count)
265 266 267 268 269 270 271
{
	struct ib_mad_recv_buf *recv_buf;
	int left, seg_payload, offset, max_seg_payload;

	/* We need enough room to copy the first (or only) MAD segment. */
	recv_buf = &packet->recv_wc->recv_buf;
	if ((packet->length <= sizeof (*recv_buf->mad) &&
R
Roland Dreier 已提交
272
	     count < hdr_size(file) + packet->length) ||
273
	    (packet->length > sizeof (*recv_buf->mad) &&
R
Roland Dreier 已提交
274
	     count < hdr_size(file) + sizeof (*recv_buf->mad)))
275 276
		return -EINVAL;

R
Roland Dreier 已提交
277
	if (copy_to_user(buf, &packet->mad, hdr_size(file)))
278 279
		return -EFAULT;

R
Roland Dreier 已提交
280
	buf += hdr_size(file);
281 282 283 284 285 286 287 288 289
	seg_payload = min_t(int, packet->length, sizeof (*recv_buf->mad));
	if (copy_to_user(buf, recv_buf->mad, seg_payload))
		return -EFAULT;

	if (seg_payload < packet->length) {
		/*
		 * Multipacket RMPP MAD message. Copy remainder of message.
		 * Note that last segment may have a shorter payload.
		 */
R
Roland Dreier 已提交
290
		if (count < hdr_size(file) + packet->length) {
291 292 293 294 295 296
			/*
			 * The buffer is too small, return the first RMPP segment,
			 * which includes the RMPP message length.
			 */
			return -ENOSPC;
		}
297
		offset = ib_get_mad_data_offset(recv_buf->mad->mad_hdr.mgmt_class);
298 299 300 301 302 303 304 305 306 307 308 309
		max_seg_payload = sizeof (struct ib_mad) - offset;

		for (left = packet->length - seg_payload, buf += seg_payload;
		     left; left -= seg_payload, buf += seg_payload) {
			recv_buf = container_of(recv_buf->list.next,
						struct ib_mad_recv_buf, list);
			seg_payload = min(left, max_seg_payload);
			if (copy_to_user(buf, ((void *) recv_buf->mad) + offset,
					 seg_payload))
				return -EFAULT;
		}
	}
R
Roland Dreier 已提交
310
	return hdr_size(file) + packet->length;
311 312
}

R
Roland Dreier 已提交
313 314
static ssize_t copy_send_mad(struct ib_umad_file *file, char __user *buf,
			     struct ib_umad_packet *packet, size_t count)
315
{
R
Roland Dreier 已提交
316
	ssize_t size = hdr_size(file) + packet->length;
317 318 319 320

	if (count < size)
		return -EINVAL;

R
Roland Dreier 已提交
321 322 323 324 325 326
	if (copy_to_user(buf, &packet->mad, hdr_size(file)))
		return -EFAULT;

	buf += hdr_size(file);

	if (copy_to_user(buf, packet->mad.data, packet->length))
327 328 329 330 331
		return -EFAULT;

	return size;
}

L
Linus Torvalds 已提交
332 333 334 335 336 337 338
static ssize_t ib_umad_read(struct file *filp, char __user *buf,
			    size_t count, loff_t *pos)
{
	struct ib_umad_file *file = filp->private_data;
	struct ib_umad_packet *packet;
	ssize_t ret;

R
Roland Dreier 已提交
339
	if (count < hdr_size(file))
L
Linus Torvalds 已提交
340 341
		return -EINVAL;

342
	mutex_lock(&file->mutex);
L
Linus Torvalds 已提交
343 344

	while (list_empty(&file->recv_list)) {
345
		mutex_unlock(&file->mutex);
L
Linus Torvalds 已提交
346 347 348 349 350 351 352 353

		if (filp->f_flags & O_NONBLOCK)
			return -EAGAIN;

		if (wait_event_interruptible(file->recv_wait,
					     !list_empty(&file->recv_list)))
			return -ERESTARTSYS;

354
		mutex_lock(&file->mutex);
L
Linus Torvalds 已提交
355 356 357 358 359
	}

	packet = list_entry(file->recv_list.next, struct ib_umad_packet, list);
	list_del(&packet->list);

360
	mutex_unlock(&file->mutex);
L
Linus Torvalds 已提交
361

362
	if (packet->recv_wc)
R
Roland Dreier 已提交
363
		ret = copy_recv_mad(file, buf, packet, count);
L
Linus Torvalds 已提交
364
	else
R
Roland Dreier 已提交
365
		ret = copy_send_mad(file, buf, packet, count);
366

367 368
	if (ret < 0) {
		/* Requeue packet */
369
		mutex_lock(&file->mutex);
370
		list_add(&packet->list, &file->recv_list);
371
		mutex_unlock(&file->mutex);
372 373 374
	} else {
		if (packet->recv_wc)
			ib_free_recv_mad(packet->recv_wc);
375
		kfree(packet);
376
	}
L
Linus Torvalds 已提交
377 378 379
	return ret;
}

380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
static int copy_rmpp_mad(struct ib_mad_send_buf *msg, const char __user *buf)
{
	int left, seg;

	/* Copy class specific header */
	if ((msg->hdr_len > IB_MGMT_RMPP_HDR) &&
	    copy_from_user(msg->mad + IB_MGMT_RMPP_HDR, buf + IB_MGMT_RMPP_HDR,
			   msg->hdr_len - IB_MGMT_RMPP_HDR))
		return -EFAULT;

	/* All headers are in place.  Copy data segments. */
	for (seg = 1, left = msg->data_len, buf += msg->hdr_len; left > 0;
	     seg++, left -= msg->seg_size, buf += msg->seg_size) {
		if (copy_from_user(ib_get_rmpp_segment(msg, seg), buf,
				   min(left, msg->seg_size)))
			return -EFAULT;
	}
	return 0;
}

400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444
static int same_destination(struct ib_user_mad_hdr *hdr1,
			    struct ib_user_mad_hdr *hdr2)
{
	if (!hdr1->grh_present && !hdr2->grh_present)
	   return (hdr1->lid == hdr2->lid);

	if (hdr1->grh_present && hdr2->grh_present)
	   return !memcmp(hdr1->gid, hdr2->gid, 16);

	return 0;
}

static int is_duplicate(struct ib_umad_file *file,
			struct ib_umad_packet *packet)
{
	struct ib_umad_packet *sent_packet;
	struct ib_mad_hdr *sent_hdr, *hdr;

	hdr = (struct ib_mad_hdr *) packet->mad.data;
	list_for_each_entry(sent_packet, &file->send_list, list) {
		sent_hdr = (struct ib_mad_hdr *) sent_packet->mad.data;

		if ((hdr->tid != sent_hdr->tid) ||
		    (hdr->mgmt_class != sent_hdr->mgmt_class))
			continue;

		/*
		 * No need to be overly clever here.  If two new operations have
		 * the same TID, reject the second as a duplicate.  This is more
		 * restrictive than required by the spec.
		 */
		if (!ib_response_mad((struct ib_mad *) hdr)) {
			if (!ib_response_mad((struct ib_mad *) sent_hdr))
				return 1;
			continue;
		} else if (!ib_response_mad((struct ib_mad *) sent_hdr))
			continue;

		if (same_destination(&packet->mad.hdr, &sent_packet->mad.hdr))
			return 1;
	}

	return 0;
}

L
Linus Torvalds 已提交
445 446 447 448 449 450 451
static ssize_t ib_umad_write(struct file *filp, const char __user *buf,
			     size_t count, loff_t *pos)
{
	struct ib_umad_file *file = filp->private_data;
	struct ib_umad_packet *packet;
	struct ib_mad_agent *agent;
	struct ib_ah_attr ah_attr;
452
	struct ib_ah *ah;
453
	struct ib_rmpp_mad *rmpp_mad;
454
	__be64 *tid;
455
	int ret, data_len, hdr_len, copy_offset, rmpp_active;
L
Linus Torvalds 已提交
456

R
Roland Dreier 已提交
457
	if (count < hdr_size(file) + IB_MGMT_RMPP_HDR)
L
Linus Torvalds 已提交
458 459
		return -EINVAL;

460
	packet = kzalloc(sizeof *packet + IB_MGMT_RMPP_HDR, GFP_KERNEL);
L
Linus Torvalds 已提交
461 462 463
	if (!packet)
		return -ENOMEM;

R
Roland Dreier 已提交
464
	if (copy_from_user(&packet->mad, buf, hdr_size(file))) {
465 466
		ret = -EFAULT;
		goto err;
L
Linus Torvalds 已提交
467 468
	}

469 470
	if (packet->mad.hdr.id < 0 ||
	    packet->mad.hdr.id >= IB_UMAD_MAX_AGENTS) {
L
Linus Torvalds 已提交
471 472 473 474
		ret = -EINVAL;
		goto err;
	}

R
Roland Dreier 已提交
475 476 477 478 479 480 481
	buf += hdr_size(file);

	if (copy_from_user(packet->mad.data, buf, IB_MGMT_RMPP_HDR)) {
		ret = -EFAULT;
		goto err;
	}

482
	mutex_lock(&file->mutex);
L
Linus Torvalds 已提交
483

484
	agent = __get_agent(file, packet->mad.hdr.id);
L
Linus Torvalds 已提交
485 486 487 488 489 490
	if (!agent) {
		ret = -EINVAL;
		goto err_up;
	}

	memset(&ah_attr, 0, sizeof ah_attr);
491 492 493
	ah_attr.dlid          = be16_to_cpu(packet->mad.hdr.lid);
	ah_attr.sl            = packet->mad.hdr.sl;
	ah_attr.src_path_bits = packet->mad.hdr.path_bits;
L
Linus Torvalds 已提交
494
	ah_attr.port_num      = file->port->port_num;
495
	if (packet->mad.hdr.grh_present) {
L
Linus Torvalds 已提交
496
		ah_attr.ah_flags = IB_AH_GRH;
497
		memcpy(ah_attr.grh.dgid.raw, packet->mad.hdr.gid, 16);
498
		ah_attr.grh.sgid_index	   = packet->mad.hdr.gid_index;
499
		ah_attr.grh.flow_label 	   = be32_to_cpu(packet->mad.hdr.flow_label);
500 501
		ah_attr.grh.hop_limit  	   = packet->mad.hdr.hop_limit;
		ah_attr.grh.traffic_class  = packet->mad.hdr.traffic_class;
L
Linus Torvalds 已提交
502 503
	}

504 505 506
	ah = ib_create_ah(agent->qp->pd, &ah_attr);
	if (IS_ERR(ah)) {
		ret = PTR_ERR(ah);
L
Linus Torvalds 已提交
507 508 509
		goto err_up;
	}

510
	rmpp_mad = (struct ib_rmpp_mad *) packet->mad.data;
511 512 513 514 515
	hdr_len = ib_get_mad_data_offset(rmpp_mad->mad_hdr.mgmt_class);
	if (!ib_is_mad_class_rmpp(rmpp_mad->mad_hdr.mgmt_class)) {
		copy_offset = IB_MGMT_MAD_HDR;
		rmpp_active = 0;
	} else {
516 517 518
		copy_offset = IB_MGMT_RMPP_HDR;
		rmpp_active = ib_get_rmpp_flags(&rmpp_mad->rmpp_hdr) &
			      IB_MGMT_RMPP_FLAG_ACTIVE;
519 520
	}

R
Roland Dreier 已提交
521
	data_len = count - hdr_size(file) - hdr_len;
522 523
	packet->msg = ib_create_send_mad(agent,
					 be32_to_cpu(packet->mad.hdr.qpn),
R
Roland Dreier 已提交
524 525
					 packet->mad.hdr.pkey_index, rmpp_active,
					 hdr_len, data_len, GFP_KERNEL);
526 527 528 529
	if (IS_ERR(packet->msg)) {
		ret = PTR_ERR(packet->msg);
		goto err_ah;
	}
L
Linus Torvalds 已提交
530

531 532 533 534
	packet->msg->ah 	= ah;
	packet->msg->timeout_ms = packet->mad.hdr.timeout_ms;
	packet->msg->retries 	= packet->mad.hdr.retries;
	packet->msg->context[0] = packet;
L
Linus Torvalds 已提交
535

536
	/* Copy MAD header.  Any RMPP header is already in place. */
S
Sean Hefty 已提交
537
	memcpy(packet->msg->mad, packet->mad.data, IB_MGMT_MAD_HDR);
538 539 540 541 542 543 544 545 546 547 548 549

	if (!rmpp_active) {
		if (copy_from_user(packet->msg->mad + copy_offset,
				   buf + copy_offset,
				   hdr_len + data_len - copy_offset)) {
			ret = -EFAULT;
			goto err_msg;
		}
	} else {
		ret = copy_rmpp_mad(packet->msg, buf);
		if (ret)
			goto err_msg;
550 551 552
	}

	/*
553 554 555
	 * Set the high-order part of the transaction ID to make MADs from
	 * different agents unique, and allow routing responses back to the
	 * original requestor.
556
	 */
557
	if (!ib_response_mad(packet->msg->mad)) {
558
		tid = &((struct ib_mad_hdr *) packet->msg->mad)->tid;
559 560
		*tid = cpu_to_be64(((u64) agent->hi_tid) << 32 |
				   (be64_to_cpup(tid) & 0xffffffff));
561 562 563 564 565 566 567 568 569 570 571
		rmpp_mad->mad_hdr.tid = *tid;
	}

	spin_lock_irq(&file->send_lock);
	ret = is_duplicate(file, packet);
	if (!ret)
		list_add_tail(&packet->list, &file->send_list);
	spin_unlock_irq(&file->send_lock);
	if (ret) {
		ret = -EINVAL;
		goto err_msg;
L
Linus Torvalds 已提交
572 573
	}

574
	ret = ib_post_send_mad(packet->msg, NULL);
575
	if (ret)
576
		goto err_send;
577

578
	mutex_unlock(&file->mutex);
S
Sean Hefty 已提交
579
	return count;
580

581 582
err_send:
	dequeue_send(file, packet);
583 584 585
err_msg:
	ib_free_send_mad(packet->msg);
err_ah:
586
	ib_destroy_ah(ah);
L
Linus Torvalds 已提交
587
err_up:
588
	mutex_unlock(&file->mutex);
L
Linus Torvalds 已提交
589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608
err:
	kfree(packet);
	return ret;
}

static unsigned int ib_umad_poll(struct file *filp, struct poll_table_struct *wait)
{
	struct ib_umad_file *file = filp->private_data;

	/* we will always be able to post a MAD send */
	unsigned int mask = POLLOUT | POLLWRNORM;

	poll_wait(filp, &file->recv_wait, wait);

	if (!list_empty(&file->recv_list))
		mask |= POLLIN | POLLRDNORM;

	return mask;
}

609 610
static int ib_umad_reg_agent(struct ib_umad_file *file, void __user *arg,
			     int compat_method_mask)
L
Linus Torvalds 已提交
611 612 613
{
	struct ib_user_mad_reg_req ureq;
	struct ib_mad_reg_req req;
614
	struct ib_mad_agent *agent = NULL;
L
Linus Torvalds 已提交
615 616 617
	int agent_id;
	int ret;

618 619
	mutex_lock(&file->port->file_mutex);
	mutex_lock(&file->mutex);
620 621 622 623 624

	if (!file->port->ib_dev) {
		ret = -EPIPE;
		goto out;
	}
L
Linus Torvalds 已提交
625

626
	if (copy_from_user(&ureq, arg, sizeof ureq)) {
L
Linus Torvalds 已提交
627 628 629 630 631 632 633 634 635 636
		ret = -EFAULT;
		goto out;
	}

	if (ureq.qpn != 0 && ureq.qpn != 1) {
		ret = -EINVAL;
		goto out;
	}

	for (agent_id = 0; agent_id < IB_UMAD_MAX_AGENTS; ++agent_id)
637
		if (!__get_agent(file, agent_id))
L
Linus Torvalds 已提交
638 639 640 641 642 643
			goto found;

	ret = -ENOMEM;
	goto out;

found:
644 645 646
	if (ureq.mgmt_class) {
		req.mgmt_class         = ureq.mgmt_class;
		req.mgmt_class_version = ureq.mgmt_class_version;
647 648 649 650 651 652 653 654 655 656 657 658
		memcpy(req.oui, ureq.oui, sizeof req.oui);

		if (compat_method_mask) {
			u32 *umm = (u32 *) ureq.method_mask;
			int i;

			for (i = 0; i < BITS_TO_LONGS(IB_MGMT_MAX_METHODS); ++i)
				req.method_mask[i] =
					umm[i * 2] | ((u64) umm[i * 2 + 1] << 32);
		} else
			memcpy(req.method_mask, ureq.method_mask,
			       sizeof req.method_mask);
659
	}
L
Linus Torvalds 已提交
660 661 662

	agent = ib_register_mad_agent(file->port->ib_dev, file->port->port_num,
				      ureq.qpn ? IB_QPT_GSI : IB_QPT_SMI,
663
				      ureq.mgmt_class ? &req : NULL,
664 665
				      ureq.rmpp_version,
				      send_handler, recv_handler, file);
L
Linus Torvalds 已提交
666 667
	if (IS_ERR(agent)) {
		ret = PTR_ERR(agent);
668
		agent = NULL;
L
Linus Torvalds 已提交
669 670 671 672 673 674
		goto out;
	}

	if (put_user(agent_id,
		     (u32 __user *) (arg + offsetof(struct ib_user_mad_reg_req, id)))) {
		ret = -EFAULT;
675
		goto out;
L
Linus Torvalds 已提交
676 677
	}

R
Roland Dreier 已提交
678 679 680 681 682 683 684 685 686 687
	if (!file->already_used) {
		file->already_used = 1;
		if (!file->use_pkey_index) {
			printk(KERN_WARNING "user_mad: process %s did not enable "
			       "P_Key index support.\n", current->comm);
			printk(KERN_WARNING "user_mad:   Documentation/infiniband/user_mad.txt "
			       "has info on the new ABI.\n");
		}
	}

688
	file->agent[agent_id] = agent;
L
Linus Torvalds 已提交
689
	ret = 0;
690

L
Linus Torvalds 已提交
691
out:
692 693 694 695 696 697 698
	mutex_unlock(&file->mutex);

	if (ret && agent)
		ib_unregister_mad_agent(agent);

	mutex_unlock(&file->port->file_mutex);

L
Linus Torvalds 已提交
699 700 701
	return ret;
}

702
static int ib_umad_unreg_agent(struct ib_umad_file *file, u32 __user *arg)
L
Linus Torvalds 已提交
703
{
704
	struct ib_mad_agent *agent = NULL;
L
Linus Torvalds 已提交
705 706 707
	u32 id;
	int ret = 0;

708
	if (get_user(id, arg))
709
		return -EFAULT;
L
Linus Torvalds 已提交
710

711 712
	mutex_lock(&file->port->file_mutex);
	mutex_lock(&file->mutex);
L
Linus Torvalds 已提交
713

714
	if (id < 0 || id >= IB_UMAD_MAX_AGENTS || !__get_agent(file, id)) {
L
Linus Torvalds 已提交
715 716 717 718
		ret = -EINVAL;
		goto out;
	}

719
	agent = file->agent[id];
L
Linus Torvalds 已提交
720 721 722
	file->agent[id] = NULL;

out:
723
	mutex_unlock(&file->mutex);
724

725
	if (agent)
726 727
		ib_unregister_mad_agent(agent);

728 729
	mutex_unlock(&file->port->file_mutex);

L
Linus Torvalds 已提交
730 731 732
	return ret;
}

R
Roland Dreier 已提交
733 734 735 736
static long ib_umad_enable_pkey(struct ib_umad_file *file)
{
	int ret = 0;

737
	mutex_lock(&file->mutex);
R
Roland Dreier 已提交
738 739 740 741
	if (file->already_used)
		ret = -EINVAL;
	else
		file->use_pkey_index = 1;
742
	mutex_unlock(&file->mutex);
R
Roland Dreier 已提交
743 744 745 746

	return ret;
}

747 748
static long ib_umad_ioctl(struct file *filp, unsigned int cmd,
			  unsigned long arg)
L
Linus Torvalds 已提交
749 750 751
{
	switch (cmd) {
	case IB_USER_MAD_REGISTER_AGENT:
752
		return ib_umad_reg_agent(filp->private_data, (void __user *) arg, 0);
L
Linus Torvalds 已提交
753
	case IB_USER_MAD_UNREGISTER_AGENT:
754
		return ib_umad_unreg_agent(filp->private_data, (__u32 __user *) arg);
R
Roland Dreier 已提交
755 756
	case IB_USER_MAD_ENABLE_PKEY:
		return ib_umad_enable_pkey(filp->private_data);
L
Linus Torvalds 已提交
757 758 759 760 761
	default:
		return -ENOIOCTLCMD;
	}
}

762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778
#ifdef CONFIG_COMPAT
static long ib_umad_compat_ioctl(struct file *filp, unsigned int cmd,
				 unsigned long arg)
{
	switch (cmd) {
	case IB_USER_MAD_REGISTER_AGENT:
		return ib_umad_reg_agent(filp->private_data, compat_ptr(arg), 1);
	case IB_USER_MAD_UNREGISTER_AGENT:
		return ib_umad_unreg_agent(filp->private_data, compat_ptr(arg));
	case IB_USER_MAD_ENABLE_PKEY:
		return ib_umad_enable_pkey(filp->private_data);
	default:
		return -ENOIOCTLCMD;
	}
}
#endif

779 780 781 782 783 784 785 786 787 788 789 790 791
/*
 * ib_umad_open() does not need the BKL:
 *
 *  - umad_port[] accesses are protected by port_lock, the
 *    ib_umad_port structures are properly reference counted, and
 *    everything else is purely local to the file being created, so
 *    races against other open calls are not a problem;
 *  - the ioctl method does not affect any global state outside of the
 *    file structure being operated on;
 *  - the port is added to umad_port[] as the last part of module
 *    initialization so the open method will either immediately run
 *    -ENXIO, or all required initialization will be done.
 */
L
Linus Torvalds 已提交
792 793
static int ib_umad_open(struct inode *inode, struct file *filp)
{
794
	struct ib_umad_port *port;
L
Linus Torvalds 已提交
795
	struct ib_umad_file *file;
796
	int ret = 0;
L
Linus Torvalds 已提交
797

798 799 800 801 802 803
	spin_lock(&port_lock);
	port = umad_port[iminor(inode) - IB_UMAD_MINOR_BASE];
	if (port)
		kref_get(&port->umad_dev->ref);
	spin_unlock(&port_lock);

804
	if (!port)
805 806
		return -ENXIO;

807
	mutex_lock(&port->file_mutex);
808 809 810 811 812 813

	if (!port->ib_dev) {
		ret = -ENXIO;
		goto out;
	}

S
Sean Hefty 已提交
814
	file = kzalloc(sizeof *file, GFP_KERNEL);
815 816
	if (!file) {
		kref_put(&port->umad_dev->ref, ib_umad_release_dev);
817 818
		ret = -ENOMEM;
		goto out;
819
	}
L
Linus Torvalds 已提交
820

821
	mutex_init(&file->mutex);
822
	spin_lock_init(&file->send_lock);
L
Linus Torvalds 已提交
823
	INIT_LIST_HEAD(&file->recv_list);
824
	INIT_LIST_HEAD(&file->send_list);
L
Linus Torvalds 已提交
825 826 827 828 829
	init_waitqueue_head(&file->recv_wait);

	file->port = port;
	filp->private_data = file;

830 831 832
	list_add_tail(&file->port_list, &port->file_list);

out:
833
	mutex_unlock(&port->file_mutex);
834
	return ret;
L
Linus Torvalds 已提交
835 836 837 838 839
}

static int ib_umad_close(struct inode *inode, struct file *filp)
{
	struct ib_umad_file *file = filp->private_data;
840
	struct ib_umad_device *dev = file->port->umad_dev;
841
	struct ib_umad_packet *packet, *tmp;
842
	int already_dead;
L
Linus Torvalds 已提交
843 844
	int i;

845 846
	mutex_lock(&file->port->file_mutex);
	mutex_lock(&file->mutex);
847 848 849

	already_dead = file->agents_dead;
	file->agents_dead = 1;
L
Linus Torvalds 已提交
850

851 852 853
	list_for_each_entry_safe(packet, tmp, &file->recv_list, list) {
		if (packet->recv_wc)
			ib_free_recv_mad(packet->recv_wc);
854
		kfree(packet);
855
	}
856

857 858
	list_del(&file->port_list);

859
	mutex_unlock(&file->mutex);
860 861 862 863 864

	if (!already_dead)
		for (i = 0; i < IB_UMAD_MAX_AGENTS; ++i)
			if (file->agent[i])
				ib_unregister_mad_agent(file->agent[i]);
L
Linus Torvalds 已提交
865

866
	mutex_unlock(&file->port->file_mutex);
867 868

	kfree(file);
869 870
	kref_put(&dev->ref, ib_umad_release_dev);

L
Linus Torvalds 已提交
871 872 873
	return 0;
}

874
static const struct file_operations umad_fops = {
875 876 877 878
	.owner 	 	= THIS_MODULE,
	.read 	 	= ib_umad_read,
	.write 	 	= ib_umad_write,
	.poll 	 	= ib_umad_poll,
L
Linus Torvalds 已提交
879
	.unlocked_ioctl = ib_umad_ioctl,
880 881 882
#ifdef CONFIG_COMPAT
	.compat_ioctl 	= ib_umad_compat_ioctl,
#endif
883 884
	.open 	 	= ib_umad_open,
	.release 	= ib_umad_close
L
Linus Torvalds 已提交
885 886 887 888
};

static int ib_umad_sm_open(struct inode *inode, struct file *filp)
{
889
	struct ib_umad_port *port;
L
Linus Torvalds 已提交
890 891 892 893 894
	struct ib_port_modify props = {
		.set_port_cap_mask = IB_PORT_SM
	};
	int ret;

895 896 897 898 899 900 901 902 903
	spin_lock(&port_lock);
	port = umad_port[iminor(inode) - IB_UMAD_MINOR_BASE - IB_UMAD_MAX_PORTS];
	if (port)
		kref_get(&port->umad_dev->ref);
	spin_unlock(&port_lock);

	if (!port)
		return -ENXIO;

L
Linus Torvalds 已提交
904
	if (filp->f_flags & O_NONBLOCK) {
905 906 907 908
		if (down_trylock(&port->sm_sem)) {
			ret = -EAGAIN;
			goto fail;
		}
L
Linus Torvalds 已提交
909
	} else {
910 911 912 913
		if (down_interruptible(&port->sm_sem)) {
			ret = -ERESTARTSYS;
			goto fail;
		}
L
Linus Torvalds 已提交
914 915 916 917 918
	}

	ret = ib_modify_port(port->ib_dev, port->port_num, 0, &props);
	if (ret) {
		up(&port->sm_sem);
919
		goto fail;
L
Linus Torvalds 已提交
920 921 922 923 924
	}

	filp->private_data = port;

	return 0;
925 926 927 928

fail:
	kref_put(&port->umad_dev->ref, ib_umad_release_dev);
	return ret;
L
Linus Torvalds 已提交
929 930 931 932 933 934 935 936
}

static int ib_umad_sm_close(struct inode *inode, struct file *filp)
{
	struct ib_umad_port *port = filp->private_data;
	struct ib_port_modify props = {
		.clr_port_cap_mask = IB_PORT_SM
	};
937 938
	int ret = 0;

939
	mutex_lock(&port->file_mutex);
940 941
	if (port->ib_dev)
		ret = ib_modify_port(port->ib_dev, port->port_num, 0, &props);
942
	mutex_unlock(&port->file_mutex);
L
Linus Torvalds 已提交
943 944 945

	up(&port->sm_sem);

946 947
	kref_put(&port->umad_dev->ref, ib_umad_release_dev);

L
Linus Torvalds 已提交
948 949 950
	return ret;
}

951
static const struct file_operations umad_sm_fops = {
L
Linus Torvalds 已提交
952 953 954 955 956 957 958 959 960 961 962
	.owner 	 = THIS_MODULE,
	.open 	 = ib_umad_sm_open,
	.release = ib_umad_sm_close
};

static struct ib_client umad_client = {
	.name   = "umad",
	.add    = ib_umad_add_one,
	.remove = ib_umad_remove_one
};

963 964
static ssize_t show_ibdev(struct device *dev, struct device_attribute *attr,
			  char *buf)
L
Linus Torvalds 已提交
965
{
966
	struct ib_umad_port *port = dev_get_drvdata(dev);
L
Linus Torvalds 已提交
967

968 969 970
	if (!port)
		return -ENODEV;

L
Linus Torvalds 已提交
971 972
	return sprintf(buf, "%s\n", port->ib_dev->name);
}
973
static DEVICE_ATTR(ibdev, S_IRUGO, show_ibdev, NULL);
L
Linus Torvalds 已提交
974

975 976
static ssize_t show_port(struct device *dev, struct device_attribute *attr,
			 char *buf)
L
Linus Torvalds 已提交
977
{
978
	struct ib_umad_port *port = dev_get_drvdata(dev);
L
Linus Torvalds 已提交
979

980 981 982
	if (!port)
		return -ENODEV;

L
Linus Torvalds 已提交
983 984
	return sprintf(buf, "%d\n", port->port_num);
}
985
static DEVICE_ATTR(port, S_IRUGO, show_port, NULL);
L
Linus Torvalds 已提交
986 987 988 989 990 991 992 993 994 995

static ssize_t show_abi_version(struct class *class, char *buf)
{
	return sprintf(buf, "%d\n", IB_USER_MAD_ABI_VERSION);
}
static CLASS_ATTR(abi_version, S_IRUGO, show_abi_version, NULL);

static int ib_umad_init_port(struct ib_device *device, int port_num,
			     struct ib_umad_port *port)
{
996 997 998 999
	spin_lock(&port_lock);
	port->dev_num = find_first_zero_bit(dev_map, IB_UMAD_MAX_PORTS);
	if (port->dev_num >= IB_UMAD_MAX_PORTS) {
		spin_unlock(&port_lock);
L
Linus Torvalds 已提交
1000 1001
		return -1;
	}
1002 1003
	set_bit(port->dev_num, dev_map);
	spin_unlock(&port_lock);
L
Linus Torvalds 已提交
1004 1005 1006 1007

	port->ib_dev   = device;
	port->port_num = port_num;
	init_MUTEX(&port->sm_sem);
1008
	mutex_init(&port->file_mutex);
1009
	INIT_LIST_HEAD(&port->file_list);
L
Linus Torvalds 已提交
1010

1011 1012
	port->cdev = cdev_alloc();
	if (!port->cdev)
L
Linus Torvalds 已提交
1013
		return -1;
1014 1015 1016 1017
	port->cdev->owner = THIS_MODULE;
	port->cdev->ops   = &umad_fops;
	kobject_set_name(&port->cdev->kobj, "umad%d", port->dev_num);
	if (cdev_add(port->cdev, base_dev + port->dev_num, 1))
L
Linus Torvalds 已提交
1018 1019
		goto err_cdev;

1020 1021 1022
	port->dev = device_create(umad_class, device->dma_device,
				  port->cdev->dev, port,
				  "umad%d", port->dev_num);
1023
	if (IS_ERR(port->dev))
1024
		goto err_cdev;
L
Linus Torvalds 已提交
1025

1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037
	if (device_create_file(port->dev, &dev_attr_ibdev))
		goto err_dev;
	if (device_create_file(port->dev, &dev_attr_port))
		goto err_dev;

	port->sm_cdev = cdev_alloc();
	if (!port->sm_cdev)
		goto err_dev;
	port->sm_cdev->owner = THIS_MODULE;
	port->sm_cdev->ops   = &umad_sm_fops;
	kobject_set_name(&port->sm_cdev->kobj, "issm%d", port->dev_num);
	if (cdev_add(port->sm_cdev, base_dev + port->dev_num + IB_UMAD_MAX_PORTS, 1))
1038
		goto err_sm_cdev;
L
Linus Torvalds 已提交
1039

1040 1041 1042
	port->sm_dev = device_create(umad_class, device->dma_device,
				     port->sm_cdev->dev, port,
				     "issm%d", port->dev_num);
1043
	if (IS_ERR(port->sm_dev))
L
Linus Torvalds 已提交
1044 1045
		goto err_sm_cdev;

1046 1047 1048 1049
	if (device_create_file(port->sm_dev, &dev_attr_ibdev))
		goto err_sm_dev;
	if (device_create_file(port->sm_dev, &dev_attr_port))
		goto err_sm_dev;
L
Linus Torvalds 已提交
1050

1051 1052 1053 1054
	spin_lock(&port_lock);
	umad_port[port->dev_num] = port;
	spin_unlock(&port_lock);

L
Linus Torvalds 已提交
1055 1056
	return 0;

1057 1058
err_sm_dev:
	device_destroy(umad_class, port->sm_cdev->dev);
L
Linus Torvalds 已提交
1059 1060

err_sm_cdev:
1061
	cdev_del(port->sm_cdev);
L
Linus Torvalds 已提交
1062

1063 1064
err_dev:
	device_destroy(umad_class, port->cdev->dev);
L
Linus Torvalds 已提交
1065 1066

err_cdev:
1067
	cdev_del(port->cdev);
1068
	clear_bit(port->dev_num, dev_map);
L
Linus Torvalds 已提交
1069 1070 1071 1072

	return -1;
}

1073 1074
static void ib_umad_kill_port(struct ib_umad_port *port)
{
1075
	struct ib_umad_file *file;
1076
	int already_dead;
1077 1078
	int id;

1079 1080
	dev_set_drvdata(port->dev,    NULL);
	dev_set_drvdata(port->sm_dev, NULL);
1081

1082 1083
	device_destroy(umad_class, port->cdev->dev);
	device_destroy(umad_class, port->sm_cdev->dev);
1084

1085 1086
	cdev_del(port->cdev);
	cdev_del(port->sm_cdev);
1087 1088 1089 1090 1091

	spin_lock(&port_lock);
	umad_port[port->dev_num] = NULL;
	spin_unlock(&port_lock);

1092
	mutex_lock(&port->file_mutex);
1093 1094 1095

	port->ib_dev = NULL;

1096 1097 1098
	list_for_each_entry(file, &port->file_list, port_list) {
		mutex_lock(&file->mutex);
		already_dead = file->agents_dead;
1099
		file->agents_dead = 1;
1100
		mutex_unlock(&file->mutex);
1101 1102 1103 1104 1105

		for (id = 0; id < IB_UMAD_MAX_AGENTS; ++id)
			if (file->agent[id])
				ib_unregister_mad_agent(file->agent[id]);
	}
1106

1107
	mutex_unlock(&port->file_mutex);
1108

1109 1110 1111
	clear_bit(port->dev_num, dev_map);
}

L
Linus Torvalds 已提交
1112 1113 1114 1115 1116
static void ib_umad_add_one(struct ib_device *device)
{
	struct ib_umad_device *umad_dev;
	int s, e, i;

T
Tom Tucker 已提交
1117 1118 1119 1120
	if (rdma_node_get_transport(device->node_type) != RDMA_TRANSPORT_IB)
		return;

	if (device->node_type == RDMA_NODE_IB_SWITCH)
L
Linus Torvalds 已提交
1121 1122 1123 1124 1125 1126
		s = e = 0;
	else {
		s = 1;
		e = device->phys_port_cnt;
	}

S
Sean Hefty 已提交
1127
	umad_dev = kzalloc(sizeof *umad_dev +
L
Linus Torvalds 已提交
1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149
			   (e - s + 1) * sizeof (struct ib_umad_port),
			   GFP_KERNEL);
	if (!umad_dev)
		return;

	kref_init(&umad_dev->ref);

	umad_dev->start_port = s;
	umad_dev->end_port   = e;

	for (i = s; i <= e; ++i) {
		umad_dev->port[i - s].umad_dev = umad_dev;

		if (ib_umad_init_port(device, i, &umad_dev->port[i - s]))
			goto err;
	}

	ib_set_client_data(device, &umad_client, umad_dev);

	return;

err:
1150
	while (--i >= s)
M
Michael S. Tsirkin 已提交
1151
		ib_umad_kill_port(&umad_dev->port[i - s]);
L
Linus Torvalds 已提交
1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163

	kref_put(&umad_dev->ref, ib_umad_release_dev);
}

static void ib_umad_remove_one(struct ib_device *device)
{
	struct ib_umad_device *umad_dev = ib_get_client_data(device, &umad_client);
	int i;

	if (!umad_dev)
		return;

1164 1165
	for (i = 0; i <= umad_dev->end_port - umad_dev->start_port; ++i)
		ib_umad_kill_port(&umad_dev->port[i]);
L
Linus Torvalds 已提交
1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180

	kref_put(&umad_dev->ref, ib_umad_release_dev);
}

static int __init ib_umad_init(void)
{
	int ret;

	ret = register_chrdev_region(base_dev, IB_UMAD_MAX_PORTS * 2,
				     "infiniband_mad");
	if (ret) {
		printk(KERN_ERR "user_mad: couldn't register device number\n");
		goto out;
	}

1181 1182 1183
	umad_class = class_create(THIS_MODULE, "infiniband_mad");
	if (IS_ERR(umad_class)) {
		ret = PTR_ERR(umad_class);
L
Linus Torvalds 已提交
1184 1185 1186 1187
		printk(KERN_ERR "user_mad: couldn't create class infiniband_mad\n");
		goto out_chrdev;
	}

1188
	ret = class_create_file(umad_class, &class_attr_abi_version);
L
Linus Torvalds 已提交
1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202
	if (ret) {
		printk(KERN_ERR "user_mad: couldn't create abi_version attribute\n");
		goto out_class;
	}

	ret = ib_register_client(&umad_client);
	if (ret) {
		printk(KERN_ERR "user_mad: couldn't register ib_umad client\n");
		goto out_class;
	}

	return 0;

out_class:
1203
	class_destroy(umad_class);
L
Linus Torvalds 已提交
1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214

out_chrdev:
	unregister_chrdev_region(base_dev, IB_UMAD_MAX_PORTS * 2);

out:
	return ret;
}

static void __exit ib_umad_cleanup(void)
{
	ib_unregister_client(&umad_client);
1215
	class_destroy(umad_class);
L
Linus Torvalds 已提交
1216 1217 1218 1219 1220
	unregister_chrdev_region(base_dev, IB_UMAD_MAX_PORTS * 2);
}

module_init(ib_umad_init);
module_exit(ib_umad_cleanup);